3

我正在使用 Google 搜索结果实现创建自定义 ComboBox 控件,但我在使用箭头键时遇到了一些问题。

问题的详细信息
请观看此视频,因为我使用 UP 和 DOWN 箭头键演示问题,并查看输出面板。

http://screencast.com/t/DFkmlDKR

在视频中,我尝试搜索“a”(只是一个测试)并返回以“a”开头的 Google 搜索结果,然后我按下向下和向上箭头键。在输出面板中,当我按下这些键时,它会显示突出显示的项目。
在此处输入图像描述

然后我输入下一个字母“l”。现在,如果您在输出面板上注意到,现在选择为“null”,但我仍在按向上和向下箭头键。
在此处输入图像描述

当您再次使用鼠标指针将鼠标悬停在该项目上时,它将再次开始工作。
在此处输入图像描述

我被这个问题困扰了好几天,还没有找到解决方案。

我已经上传了这个控件的测试版本,所以你也可以使用它。这是GoogleSuggestionComboBox
在此处输入图像描述

我的目标是让向上和向下箭头键一直工作。

在这部分代码中, 在此处输入图像描述 我尝试添加

SelectedIndex = 0;

在 ForEach 语句之后,因此每次使用新结果重新填充集合时,它将选择第一个结果。不幸的是,它没有用。

您可以下载测试代码,以便您可以播放和测试问题。http://sdrv.ms/1eWV3Bc这里也是 ComboBox 的代码。

using GoogleSuggestionComboBox.Model;
using System.Collections.Generic;
using System.ComponentModel;
using System.Diagnostics;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;

namespace GoogleSuggestionComboBox
{
    public class ComboBoxExC : ComboBox
    {
        GoogleSuggest google;
        TextBox textbox;
        string _text = string.Empty;
        string _last_text = string.Empty;

        public ComboBoxExC()
        {
            if (DesignerProperties.GetIsInDesignMode(this))
            {

            }
            else
            {
                this.Loaded += ComboBoxExC_Loaded;

                google = new GoogleSuggest();
                google.OnGoogleSuggestAvailable += google_OnGoogleSuggestAvailable;

                // since we have OnSelectionChanged "disabled"
                // we need a way to know if the item in ComboBox is selected using mouse
                EventManager.RegisterClassHandler(typeof(ComboBoxItem), ComboBoxItem.MouseDownEvent, new MouseButtonEventHandler(OnItemMouseDown));
            }
        }

        void ComboBoxExC_Loaded(object sender, RoutedEventArgs e)
        {
            this.textbox = (TextBox)Template.FindName("PART_EditableTextBox", this);
        }

        private void OnItemMouseDown(object sender, MouseButtonEventArgs e)
        {
            var comboBoxItem = sender as ComboBoxItem;
            if (comboBoxItem != null && comboBoxItem.IsHighlighted)
            {
                Model_SuggestedQueries m = (Model_SuggestedQueries)comboBoxItem.Content;
                Go(m.Query);
            }
        }

        protected override void OnSelectionChanged(SelectionChangedEventArgs e)
        {
            // don't do anything so the .Text value won't change.

            //base.OnSelectionChanged(e);
        }

        protected override void OnPreviewKeyDown(KeyEventArgs e)
        {
            this.IsDropDownOpen = true;

            base.OnPreviewKeyDown(e);

            d("key: " + e.Key.ToString());

            if (this.SelectedItem != null)
            {
                Model_SuggestedQueries m = (Model_SuggestedQueries)this.SelectedItem;
                d("selected: " + m.Query);
            }
            else
            {
                d("null");
            }
        }

        protected override void OnPreviewKeyUp(KeyEventArgs e)
        {
            base.OnPreviewKeyUp(e);

            if (e.Key == Key.Enter)
            {
                if (this.SelectedItem == null)
                {
                    Go(this.Text);

                }
                else
                {
                    Model_SuggestedQueries m = (Model_SuggestedQueries)this.SelectedItem;
                    Go(m.Query);
                }
            }
            else
            {
                if (this.Text != this._last_text)
                {
                    google.LookForSuggestion(this.Text);

                    this._last_text = this.Text;
                }
            }
        }

        void google_OnGoogleSuggestAvailable(object sender, List<Model.Model_SuggestedQueries> suggestions)
        {
            this.Items.Clear();

            suggestions.ForEach((a) =>
            {
                this.Items.Add(a);
            });
        }

        void d(object a)
        {
            Debug.WriteLine(">>> " + a);
        }

        void Go(string query)
        {
            Process.Start("https://www.google.com.ph/search?q=" + query);

            // clear suggestions
            this.Items.Clear();
        }
    }
}

主窗口.xaml

<Window x:Class="GoogleSuggestionComboBox.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:l="clr-namespace:GoogleSuggestionComboBox"
        Title="MainWindow" Height="133" Width="261" WindowStartupLocation="CenterScreen"
        >
    <Grid>
        <StackPanel Margin="10">
            <TextBlock Text="Search" />
            <l:ComboBoxExC 
                IsEditable="True" 
                IsTextSearchEnabled="False"
                TextSearch.TextPath="Query"
                >
                <l:ComboBoxExC.ItemTemplate>
                    <DataTemplate>
                        <TextBlock Text="{Binding Query}" />
                    </DataTemplate>
                </l:ComboBoxExC.ItemTemplate>
            </l:ComboBoxExC>
        </StackPanel>
    </Grid>
</Window>

谢谢你,
杰森

4

2 回答 2

4

这是一个很老的问题,但我遇到了同样的问题。对我来说主要问题是在选择一个项目或清除选择(将其设置为 -1)之后,我无法再用箭头键浏览这些项目,直到我用鼠标悬停在它们上方。

经过几天的努力,我找到了解决这个问题的方法:

KeyboardNavigation.SetDirectionalNavigation( this, KeyboardNavigationMode.Cycle );

this是组合框,这一行在子类组合框的构造函数中。

于 2019-09-25T07:43:33.190 回答
-1

我不知道确切的问题,但这里有一个建议:

不要子类化 ComboBox - 你不需要。

只需将一个放在 MainForm 上,然后从后面的代码中使用它——这样使用它要容易得多。如果您需要更改不典型的组合框的工作方式,您只需要子类化。

我尝试下载测试项目,但它不是一个 zip 文件——它是一个 .7z,我无法打开。

此外,您的主要形式是一个窗口。这可能不是问题,但请尝试创建一个新项目并使用项目为您提供的 Main 表单。在调用 MainForm 的 App 中完成了一些接线,该 MainForm 将捕获异常等,这可能会为您提供更多关于问题所在的数据。

格雷格

于 2013-10-01T21:00:26.363 回答