// -----------------------------------------------------------------------
//
// Copyright 2013 Tricycle. All rights reserved.
//
// -----------------------------------------------------------------------
namespace Perspex
{
using System;
using System.Collections.Generic;
using System.Diagnostics.Contracts;
using System.Linq;
using System.Linq.Expressions;
using System.Reactive;
using System.Reactive.Linq;
using System.Reflection;
///
/// An object with support.
///
///
/// This class is analogous to DependencyObject in WPF.
///
public class PerspexObject
{
///
/// The registered properties by type.
///
private static Dictionary> registered =
new Dictionary>();
///
/// The parent object that inherited values are inherited from.
///
private PerspexObject inheritanceParent;
///
/// The set values on this object.
///
private Dictionary values =
new Dictionary();
///
/// The current bindings on this object.
///
private Dictionary bindings =
new Dictionary();
///
/// Raised when a value changes on this object/
///
public event EventHandler PropertyChanged;
///
/// Gets or sets the parent object that inherited values
/// are inherited from.
///
protected PerspexObject InheritanceParent
{
get
{
return this.inheritanceParent;
}
set
{
if (this.inheritanceParent != value)
{
if (this.inheritanceParent != null)
{
this.inheritanceParent.PropertyChanged -= this.ParentPropertyChanged;
}
var inherited = (from property in GetProperties(this.GetType())
where property.Inherits
select new
{
Property = property,
Value = this.GetValue(property),
}).ToList();
this.inheritanceParent = value;
foreach (var i in inherited)
{
object newValue = this.GetValue(i.Property);
if (!object.Equals(i.Value, newValue))
{
this.RaisePropertyChanged(i.Property, i.Value, newValue);
}
}
if (this.inheritanceParent != null)
{
this.inheritanceParent.PropertyChanged += this.ParentPropertyChanged;
}
}
}
}
///
/// Gets all s registered on a type.
///
/// The type.
/// A collection of definitions.
public static IEnumerable GetProperties(Type type)
{
Contract.Requires(type != null);
TypeInfo i = type.GetTypeInfo();
while (type != null)
{
List list;
if (registered.TryGetValue(type, out list))
{
foreach (PerspexProperty p in list)
{
yield return p;
}
}
type = type.GetTypeInfo().BaseType;
}
}
///
/// Registers a on a type.
///
/// The type.
/// The property.
///
/// You won't usually want to call this method directly, instead use the
/// method.
///
public static void Register(Type type, PerspexProperty property)
{
Contract.Requires(type != null);
Contract.Requires(property != null);
List list;
if (!registered.TryGetValue(type, out list))
{
list = new List();
registered.Add(type, list);
}
if (!list.Contains(property))
{
list.Add(property);
}
}
///
/// Clears a binding on a , leaving the last bound value in
/// place.
///
/// The property.
public void ClearBinding(PerspexProperty property)
{
Contract.Requires(property != null);
Binding binding;
if (this.bindings.TryGetValue(property, out binding))
{
binding.Dispose.Dispose();
this.bindings.Remove(property);
}
}
///
/// Clears a value, including its binding.
///
/// The property.
public void ClearValue(PerspexProperty property)
{
Contract.Requires(property != null);
this.ClearBinding(property);
this.values.Remove(property);
}
///
/// Clears a binding on a , returning the bound observable and
/// leaving the last bound value in place.
///
/// The property.
public IObservable