// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
namespace System.Reactive.Disposables
{
///
/// Represents a disposable resource whose underlying disposable resource can be replaced by another disposable resource, causing automatic disposal of the previous underlying disposable resource.
///
public sealed class SerialDisposable : ICancelable
{
private readonly object _gate = new object();
private IDisposable _current;
private bool _disposed;
///
/// Initializes a new instance of the class.
///
public SerialDisposable()
{
}
///
/// Gets a value that indicates whether the object is disposed.
///
public bool IsDisposed
{
get
{
lock (_gate)
{
return _disposed;
}
}
}
///
/// Gets or sets the underlying disposable.
///
/// If the SerialDisposable has already been disposed, assignment to this property causes immediate disposal of the given disposable object. Assigning this property disposes the previous disposable object.
public IDisposable Disposable
{
get
{
return _current;
}
set
{
var shouldDispose = false;
var old = default(IDisposable);
lock (_gate)
{
shouldDispose = _disposed;
if (!shouldDispose)
{
old = _current;
_current = value;
}
}
if (old != null)
old.Dispose();
if (shouldDispose && value != null)
value.Dispose();
}
}
///
/// Disposes the underlying disposable as well as all future replacements.
///
public void Dispose()
{
var old = default(IDisposable);
lock (_gate)
{
if (!_disposed)
{
_disposed = true;
old = _current;
_current = null;
}
}
if (old != null)
old.Dispose();
}
}
}