设置XAML属性会在运行时被覆盖吗?

时间:2010-09-24 21:29:46

标签: silverlight xaml controls dependency-properties controltemplate

当我使用以下行时,从Silverlight类库实现模板化控件;

  

<labelSliderControl:SliderControl x:Name="sldX" Label="X" Value="4" />

我在VS Disign视图中看到Slider得到的值为4.但是当我运行应用程序时,Slider从0开始。所以当我在XAML中更改属性时,我可以看到更改,但我也想要值在运行期间的4个。在这种情况下,depencyproperty是否会覆盖该值?

以下是Silverlight类库中的模板化控件;

<Style TargetType="local:SliderControl">
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="local:SliderControl">
                <Grid x:Name="LayoutRoot" Background="White">
                    <Grid.RowDefinitions>
                        <RowDefinition Height="Auto" />
                        <RowDefinition />
                        <RowDefinition Height="Auto" />
                    </Grid.RowDefinitions>
                    <sdk:Label x:Name="lblLabel" Content="" HorizontalContentAlignment="Center" />
                    <Slider x:Name="sldValue" Grid.Row="1" Orientation="Vertical" />
                    <TextBox x:Name="txtValue" Grid.Row="2" MaxWidth="50" MinWidth="50" Text="{Binding ElementName=sldValue, Path=Value, StringFormat=\{0:N2\}}" />
                </Grid>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

[TemplatePart(Name = "sldValue", Type = typeof(Slider)), TemplatePart(Name = "lblLabel", Type = typeof(Label))]
public class SliderControl : Control
{
    private Slider _sliderElement;

    private Slider SliderElement
    {
        get  { return _sliderElement; }
        set  {
            if (_sliderElement != null)
                _sliderElement.ValueChanged -= new RoutedPropertyChangedEventHandler<double>(sldValue_ValueChanged);;

            _sliderElement = value;

            if (_sliderElement != null)
                _sliderElement.ValueChanged += new RoutedPropertyChangedEventHandler<double>(sldValue_ValueChanged);
        }
    }

    public event EventHandler<ValueChangingEventArgs> ValueChanging;

    public static readonly DependencyProperty ValueProperty =
        DependencyProperty.Register("Value", typeof(double), typeof(SliderControl), new PropertyMetadata(0.0, ValuePropertyChanged));

    private static void ValuePropertyChanged(DependencyObject obj, DependencyPropertyChangedEventArgs args)
    {
        if (obj is SliderControl)
        {
            SliderControl sliderUserControl = (SliderControl)obj;
            if (sliderUserControl.SliderElement != null)
            {
                sliderUserControl.SliderElement.Value = (double)args.NewValue;
            }
        }
    }

    public SliderControl()
    {
        this.DefaultStyleKey = typeof(SliderControl);
    }

    private void sldValue_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
    {
        if (ValueChanging != null)
        {
            ValueChanging(this, new ValueChangingEventArgs(e.OldValue, e.NewValue));
        }
    }

    public override void OnApplyTemplate()
    {
        SliderElement = GetTemplateChild("sldValue") as Slider;
        base.OnApplyTemplate();
    }
}

2 个答案:

答案 0 :(得分:0)

注册DependencyProperty时,我没有看到声明的“Value”属性。

public double Value
{
   get { return (double)base.GetValue(ValueProperty); }
   set { base.SetValue(ValueProperty, value); }
}

然后将属性绑定到构造函数中的控件:

public SliderControl(){

   InitializeComponent();

   this.sldValue.SetBinding(Slider.ValueProperty, new Binding()
   {
      Source = this,
      Path = new PropertyPath("Value"),
      Mode = BindingMode.TwoWay
   });
}

答案 1 :(得分:0)

我得到了它的工作!我只是将gmcalab的解决方案放在OnApplyTemplate中。 感谢gmcalab。

    public override void OnApplyTemplate()
    {
        SliderElement = GetTemplateChild("sldValue") as Slider;
        if (SliderElement != null)
        {
            SliderElement.SetBinding(Slider.ValueProperty, new Binding()
            {
                Source = this,
                Path = new PropertyPath("Value"),
                Mode = BindingMode.TwoWay
            });
        }
        base.OnApplyTemplate();
    }