A cross-platform UI framework for .NET
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

278 lines
8.2 KiB

// -----------------------------------------------------------------------
// <copyright file="Control.cs" company="Steven Kirk">
// Copyright 2014 MIT Licence. See licence.md for more information.
// </copyright>
// -----------------------------------------------------------------------
namespace Perspex.Controls
{
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reactive.Linq;
using Perspex.Collections;
using Perspex.Controls.Primitives;
using Perspex.Controls.Shapes;
using Perspex.Input;
using Perspex.Interactivity;
using Perspex.Media;
using Perspex.Rendering;
using Perspex.Styling;
using Splat;
public class Control : InputElement, INamed, ILogical, IStyleable, IStyleHost
{
public static readonly PerspexProperty<object> DataContextProperty =
PerspexProperty.Register<Control, object>("DataContext", inherits: true);
public static readonly PerspexProperty<AdornerTemplate> FocusAdornerProperty =
PerspexProperty.Register<Control, AdornerTemplate>("FocusAdorner");
public static readonly PerspexProperty<Control> ParentProperty =
PerspexProperty.Register<Control, Control>("Parent");
public static readonly PerspexProperty<object> TagProperty =
PerspexProperty.Register<Control, object>("Tag");
public static readonly PerspexProperty<ITemplatedControl> TemplatedParentProperty =
PerspexProperty.Register<Control, ITemplatedControl>("TemplatedParent");
public static readonly RoutedEvent<RequestBringIntoViewEventArgs> RequestBringIntoViewEvent =
RoutedEvent.Register<Control, RequestBringIntoViewEventArgs>("RequestBringIntoView", RoutingStrategies.Bubble);
private static readonly IPerspexReadOnlyList<ILogical> EmptyChildren = new PerspexSingleItemList<ILogical>();
private Classes classes = new Classes();
private DataTemplates dataTemplates;
private Control focusAdorner;
private string id;
private Styles styles;
static Control()
{
Control.AffectsMeasure(Control.IsVisibleProperty);
PseudoClass(InputElement.IsEnabledCoreProperty, x => !x, ":disabled");
PseudoClass(InputElement.IsFocusedProperty, ":focus");
PseudoClass(InputElement.IsPointerOverProperty, ":pointerover");
}
public Classes Classes
{
get
{
return this.classes;
}
set
{
if (this.classes != value)
{
this.classes.Clear();
this.classes.Add(value);
}
}
}
public object DataContext
{
get { return this.GetValue(DataContextProperty); }
set { this.SetValue(DataContextProperty, value); }
}
public DataTemplates DataTemplates
{
get
{
if (this.dataTemplates == null)
{
this.dataTemplates = new DataTemplates();
}
return this.dataTemplates;
}
set
{
this.dataTemplates = value;
}
}
public string Name
{
get
{
return this.id;
}
set
{
if (this.id != null)
{
throw new InvalidOperationException("ID already set.");
}
if (((IVisual)this).VisualParent != null)
{
throw new InvalidOperationException("Cannot set ID : control already added to tree.");
}
this.id = value;
}
}
public Styles Styles
{
get
{
if (this.styles == null)
{
this.styles = new Styles();
}
return this.styles;
}
set
{
this.styles = value;
}
}
public Control Parent
{
get { return this.GetValue(ParentProperty); }
internal set { this.SetValue(ParentProperty, value); }
}
public object Tag
{
get { return this.GetValue(TagProperty); }
set { this.SetValue(TagProperty, value); }
}
public ITemplatedControl TemplatedParent
{
get { return this.GetValue(TemplatedParentProperty); }
internal set { this.SetValue(TemplatedParentProperty, value); }
}
ILogical ILogical.LogicalParent
{
get { return this.Parent; }
}
IPerspexReadOnlyList<ILogical> ILogical.LogicalChildren
{
get { return EmptyChildren; }
}
Type IStyleable.StyleKey
{
get { return this.GetType(); }
}
public void BringIntoView()
{
this.BringIntoView(new Rect(this.Bounds.Size));
}
public void BringIntoView(Rect rect)
{
var ev = new RequestBringIntoViewEventArgs
{
RoutedEvent = RequestBringIntoViewEvent,
TargetObject = this,
TargetRect = rect,
};
this.RaiseEvent(ev);
}
protected static void PseudoClass(PerspexProperty<bool> property, string className)
{
PseudoClass(property, x => x, className);
}
protected static void PseudoClass<T>(
PerspexProperty<T> property,
Func<T, bool> selector,
string className)
{
Contract.Requires<ArgumentNullException>(property != null);
Contract.Requires<ArgumentNullException>(selector != null);
Contract.Requires<ArgumentNullException>(className != null);
Contract.Requires<ArgumentNullException>(property != null);
if (string.IsNullOrWhiteSpace(className))
{
throw new ArgumentException("Cannot supply an empty className.");
}
Observable.Merge(property.Changed, property.Initialized)
.Subscribe(e =>
{
if (selector((T)e.NewValue))
{
((Control)e.Sender).Classes.Add(className);
}
else
{
((Control)e.Sender).Classes.Remove(className);
}
});
}
protected override void OnGotFocus(GotFocusEventArgs e)
{
base.OnGotFocus(e);
if (this.IsFocused && e.KeyboardNavigated)
{
var adornerLayer = AdornerLayer.GetAdornerLayer(this);
if (adornerLayer != null)
{
if (this.focusAdorner == null)
{
var template = this.GetValue(FocusAdornerProperty);
if (template != null)
{
this.focusAdorner = template.Build();
}
}
if (this.focusAdorner != null)
{
AdornerLayer.SetAdornedElement(this.focusAdorner, this);
adornerLayer.Children.Add(this.focusAdorner);
}
}
}
}
protected override void OnLostFocus(RoutedEventArgs e)
{
base.OnLostFocus(e);
if (this.focusAdorner != null)
{
var adornerLayer = this.focusAdorner.Parent as Panel;
adornerLayer.Children.Remove(this.focusAdorner);
this.focusAdorner = null;
}
}
protected override void OnAttachedToVisualTree(IRenderRoot root)
{
base.OnAttachedToVisualTree(root);
IStyler styler = Locator.Current.GetService<IStyler>();
styler.ApplyStyles(this);
}
}
}