текстовый.Событие TextChanged дважды срабатывает в эмуляторе Windows Phone 7

у меня есть очень простое тестовое приложение, чтобы играть с Windows Phone 7. Я только что добавил TextBox и TextBlock к стандартному шаблону пользовательского интерфейса. Единственным пользовательским кодом является следующий:

public partial class MainPage : PhoneApplicationPage
{
    public MainPage()
    {
        InitializeComponent();
    }

    private int counter = 0;

    private void TextBoxChanged(object sender, TextChangedEventArgs e)
    {
        textBlock1.Text += "Text changed " + (counter++) + "rn";
    }
}

на TextBox.TextChanged событие подключено к TextBoxChanged в XAML:

<TextBox Height="72" HorizontalAlignment="Left" Margin="6,37,0,0"
         Name="textBox1" Text="" VerticalAlignment="Top"
         Width="460" TextChanged="TextBoxChanged" />

однако каждый раз, когда я нажимаю клавишу при запуске в эмуляторе (либо на экранной клавиатуре, либо на физической, нажав паузу, чтобы включить последнюю), он увеличивает счетчик дважды, отображая две строки в TextBlock. Все, что я пробовал, показывает, что мероприятие действительно стреляет дважды, и я понятия не имею, почему. Я проверил, что он подписывается только один раз - если я отпишусь в MainPage конструктор, ничего не происходит вообще (с текстовым блоком) при изменении текста.

я пробовал эквивалентный код в обычном приложении Silverlight, и это не происходит. У меня нет физического телефона, чтобы воспроизвести это на данный момент. Я не нашел запись об этом является известной проблемой в Windows Phone 7.

кто-нибудь может объяснить что я делаю неправильно, или я должен сообщить об этом как об ошибке?

EDIT: чтобы уменьшить возможность этого до двух текстовых элементов управления, я попытался удалить TextBlock полностью и изменение метода TextBoxChanged на просто инкремент counter. Затем я запустил эмулятор, набрал 10 букв и затем поставить точку останова на counter++; линия (просто чтобы избавиться от любой возможности, что взлом в отладчик вызывает проблемы) - и это показывает counter как 20.

EDIT: у меня сейчас спросил на форуме Windows Phone 7... посмотрим, что получится.

10 ответов


причина TextChanged событие срабатывает дважды в WP7 является побочным эффектом того, как TextBox был шаблоном для Metro look.

если вы измените TextBox шаблон в Blend вы увидите, что он содержит вторичный TextBox для инвалидов/только чтение. Это приводит к тому, что в качестве побочного эффекта событие срабатывает дважды.

вы можете изменить шаблон, чтобы удалить лишний TextBox (и связанные состояния), если вам не нужны эти состояния или измените шаблон на достигните другого взгляда в отключенном / только для чтения состоянии, без использования вторичного TextBox.

С этим событие будет срабатывать только один раз.


Я бы пошел на ошибку, в основном потому, что если вы положите KeyDown и KeyUp события там, это показывает, что они запускаются только один раз (каждый из них), но TextBoxChanged событие запускается дважды


Это звучит как ошибка для меня. В качестве обходного пути, вы всегда можете использовать . Существует перегрузка, которая позволяет указать отдельный ключ.

этот метод расширения возвращает наблюдаемое событие TextChanged, но пропускает последовательные дубликаты:

public static IObservable<IEvent<TextChangedEventArgs>> GetTextChanged(
    this TextBox tb)
{
    return Observable.FromEvent<TextChangedEventArgs>(
               h => textBox1.TextChanged += h, 
               h => textBox1.TextChanged -= h
           )
           .DistinctUntilChanged(t => t.Text);
}

после исправления ошибки вы можете просто удалить DistinctUntilChanged линии.


приятно! Я нашел этот вопрос, ища связанную проблему, а также нашел эту раздражающую вещь в моем коде. Двойное событие съедает больше ресурсов процессора в моем случае. Итак, я исправил текстовое поле фильтра в реальном времени с помощью этого решения:

private string filterText = String.Empty;

private void SearchBoxUpdated( object sender, TextChangedEventArgs e )
{
    if ( filterText != filterTextBox.Text )
    {
        // one call per change
        filterText = filterTextBox.Text;
        ...
    }

}

Я считаю, что это всегда было ошибкой в компактной структуре. Должно быть, его перенесли в РГ7.


конечно, похоже на ошибку для меня, если вы пытаетесь вызвать событие каждый раз, когда текст изменяется, вы можете попробовать использовать двустороннюю привязку вместо этого, к сожалению, это не вызовет события изменения нажатия клавиши (только когда поле теряет фокус). Вот обходной путь, если он вам нужен:

        this.textBox1.TextChanged -= this.TextBoxChanged;
        textBlock1.Text += "Text changed " + (counter++) + "\r\n";
        this.textBox1.TextChanged += this.TextBoxChanged;

отказ от ответственности-я не знаком с нюансами xaml, и я знаю, что это звучит нелогично... но в любом случае-моя первая мысль-попытаться пройти как просто eventargs, а не textchangedeventargs. Не имеет смысла, но может быть это поможет? Похоже, когда я видел двойные стрельбы, как это раньше, это либо из-за ошибки, либо из-за того, что вызовы обработчика событий 2 add происходят за кулисами... Я не уверен, что именно?

Если вам нужно быстро и грязно, снова, я не будучи опытным с XAML - мой следующий шаг будет просто пропустить XAML для этого текстового поля в качестве быстрого решения проблемы... сделайте это текстовое поле полностью на c#, пока вы не сможете определить ошибку или сложный код... то есть, если вам нужно временное решение.


Я не думаю, что это ошибка .. При назначении значения свойству text внутри события textchanged изменяется значение textbox, которое снова вызовет событие text changed ..

попробуйте это в приложении Windows Forms, вы можете получить сообщение об ошибке

"необработанное исключение типа 'System.StackOverflowException ' произошло в системе.Окна.Формы.DLL-файл"


Стефанвик прав, подумайте об использовании этого шаблона

<Application.Resources>
        <ControlTemplate x:Key="PhoneDisabledTextBoxTemplate" TargetType="TextBox">
            <ContentControl x:Name="ContentElement" BorderThickness="0" HorizontalContentAlignment="Stretch" Margin="{StaticResource PhoneTextBoxInnerMargin}" Padding="{TemplateBinding Padding}" VerticalContentAlignment="Stretch"/>
        </ControlTemplate>
        <Style x:Key="TextBoxStyle1" TargetType="TextBox">
            <Setter Property="FontFamily" Value="{StaticResource PhoneFontFamilyNormal}"/>
            <Setter Property="FontSize" Value="{StaticResource PhoneFontSizeMediumLarge}"/>
            <Setter Property="Background" Value="{StaticResource PhoneTextBoxBrush}"/>
            <Setter Property="Foreground" Value="{StaticResource PhoneTextBoxForegroundBrush}"/>
            <Setter Property="BorderBrush" Value="{StaticResource PhoneTextBoxBrush}"/>
            <Setter Property="SelectionBackground" Value="{StaticResource PhoneAccentBrush}"/>
            <Setter Property="SelectionForeground" Value="{StaticResource PhoneTextBoxSelectionForegroundBrush}"/>
            <Setter Property="BorderThickness" Value="{StaticResource PhoneBorderThickness}"/>
            <Setter Property="Padding" Value="2"/>
            <Setter Property="Template">
                <Setter.Value>
                    <ControlTemplate TargetType="TextBox">
                        <Grid Background="Transparent">
                            <VisualStateManager.VisualStateGroups>
                                <VisualStateGroup x:Name="CommonStates" ec:ExtendedVisualStateManager.UseFluidLayout="True">
                                    <VisualState x:Name="Normal"/>
                                    <VisualState x:Name="MouseOver"/>
                                    <VisualState x:Name="Disabled">
                                        <Storyboard>
                                            <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="Visibility" Storyboard.TargetName="EnabledBorder">
                                                <DiscreteObjectKeyFrame KeyTime="0">
                                                    <DiscreteObjectKeyFrame.Value>
                                                        <Visibility>Collapsed</Visibility>
                                                    </DiscreteObjectKeyFrame.Value>
                                                </DiscreteObjectKeyFrame>
                                            </ObjectAnimationUsingKeyFrames>
                                        </Storyboard>
                                    </VisualState>
                                    <VisualState x:Name="ReadOnly">
                                        <Storyboard>
                                            <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="Visibility" Storyboard.TargetName="EnabledBorder">
                                                <DiscreteObjectKeyFrame KeyTime="0">
                                                    <DiscreteObjectKeyFrame.Value>
                                                        <Visibility>Collapsed</Visibility>
                                                    </DiscreteObjectKeyFrame.Value>
                                                </DiscreteObjectKeyFrame>
                                            </ObjectAnimationUsingKeyFrames>
                                        </Storyboard>
                                    </VisualState>
                                </VisualStateGroup>
                                <VisualStateGroup x:Name="FocusStates">
                                    <VisualState x:Name="Focused">
                                        <Storyboard>
                                            <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="Background" Storyboard.TargetName="EnabledBorder">
                                                <DiscreteObjectKeyFrame KeyTime="0" Value="{StaticResource PhoneTextBoxEditBackgroundBrush}"/>
                                            </ObjectAnimationUsingKeyFrames>
                                            <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty="BorderBrush" Storyboard.TargetName="EnabledBorder">
                                                <DiscreteObjectKeyFrame KeyTime="0" Value="{StaticResource PhoneTextBoxEditBorderBrush}"/>
                                            </ObjectAnimationUsingKeyFrames>
                                        </Storyboard>
                                    </VisualState>
                                    <VisualState x:Name="Unfocused"/>
                                </VisualStateGroup>
                            </VisualStateManager.VisualStateGroups>
                            <VisualStateManager.CustomVisualStateManager>
                                <ec:ExtendedVisualStateManager/>
                            </VisualStateManager.CustomVisualStateManager>
                            <Border x:Name="EnabledBorder" BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}" Background="{TemplateBinding Background}" Margin="{StaticResource PhoneTouchTargetOverhang}">
                                <ContentControl x:Name="ContentElement" BorderThickness="0" HorizontalContentAlignment="Stretch" Margin="{StaticResource PhoneTextBoxInnerMargin}" Padding="{TemplateBinding Padding}" VerticalContentAlignment="Stretch"/>
                            </Border>
                        </Grid>
                    </ControlTemplate>
                </Setter.Value>
            </Setter>
        </Style>
    </Application.Resources>

Это старая тема, но вместо шаблона изменения (который не работает для меня, я не вижу другого текстового поля с Blend), вы можете добавить boolean, чтобы проверить, выполнило ли событие функцию или нет.

boolean already = false;
private void Tweet_SizeChanged(object sender, EventArgs e)
{
    if (!already)
    {
        already = true;
        ...
    }
    else
    {
    already = false;
    }
}

Я знаю, что это не идеальный способ, но я думаю, что это простой способ сделать это. И это работает.