Как использовать TextBox.Watermark в Silverlight 4? - PullRequest
8 голосов
/ 03 апреля 2012

Просматривая документацию MSDN, вы можете встретить этот драгоценный камень: TextBox.Watermark.

«Круто! Я давно хотел встроить водяные знаки в свои текстовые поля! Это здорово, позвольте мне продолжить и установить это в XAML!»

<TextBox Watermark="This is my watermark" Margin="20"></TextBox>

К сожалению, если вы запустите это, вы не получите то, что ожидаете:

enter image description here

И деталь: enter image description here

Что это? Хорошо, посмотрите внимательно на документацию MSDN: enter image description here

Это верно. Он поддерживается в Silverlight 4, но в нем также указано «Не использовать в приложении Silverlight 4». Если вы используете его, вы получите исключение System.NotImplemented. Чтобы проверить, вот код для свойства, декомпилированного через Reflector:

[EditorBrowsable(EditorBrowsableState.Never)]
public object Watermark
{
get
{
StubHelper.ThrowIfNotInDesignMode();
return base.GetValue(WatermarkProperty);
}
set
{
StubHelper.ThrowIfNotInDesignMode();
base.SetValue(WatermarkProperty, value);
}
}

Вот оно, оно выдает исключение каждый раз, когда не находится в режиме разработки. Это не имеет смысла, верно? Зачем Microsoft это делать?

К сожалению, я пока не нашел однозначного ответа, однако, если бы мне пришлось догадываться, это потому, что Microsoft планирует реализовать поведение Watermark для элемента управления TextBox в будущей версии (возможно, v5) и хотела эффективно зарезервировать это свойство, чтобы сторонние создатели элементов управления не наследуют TextBox и не создают свое собственное свойство Watermark. Я знаю по крайней мере одного поставщика элементов управления, ComponentOne, который имеет элемент управления, который наследуется от TextBox и предоставляет свойство Watermark. Мне кажется, что это способ Microsoft отговаривать людей использовать это имя свойства на своих собственных подклассах TextBox.

Ответы [ 5 ]

14 голосов
/ 05 апреля 2012

Создать один проект библиотеки классов. Добавьте файл класса, используя следующий код ..... После этого добавьте In this dll In Your Project.

public class WatermarkTextBox : TextBox 
{ 
    private bool displayWatermark = true; 
    private bool hasFocus = false; 
     public WatermarkTextBox() 
    { 
        this.GotFocus += new RoutedEventHandler(WatermarkTextBox_GotFocus); 
        this.LostFocus += new RoutedEventHandler(WatermarkTextBox_LostFocus); 
        this.TextChanged += new TextChangedEventHandler(WatermarkTextBox_TextChanged); 
        this.Unloaded += new RoutedEventHandler(WatermarkTextBox_Unloaded); 
    } 

    private void WatermarkTextBox_TextChanged(object sender, TextChangedEventArgs e) 
    { 
        if (!hasFocus && Text == "") 
        { 
            setMode(true); 
            displayWatermark = true; 
            this.Text = Watermark; 
        } 
    } 

    private void WatermarkTextBox_Unloaded(object sender, RoutedEventArgs e) 
    { 
        this.GotFocus -= WatermarkTextBox_GotFocus; 
        this.LostFocus -= WatermarkTextBox_LostFocus; 
        this.Unloaded -= WatermarkTextBox_Unloaded; 
        this.TextChanged -= WatermarkTextBox_TextChanged; 
    } 

    private void WatermarkTextBox_GotFocus(object sender, RoutedEventArgs e) 
    { 
        hasFocus = true; 
        if (displayWatermark) 
        { 
            setMode(false); 
            this.Text = ""; 
        } 
    } 
    private void WatermarkTextBox_LostFocus(object sender, RoutedEventArgs e) 
    { 
        hasFocus = false; 
        if (this.Text == "") 
        { 
            displayWatermark = true; 
            setMode(true); 
            this.Text = Watermark; 
        } 
        else 
        { 
            displayWatermark = false; 
        } 
    } 
    private void setMode(bool watermarkStyle) 
    { 
        if (watermarkStyle) 
        { 
            this.FontStyle = FontStyles.Italic; 
            this.Foreground = new SolidColorBrush(Colors.Gray); 
        } 
        else 
        { 
            this.FontStyle = FontStyles.Normal; 
            this.Foreground = new SolidColorBrush(Colors.Black); 
        } 
    } 
    public new string Watermark 
    { 
        get { return GetValue(WatermarkProperty) as string; } 
        set { SetValue(WatermarkProperty, value); } 
    } 
    public static new readonly DependencyProperty WatermarkProperty = 
        DependencyProperty.Register("Watermark", typeof(string), typeof(WatermarkTextBox), new PropertyMetadata(watermarkPropertyChanged)); 
    private static void watermarkPropertyChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e) 
    { 
        WatermarkTextBox textBox = obj as WatermarkTextBox; 
        if (textBox.displayWatermark) 
        { 
            textBox.Text = e.NewValue.ToString(); 
            textBox.setMode(true); 
        } 
    } 

XAML:

  xmlns:watertext="clr-namespace:SilverlightClassLibrary1;assembly=SilverlightClassLibrary1"


    <watertext:WatermarkTextBox Watermark="WElcome" Margin="150,115,120,166"></watertext:WatermarkTextBox>
3 голосов
/ 05 сентября 2012

Я немного пересмотрел решение @mani kandan, чтобы исправить ошибку времени разработки, если свойство Watermark не установлено. Также добавлено логическое свойство HasValue, чтобы можно было легко проверять, ввел ли пользователь текст в TextBox, и, наконец, изменилось, чтобы обрабатывать все пропуски как не входящие (то есть продолжать отображать водяной знак).

Пересмотренный код:

public class WatermarkTextBox : TextBox
{

    private bool displayWatermark = true;
    private bool hasFocus = false;

    public WatermarkTextBox()
    {
        this.GotFocus += new RoutedEventHandler(WatermarkTextBox_GotFocus);
        this.LostFocus += new RoutedEventHandler(WatermarkTextBox_LostFocus);
        this.TextChanged += new TextChangedEventHandler(WatermarkTextBox_TextChanged);
        this.Unloaded += new RoutedEventHandler(WatermarkTextBox_Unloaded);
    }

    private void WatermarkTextBox_TextChanged(object sender, TextChangedEventArgs e)
    {
        if (!hasFocus && string.IsNullOrWhiteSpace(this.Text))
        {
            setMode(true);
            displayWatermark = true;
            // avoid design-time error if Watermark not specified
            this.Text = (Watermark == null ? string.Empty : Watermark);
        }
    }

    private void WatermarkTextBox_Unloaded(object sender, RoutedEventArgs e)
    {
        this.GotFocus -= WatermarkTextBox_GotFocus;
        this.LostFocus -= WatermarkTextBox_LostFocus;
        this.Unloaded -= WatermarkTextBox_Unloaded;
        this.TextChanged -= WatermarkTextBox_TextChanged;
    }

    private void WatermarkTextBox_GotFocus(object sender, RoutedEventArgs e)
    {
        hasFocus = true;
        if (displayWatermark)
        {
            setMode(false);
            this.Text = "";
        }
    }

    private void WatermarkTextBox_LostFocus(object sender, RoutedEventArgs e)
    {
        hasFocus = false;
        if (string.IsNullOrWhiteSpace(this.Text))
        {
            displayWatermark = true;
            setMode(true);
            this.Text = (Watermark == null ? string.Empty : Watermark);
        }
        else
        {
            displayWatermark = false;
        }
    }

    private void setMode(bool watermarkStyle)
    {
        if (watermarkStyle)
        {
            this.FontStyle = FontStyles.Italic;
            this.Foreground = new SolidColorBrush(Colors.Gray);
        }
        else
        {
            this.FontStyle = FontStyles.Normal;
            this.Foreground = new SolidColorBrush(Colors.Black);
        }
    }

    public new string Watermark
    {
        get { return GetValue(WatermarkProperty) as string; }
        set { SetValue(WatermarkProperty, value); }
    }

    public static new readonly DependencyProperty WatermarkProperty =
        DependencyProperty.Register("Watermark", typeof(string), typeof(WatermarkTextBox), new PropertyMetadata(watermarkPropertyChanged));
    private static void watermarkPropertyChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e)
    {
        WatermarkTextBox textBox = obj as WatermarkTextBox;
        if (textBox.displayWatermark)
        {
            textBox.Text =  e.NewValue.ToString();
            textBox.setMode(true);
        }
    }

    public bool HasValue
    {
        get 
        {
            // if watermark has been specified, then compare to text value to determine if text set by user,
            // otherwise check to see if empty or whitespace.
            if (this.Watermark != null)
                return this.Watermark != this.Text;
            else
                return !string.IsNullOrWhiteSpace(this.Text);
        }
    }

}
2 голосов
/ 30 ноября 2012

Проверьте это на основе поведения

namespace MyNamespace
{
    public class WatermarkBehavior : Behavior<TextBox>
    {
        public String Watermark
        {
            get { return this.GetValue(WatermarkProperty) as String; }
            set { this.SetValue(WatermarkProperty, value); }
        }

        public static readonly DependencyProperty WatermarkProperty = DependencyProperty.Register("Watermark", typeof(String), typeof(WatermarkBehavior), new PropertyMetadata("", new PropertyChangedCallback(OnWatermarkChanged)));

        private static void OnWatermarkChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            var behavior = d as WatermarkBehavior;
            if (!String.IsNullOrWhiteSpace(e.NewValue as String))
            {
                behavior.SetWatermarkIfNeeded();
            }
        }

        protected override void OnAttached()
        {
            base.OnAttached();
            this.AssociatedObject.GotFocus += GotFocus;
            this.AssociatedObject.LostFocus += LostFocus;
        }

        protected override void OnDetaching()
        {
            base.OnDetaching();
            this.AssociatedObject.GotFocus -= GotFocus;
            this.AssociatedObject.LostFocus -= LostFocus;
        }

        private void GotFocus(object sender, RoutedEventArgs e)
        {
            if (this.AssociatedObject.Text == this.Watermark)
            {
                this.AssociatedObject.Text = String.Empty;
                this.AssociatedObject.Foreground = new SolidColorBrush(Colors.Black);
            }
        }

        private void LostFocus(object sender, RoutedEventArgs e)
        {
            this.SetWatermarkIfNeeded();
        }

        private void SetWatermarkIfNeeded()
        {
            if (String.IsNullOrWhiteSpace(this.AssociatedObject.Text))
            {
                this.SetWatermark();
            }
        }

        private void SetWatermark()
        {
            this.AssociatedObject.Text = this.Watermark;
            this.AssociatedObject.Foreground = new SolidColorBrush(Colors.Gray);
        }
    }
}

XAML

<UserControl x:Class="GreenField.Targeting.Controls.BasketTargets.EditorView"
  xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"  
  xmlns:local="clr-namespace:MyNamespace"
  xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity"
  mc:Ignorable="d" d:DesignHeight="300" d:DesignWidth="400">
  <TextBox Text="{Binding Text}">
    <i:Interaction.Behaviors>
      <local:WatermarkBehavior Watermark="{Binding Watermark}" />
    </i:Interaction.Behaviors>
  </TextBox>
</UserControl>
2 голосов
/ 11 октября 2012

Создать один проект библиотеки классов. Добавьте файл класса, используя следующий код

using System;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Ink;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;

namespace Project.Controls
{
    public class WatermarkEditBox : TextBox
    {

        TextBlock lbl = new TextBlock()
        {
            IsHitTestVisible = false,
            Foreground = new SolidColorBrush(Colors.LightGray),
            VerticalAlignment = VerticalAlignment.Center,
            HorizontalAlignment = HorizontalAlignment.Left,
            Margin = new Thickness(3,0,0,0)
        };
        public string WatermarkText { get { return lbl.Text; } set { lbl.Text = value; } }

        public WatermarkEditBox()
        {
            this.Loaded += WatermarkEditBox_Loaded;
        }

        void WatermarkEditBox_Loaded(object sender, RoutedEventArgs e)
        {
            this.UpdateLayout();
            Grid g = GetObjectOfType<Grid>(this, "RootElement");
            if (g != null)
            {
                g.Children.Add(lbl);
            }
            this.TextChanged += WatermarkEditBox_TextChanged;
        }

        void WatermarkEditBox_TextChanged(object sender, TextChangedEventArgs e)
        {
            if (this.Text.Length == 0)
                lbl.Visibility = System.Windows.Visibility.Visible;
            else
                lbl.Visibility = System.Windows.Visibility.Collapsed;
        }

        public TObject GetObjectOfType<TObject>(DependencyObject parent, string name) where TObject : DependencyObject
        {
            int count = VisualTreeHelper.GetChildrenCount(parent);
            for (int i = 0; i < count; ++i)
            {
                DependencyObject child = VisualTreeHelper.GetChild(parent, i);
                if (child is TObject && child.GetValue(NameProperty).ToString() == name)
                {
                    return child as TObject;
                }
                else
                {
                    TObject obj = GetObjectOfType<TObject>(child, name);
                    if (obj != null)
                    {
                        return obj;
                    }
                }
            }

            return null;
        }

    }
}

XAML:

xmlns:Controls="clr-namespace:Project.Controls"

<Controls:WatermarkEditBox WatermarkText="фильтр" Width="100"/>
2 голосов
/ 10 апреля 2012

Ну, вы можете успешно использовать его в Silverlight 5

Попробуйте эту ссылку: TextBox.Watermark

Я успешно использую WatermarkTextBox в приложении Silverlight 5 MVVM.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...