// -----------------------------------------------------------------------
//
// Copyright 2013 Tricycle. All rights reserved.
//
// -----------------------------------------------------------------------
namespace Perspex
{
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Reactive;
using System.Reactive.Disposables;
using System.Reactive.Linq;
using System.Reflection;
using Splat;
///
/// An object with support.
///
///
/// This class is analogous to DependencyObject in WPF.
///
public class PerspexObject : IEnableLogger
{
///
/// 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/bindings on this object.
///
private Dictionary values =
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;
}
}
}
}
///
/// Defers property updates due to style changes until
/// is called.
///
public void BeginDeferStyleChanges()
{
foreach (PriorityValue v in this.values.Values)
{
v.BeginDeferStyleChanges();
}
this.Log().Debug(string.Format(
"Defer style changes on {0} (#{1:x8})",
this.GetType().Name,
this.GetHashCode()));
}
///
/// Ends the defer of property updates due to style changes initiated by a previous call
/// to .
///
public void EndDeferStyleChanges()
{
foreach (PriorityValue v in this.values.Values)
{
v.EndDeferStyleChanges();
}
this.Log().Debug(string.Format(
"End defer style changes on {0} (#{1:x8})",
this.GetType().Name,
this.GetHashCode()));
}
///
/// 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);
PriorityValue value;
if (this.values.TryGetValue(property, out value))
{
value.ClearLocalBinding();
this.Log().Debug(string.Format(
"Cleared binding on {0}.{1} (#{2:x8})",
this.GetType().Name,
property.Name,
this.GetHashCode()));
}
}
///
/// Clears a value, including its binding.
///
/// The property.
public void ClearValue(PerspexProperty property)
{
Contract.Requires(property != null);
this.SetValue(property, PerspexProperty.UnsetValue);
}
///
/// Gets an observable for a .
///
///
///
public IObservable