2

私のプロジェクトには、mainForm2 つの textBoxestxtUserNameと1 つtxtPasswordのボタンがある Form がありますbtnLogin

次のtxtUserNameプロパティを指定しました。

txtUserName プロパティ

AutoCompleteCustomSource - Collection
                            --> Administrator
                            --> Clerk
AutoCompleteMode   - Suggest
AutoCompleteSource - CustomSource

btnLogin_Click イベント

if (txtUserName.Text.Equals("Administrator") && txtPassword.Text.Equals("123"))
{
    //function to access admin features
}
else if (txtUserName.Text.Equals("Clerk") && txtPassword.Text.Equals("123"))
{
    //function to access clerk features
}
else
{
    MessageBox.Show("Please Enter correct details", "Login Error");
}

以下のコードに示すように、mainForm keypreviewtoを設定し、関数をkeyDowntrueイベントに実装しました。mainForm

mainForm_KeyDownEvent

if (e.KeyCode.Equals(Keys.Enter))  //Invokes whenever Enter is pressed
{
    btnLogin_Click(sender,e);  //login
}

今私の問題は、フォーカスがオンtxtUserNameで を押すたびに、 A「管理者」(上記のプロパティで示したようにコレクションで定義されている) を選択するドロップダウンが表示されることです。キーボードをクリックEnterすると、「管理者」を選択する代わりにメッセージボックスが表示されます。の keydown イベントを呼び出していることはわかっていmainFormます。keyDown イベントがテキストボックスのドロップダウンにあるときに、キーを押すことができるようにするにはどうすればよいenterですか?

編集:
私は以下のコードを試しましたpublic form():(動作しません)

InitializeComponent();
if (txtUserName.AutoCompleteMode) { /* showing red scribbles */
            this.KeyDown -= mainForm_KeyDown;
        }
4

4 に答える 4

4

Enter キーをまったく処理しないでください。KeyDownハンドラーを削除し、代わりAcceptButtonにフォームのプロパティを使用して、Enter キーが押されたときに「クリック」されるボタンを設定できます。別のコントロールが既に Enter キーを処理している場合、これはボタンを「クリック」しないことになっています。

Windows の標準的な動作では、Enter キーがデフォルト ボタンを押すため、これでは十分ではありません。たとえば、Win+R を押して [ファイル名を指定して実行...] ダイアログを表示し、C:\Use と入力し始め、下を押して C:\Users を選択し、Enter を押して、何が起こるかを確認します。

その動作をオーバーライドするには、テキスト ボックスが Enter キー自体を処理することをフォームに伝え、フォームがそれをデフォルト ボタンに送信しないようにする必要があります。これは、派生クラスを作成してオーバーライドすることで実行できますIsInputKey

public class MyTextBox : TextBox
{
    protected override bool IsInputKey(Keys keyData)
    {
        return base.IsInputKey(keyData) || ((keyData & ~Keys.Shift) == Keys.Enter && IsDroppedDown);
    }
}

ただし、関数TextBoxを使用してオートコンプリートを実装し、バックグラウンドでオブジェクトを自動的に作成します。そのオブジェクトにアクセスできないため、使用したプロパティを作成できません。を使用して実装されますが、オブジェクトにアクセスできないため、ドロップダウン リストが表示されているかどうかを (確実に) 判断できません。SHAutoCompleteIAutoCompleteIsDroppedDownIsInputKeyIAutoCompleteDropDown.GetDropDownStatus

組み込みプロパティを使用せずにオートコンプリートを実装するか、常にEnter キーを抑制するAutoComplete*必要があります (上記の を削除します)。&& IsDroppedDownIsInputKey

更新IAutoComplete:オブジェクトを手動で作成する方法は次のとおりです。文字列 Administrator と Clerk はハードコードされています。GetDropDownStatus 関数は、ドロップダウン リストが表示されている場合に、既定のボタンの Enter の処理を​​抑制するために使用されます。フィードバック歓迎。

IAutoComplete.cs:

using System;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.ComTypes;

[ComImport]
[Guid("00bb2762-6a77-11d0-a535-00c04fd7d062")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
[CoClass(typeof(IAutoCompleteClass))]
interface IAutoComplete
{
    void Init(HandleRef hwndEdit, IEnumString punkACL, string pwszRegKeyPath, string pwszQuickComplete);
    void Enable(bool fEnable);
}

IAutoComplete2.cs:

using System;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.ComTypes;

[Guid("EAC04BC0-3791-11d2-BB95-0060977B464C")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
interface IAutoComplete2
{
    void Init(HandleRef hwndEdit, IEnumString punkACL, string pwszRegKeyPath, string pwszQuickComplete);
    void Enable(bool fEnable);
    void SetOptions(AutoCompleteOptions dwFlag);
    AutoCompleteOptions GetOptions();
};

AutoCompleteOptions.cs:

using System;

[Flags]
enum AutoCompleteOptions : int
{
    None = 0x00,
    AutoSuggest = 0x01,
    AutoAppend = 0x02,
    Search = 0x04,
    FilterPrefixes = 0x08,
    UseTab = 0x10,
    UpDownKeyDropsList = 0x20,
    RtlReading = 0x40,
    WordFilter = 0x80,
    NoPrefixFiltering = 0x100,
}

IAutoCompleteDropDown.cs:

using System;
using System.Runtime.InteropServices;
using System.Text;

[Guid("3CD141F4-3C6A-11d2-BCAA-00C04FD929DB")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
interface IAutoCompleteDropDown
{
    void GetDropDownStatus(out AutoCompleteDropDownFlags dwFlags, out StringBuilder wszString);
    void ResetEnumerator();
}

AutoCompleteDropDownFlags.cs:

using System;

[Flags]
enum AutoCompleteDropDownFlags : int
{
    None = 0x00,
    Visible = 0x01
}

IAutoCompleteClass.cs:

using System;
using System.Runtime.InteropServices;

[ComImport]
[Guid("00BB2763-6A77-11D0-A535-00C04FD7D062")]
class IAutoCompleteClass
{
}

EnumString.cs:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.ComTypes;

class EnumString : IEnumString
{
    const int E_INVALIDARG = unchecked((int)0x80070057);
    const int S_OK = 0;
    const int S_FALSE = 1;

    int current;
    string[] strings;

    public EnumString(IEnumerable<string> strings)
    {
        this.current = 0;
        this.strings = strings.ToArray();
    }

    public void Clone(out IEnumString ppenum)
    {
        ppenum = new EnumString(strings);
    }

    public int Next(int celt, string[] rgelt, IntPtr pceltFetched)
    {
        if (celt < 0)
            return E_INVALIDARG;

        int num = 0;
        while (current < strings.Length && celt != 0)
        {
            rgelt[num] = strings[current];
            current++;
            num++;
            celt--;
        }

        if (pceltFetched != IntPtr.Zero)
            Marshal.WriteInt32(pceltFetched, num);

        if (celt != 0)
            return S_FALSE;

        return S_OK;
    }

    public void Reset()
    {
        current = 0;
    }

    public int Skip(int celt)
    {
        if (celt < 0)
            return E_INVALIDARG;

        if (strings.Length - current > celt)
        {
            current = strings.Length;
            return S_FALSE;
        }

        current += celt;
        return S_OK;
    }
}

MyTextBox.cs:

using System;
using System.Runtime.InteropServices;
using System.Text;
using System.Windows.Forms;

public class MyTextBox : TextBox
{
    IAutoComplete2 autoComplete;
    IAutoCompleteDropDown autoCompleteDropDown;

    public bool IsDroppedDown
    {
        get
        {
            if (autoCompleteDropDown == null)
                return false;

            AutoCompleteDropDownFlags dwFlags;
            StringBuilder wszString;
            autoCompleteDropDown.GetDropDownStatus(out dwFlags, out wszString);
            return (dwFlags & AutoCompleteDropDownFlags.Visible) != AutoCompleteDropDownFlags.None;
        }
    }

    protected override void CreateHandle()
    {
        base.CreateHandle();

        autoComplete = (IAutoComplete2)new IAutoComplete();
        autoCompleteDropDown = (IAutoCompleteDropDown)autoComplete;
        autoComplete.SetOptions(AutoCompleteOptions.AutoSuggest);
        autoComplete.Init(new HandleRef(this, this.Handle), new EnumString(new string[] { "Administrator", "Clerk" }), null, null);
    }

    protected override void DestroyHandle()
    {
        ReleaseAutoComplete();
        base.DestroyHandle();
    }

    protected override void Dispose(bool disposing)
    {
        if (disposing)
        {
            ReleaseAutoComplete();
        }
        base.Dispose(disposing);
    }

    protected override bool IsInputKey(Keys keyData)
    {
        return base.IsInputKey(keyData) || ((keyData & ~Keys.Shift) == Keys.Enter && IsDroppedDown);
    }

    void ReleaseAutoComplete()
    {
        if (autoComplete != null)
        {
            Marshal.ReleaseComObject(autoComplete);
            autoComplete = null;
            autoCompleteDropDown = null;
        }
    }
}
于 2012-10-06T12:55:55.590 に答える
1

実際には、2 つの問題があります。

まず、txtUserName の AutoCompleteMode プロパティを単に「Suggest」ではなく「SuggestAppend」に設定します。このように、ユーザーが最初の 1 文字または 2 文字を入力すると、正しいエントリが txtUSerName.Text に自動的に追加されます。

次に、フォーム コードを次のように変更します。

void  Form1_KeyDown(object sender, KeyEventArgs e)
{
    if (e.KeyCode.Equals(Keys.Enter))  //Invokes whenever Enter is pressed
    {
        if (txtPassword.ContainsFocus)
        {
            btnLogin_Click(sender, e);  //login
        }
        else
        {
            this.txtPassword.Focus();
        }
    }
}


private void btnLogin_Click(object sender, EventArgs e)
{
    if (txtUserName.Text.Equals("Administrator") && txtPassword.Text.Equals("123"))
    {
        MessageBox.Show("Administrator");
    }
    else if (txtUserName.Text.Equals("Clerk") && txtPassword.Text.Equals("123"))
    {
        MessageBox.Show("Clerk");
    }
    else
    {
        MessageBox.Show("Please Enter correct details", "Login Error");
    }
}

上記の Key Down イベント処理コードは、パスワード テキスト ボックスにフォーカスがあるかどうかをテストします (つまり、ユーザーは既にユーザー名とパスワードを入力しており、データを送信する準備ができていることを意味します)。その場合、btnLogin_Click イベントが呼び出されます。それ以外の場合 (つまり、おそらく txtUserName にフォーカスがあることを意味します) 制御が txtPassword に渡され、データ入力が続行されます。

更新: re - あなたのコメント:

次のように、Key Down イベント ハンドラーのロジックを単に kill します。

改訂されたイベント処理コード:

void  Form1_KeyDown(object sender, KeyEventArgs e)
{
    if (e.KeyCode.Equals(Keys.Enter))  //Invokes whenever Enter is pressed
    {
        btnLogin_Click(sender, e);  //login
    }
}

別のマイナーな改善 (コードの全体的な構造を考慮) は、UserName の選択にコンボ ボックスを使用し、オートコンプリート ソースを "ListItems" に設定してから、テキスト ボックスと同じようにオプションを入力することです。これには、ユーザーが事前定義されたリストから選択する必要があります。これには、前と同様のスケーラビリティの問題がまだありますが、ユーザー名データの入力中に単純にタイプミスをした場合、ユーザーにとって不要な手順がなくなります。

ユーザーは、ポップアップ メッセージによる不必要な中断を嫌う傾向があることに注意してください。ドロップダウンから適切な「ユーザー名」を選択し、適切なパスワードを入力して、次に進むことができるようにします。

これらすべてを行うためのより良い方法がいくつかありますが、これにより、現在持っているものを正常に機能するように調整する必要があります。

最後に、最終的には、この種の検証を実行するためのより堅牢な方法を見つけたいと思うでしょう。ユーザーを追加する必要があるときはいつでも(コードでは、「グループ」として定義されているように見えます)、条件付きイベント処理ツリーに追加する必要があります。

暗号化されたファイルまたはデータベースに永続化されたユーザー名とパスワードをチェックインし、それらを実行時に辞書などにロードする場合があります。次に、ユーザー/パスワードでキー/値のルックアップを実行します。

か何か。

とにかく、それが役立つことを願っています。

更新 2: 完全なコードをすべてワンショットで。これは、あなたが求めているように動作するはずです:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            this.KeyDown +=new KeyEventHandler(Form1_KeyDown);
        }

        void  Form1_KeyDown(object sender, KeyEventArgs e)
        {
            if (e.KeyCode.Equals(Keys.Enter))  //Invokes whenever Enter is pressed
            {
                btnLogin_Click(sender, e);  //login
            }
        }


        private void btnLogin_Click(object sender, EventArgs e)
        {
            if (txtUserName.Text.Equals("Administrator") && txtPassword.Text.Equals("123"))
            {
                MessageBox.Show("Administrator");
            }
            else if (txtUserName.Text.Equals("Clerk") && txtPassword.Text.Equals("123"))
            {
                MessageBox.Show("Clerk");
            }
            else
            {
                MessageBox.Show("Please Enter correct details", "Login Error");
            }
        }
    }
}
于 2012-10-06T12:18:46.467 に答える
1

keydown イベント ハンドラをオーバーライドする必要があります。

    protected override void OnKeyDown(KeyEventArgs e)
    {
        //call original event handler. Remove it if you don't need it at all.
        base.OnKeyDown(e);

        //Insert your code here....
    }
于 2012-10-06T11:17:37.427 に答える
1

これを試して。フォーカスがtxtUsernameまたはその他の場所にあるときにEnterキーを押しても問題が発生しないことを願っています

aと入力しtxtUserNameてエンターを押すと、使用Admministrator中から選択され、フォーカスが に移動します。私の正規表現は非常に柔軟性があり、最初から厳密に一致するように次のように少し制限することができ、大文字と小文字を無視することもできますautocompletecustomsourceregular expressiontxtPassword

Regex rg = new Regex("^" + txtUserName.Text);

    private void mainForm_KeyDown(object sender, KeyEventArgs e)
    {
        if (e.KeyCode.Equals(Keys.Enter))// && !txtUserName.Focus())// && intFlag.Equals(0))
        {
            if (txtUserName.Text.Length > 0)
            {
                if (txtUserName.Focused)
                {
                    Regex rg = new Regex(txtUserName.Text, RegexOptions.IgnoreCase);
                    for (int i = 0; i < txtUserName.AutoCompleteCustomSource.Count; i++)
                    {
                        if (rg.IsMatch(txtUserName.AutoCompleteCustomSource[i]))
                        {
                            txtUserName.Text = txtUserName.AutoCompleteCustomSource[i];
                            txtPassword.Focus();
                            return;
                        }
                    }
                }
                if (txtPassword.Text.Length > 0)
                {
                    btnLogin_Click(null, null);  //login
                }
                else
                {
                    //MessageBox.Show("Please Give a Password");
                    txtPassword.Focus();
                }
            }
            else
            {
                //MessageBox.Show("Please Give a username");
                txtUserName.Focus();
            }
        }

        //if (txtPassword.ContainsFocus)
        //{
        //    btnLogin_Click(sender, e);  //login
        //}
        //else
        //{
        //    this.txtPassword.Focus();
        //}
    }
于 2012-10-06T13:21:28.980 に答える