// -----------------------------------------------------------------------
//
// Copyright 2013 MIT Licence. See licence.md for more information.
//
// -----------------------------------------------------------------------
namespace Perspex
{
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reactive.Linq;
using System.Reflection;
using Splat;
///
/// The priority of a binding.
///
public enum BindingPriority
{
///
/// A local value.
///
LocalValue,
///
/// A triggered style binding.
///
///
/// A style trigger is a selector such as .class which overrides a
/// binding. In this way, a basic control can have
/// for example a Background from the templated parent which changes when the
/// control has the :pointerover class.
///
StyleTrigger,
///
/// A binding to a property on the templated parent.
///
TemplatedParent,
///
/// A style binding.
///
Style,
///
/// The binding is uninitialized.
///
Unset = int.MaxValue,
}
///
/// 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;
}
}
}
}
///
/// Gets or sets the value of a .
///
/// The property.
public object this[PerspexProperty property]
{
get { return this.GetValue(property); }
set { this.SetValue(property, value); }
}
///
/// Gets or sets a binding for a .
///
/// The binding information.
public Binding this[Binding binding]
{
get
{
return new Binding
{
Mode = binding.Mode,
Priority = binding.Priority,
Property = binding.Property,
Source = this,
};
}
set
{
BindingMode mode = (binding.Mode == BindingMode.Default) ?
binding.Property.DefaultBindingMode :
binding.Mode;
switch (mode)
{
case BindingMode.Default:
case BindingMode.OneWay:
this.Bind(binding.Property, value.Source.GetObservable(value.Property), binding.Priority);
break;
case BindingMode.OneTime:
this.SetValue(binding.Property, value.Source.GetValue(value.Property));
break;
case BindingMode.OneWayToSource:
value.Source.Bind(value.Property, this.GetObservable(binding.Property), binding.Priority);
break;
case BindingMode.TwoWay:
this.BindTwoWay(binding.Property, value.Source, value.Property);
break;
}
}
}
///
/// 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 's local value.
///
/// The property.
public void ClearValue(PerspexProperty property)
{
Contract.Requires(property != null);
this.SetValue(property, PerspexProperty.UnsetValue);
}
///
/// Gets an observable for a .
///
/// The property.
/// An observable.
public IObservable