多列自动完成组合框

 

介绍 一个组合框,具

  

 

有自动完成功能,也允许多个 列。它来源于Chris Maunder的CComboCompletion。 版本1.2修复了以下问题: 尺寸问题,由马格努斯·埃格伯格和他的同事提供 AddRow()例程现在被重载,以便更容易添加 行。不再需要SetColCount(UINT)。的重载 一个CComboBox例程,已被DeleteRow()取代。通过删除额外的特写镜头,对子字符串的奇反转也得到了解决 收到“Enter”键。 这个组合框 处理以相同字符开头且仅表示匹配的文本字符串 当它是独一无二的。即. .假设盒子里装满了字符串: 隐藏,复制Code

0
1
100
1234

在键入“1”时,这是第二个项目的唯一匹配,因此组合框 就会打开并突出那个。如果你继续“0”,那么 combobox将再次打开并显示100,但最后一个0将是 突出显示,因为这是下一个字符将去的地方。键入一个 “1”后的“2”将选择“1234”,“34”将高亮显示。多个 支持列,因此combobox必须是所有者绘制,变量,与 字符串。可以为combobox提供下拉菜单或下拉菜单的属性 列表。在后者中,输入的字符不匹配 第一列字符串将发出哔哔声并被拒绝。通过对话框中的一行创建combobox 这通常是由MS类向导为相关的类提供的 当combobox被插入对话框时: 隐藏,复制Code

class CMyDialog : CDialog {
    ...
    CAutoCombobox m_myAutoCombobox;
    ...
};
CMyDialog::CMyDialog(...)
 : m_myAutoCombobox(ACB_DROP_LIST) { }

或者使用ACB_DROP_DOWN。在下面的文章中有以下内容 定义。 隐藏,复制Code

typedef vector<CString> CStringsVec;
typedef vector<int> IntVec;

特殊目的成员功能如下: 隐藏,复制Code

CAutoComboBox::AddRow(const TCHAR* pString);
CAutoComboBox::AddRow(const CStringsVec& allCols);
CAutoCombobox::AddRow(const TCHAR* pString, CStringsVec& extraCols);

这些例程被重载以在每一行中添加字符串。第一个 如果需要单列自动补全,只需为一列添加文本。 第二个允许在向量中插入所有列的字符串 字符串。向量中的第一个字符串将被添加到实际 combobox本身和其他部分将被保存在一个内部成员字符串向量中。 第三个是向后兼容性,需要分离 字符串中的第一列用于额外的列。第一次一排 ,列的数量将被记住,而所有随后的添加必须 具有完全相同数量的列字符串。如果列为空,则使用 空字符串填充空格(_T(""))。 隐藏,复制Code

CAutoCombobox::DeleteRow(int iRow) 

这将从内部删除行和相应的额外文本 向量。 隐藏,复制Code

CAutoCombobox::GetSelectedRow() const 

可以通过此函数检索所选的行。如果没有行已被 唯一标识,返回值为-1。 隐藏,复制Code

CAutoCombobox::ResetSelection() 

当前选择被取消,我们可以从头开始比赛 再一次 隐藏,复制Code

CAutoCombobox::ResetContent() 

所有的行都从组合框和内部数组中删除,这样您就可以这样做了 重新开始填充。 隐藏,复制Code

CAutoCombobox::SetReverseEntry(bool bReverse = true) 

在我的例子中,我需要允许输入来自的字符串 从右到左,而不是从左到右。对于一个数字列表,它的可变性是 通常更多的是低阶数字,所以1一般不需要输入a 许多字符得到唯一的匹配。把100写成a很奇怪 "0"后面跟着"0",然后是"1"。 隐藏,复制Code

CAutoCombobox::DeleteString(UINT uRow) 

这将从内部删除行和相应的额外文本 向量。这个例程不赞成,应该由DeleteRow(int)替换。 隐藏,复制Code

CAutoCombobox::SetColCount(int iCols) 

这个例程不再需要了,因此不建议使用它。列 count是从第一个AddRow(…)调用开始建立的。如果使用此例程,则必须在添加任何行之前调用,因为它有效 额外字符串向量的大小。 示例项目加载具有两列和一组数值的combobox 字符串。它是一个MDI应用程序,并已通过MS Visual Studio .NET 2003和MDI测试 Windows XP。在VS6.0实现中有一个剩下的.dsw文件 似乎也起作用了!)选择菜单项Test(^快捷方式)将打开 已被字符串填充的对话框。输入一个3将打开盒子和 显示选中“3”的行。现在继续输入453,盒子就会 再次打开,现在将选择显示为包含345337的行。指定从 这样就可以把345337保存在盒子里了。点击OK, MDI应用程序显示 通过项目:nnnn在主窗口中选择。 我需要一个不区分大小写的匹配,所以使用了CString::CompareNoCase(…)。 如果需要区分大小写的匹配,则将两个引用更改为比较 函数只需使用CString::Compare(…)。 我删除了下拉列表w的关闭母鸡的 “Enter”键出现了。这样做的目的是允许一次击键 关闭下拉列表,并触发的默认按钮 包含对话框。对话框中的组合框的正常行为是 第一个“Enter”关闭列表,第二个“Enter”触发默认值 动作,总共两个按键。如果这个额外的特写实现在a Win95/98/ME操作系统,然后一个问题出现了。有时字符串会恢复 到先前匹配的子字符串。这是由于在 ComboBox对象的Microsoft实现。没有问题 NT或XP。反转只在字符串发生的时候发生 首先匹配一个短字符串,然后匹配一个长字符串。在 在上面的列列表中,输入“1”将选择“1”行。现在进入 一个“0”,将选择“100”行。点击组合框外将关闭 下拉菜单,但是选择会因为一些未知的原因返回到“1”。 有趣的是,如果“1”后面跟着“0”来突出显示“100”和 然后向上箭头/向下箭头用于选择下一行并返回 “100”,焦点的改变将使“100”留在原位而不回调 “1”。它不会发生在选择的“3”后面跟着“2”的时候 “323513”如图所示。将焦点从选中的“323513”移开 即使列表中有一个“3”,它还是保留了这个。 记录这个的微软事件编号 错误是SRX020705601306。根据微软的说法,没有这样的工作 这就是在这个新版本中关闭的原因! 控件的主要部分在OnEditUpdate()例程中,如下所示。当一个 找到唯一匹配时,combobox将被放下,而所涉及的项目是 高亮显示。 PreTranslateMsg(…)必须测试退格或删除和取消 自动完成下一个字符。如果不这样做,这是不可能的 在系统继续输入下一个字符时纠正错误。这 在Chris Maunder的CComboCompletion示例中有更好的描述。 清单需要头文件来显示列出的变量定义 接下来,然后 下面是OnEditUpdate()代码。获取用户输入的文本行 通过GetWindowText()函数,并在找到匹配项时进行修改 SetWindowText()和SetEditSel()。 清单取自autocbox.h 隐藏,收缩,复制Code

using std::vector;
typedef vector<Int> IntVec;
typedef vector<CString> CStringsVec;

////////////////////////////////////////////////////////////////////////
//
// CAutoComboBox window
//
class CAutoComboBox : public CComboBox {

public:
    // define the two types to emulate
    enum AutoComboBoxType {
        ACB_DROP_DOWN,
        ACB_DROP_LIST
    };
    CAutoComboBox(AutoComboBoxType acbt = ACB_DROP_LIST);
    virtual ~CAutoComboBox();

    // ClassWizard generated virtual function overrides
private:
    virtual BOOL PreTranslateMessage(MSG* pMsg);
    virtual void DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct);
    virtual void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct);

public:
    int GetSelectedRow();
    int AddRow(const TCHAR* pString, const CStringsVec& extraCols);
    int AddRow(const CStringsVec& cols);
    int AddRow(const TCHAR* pString);
    int DeleteRow(int iRow);
    void SetReverseEntry(bool bReverse) { m_bReverseEntry = bReverse; }
    void GetLBText(int nIndex, CString& rString) const;
    void ResetSelection();
    void ResetContent();
    // the following routines are deprecated!!
    int DeleteString(UINT uRow);        // use DeleteRow(...)
    void SetColCount(int iCols);        // not needed any more

protected:
    afx_msg void OnEditUpdate();
    afx_msg void OnSelChange();
    afx_msg void OnCbnDropdown();

    DECLARE_MESSAGE_MAP()
private:
    int FindUniqueMatchedItem(const CString& line, CString& matchedLine);
    int SetDroppedWidthForTextLengths();
    bool LineInString(const CString& rsLine, const CString& rsString) const;
    int m_iSelectedRow;
    int m_iHorizontalExtent;    // list box width for all columns
    bool m_bAutoComplete;       // false after delete or backspace
    bool m_bReverseEntry;       // enter characters in reverse
    bool m_bEditHeightSet;      // true when we've set the height of the box
    bool m_bClosing;            // stops error in selection on enter
    AutoComboBoxType m_autoComboBoxType;
    IntVec m_vecEndPixels;      // ending positions for each column
    vector<CStringsVec> m_vecRows;// row text for each col after the first
};

清单取自autocbox.cpp 隐藏,收缩,复制Code

...
void
CAutoComboBox::OnEditUpdate() 
{
    CString line;               // partial line entered by user
    CString sMatchedText;       // holds full line from list

    // get the text from the user input
    GetWindowText(line);
    int iHiLightStart = line.GetLength();

    // ?? remove
    // TRACEFN(_T("Entering OnEditUpdate line:%s\n"), (LPCTSTR) line);

    // if the line is empty
    if(line.GetLength() == 0) {
        // make sure the dropdown list is closed when back to
        // zero characters in the edit box
        ShowDropDown(FALSE);

        // empty the selection
        SetWindowText(_T(""));

        // cancel any previous selection
        m_iSelectedRow = -1;

        // turn on autocomplete again turned off by deletes
        m_bAutoComplete = true;
        return;
    }
    // if we have seen a delete or back key we leave the text alone so
    // that the user can continue to go backwards to correct an entry
    if(!m_bAutoComplete) {
        // but only for one character
        m_bAutoComplete = true;

        // if reversing entry, must keep insertion point on right
        if(m_bReverseEntry) {
            SetEditSel(0, 0);
        }
        return;
    }
    // if a single match, we can display that and identify the mismatch
    m_iSelectedRow = FindUniqueMatchedItem(line, sMatchedText);

    // if we found a unique matching row
    if(m_iSelectedRow >= 0) {

        // drop down the list part of the combo box
        ShowDropDown(TRUE);

        // ?? remove
        // TRACE(_T("OnEditUpdate: ShowDropDown(true)\n"));

        // set the dropdown to show the item as well
        // ?? int iCurSel = SetCurSel(m_iSelectedRow);
        // assert(iCurSel >= 0);
        // we try this so that the selection occurs AFTER the dropdown
        // box has been filled
        PostMessage(CB_SETCURSEL, m_iSelectedRow, 0);

        // ?? remove
        // TRACE(_T("OnEditUpdate-Post: SetCurSel(%d)\n"), m_iSelectedRow);

        // now we have to also remove the selection from the edit box
        // since we may want to continue to add to the string and
        // set out own highlight on the text we have added from the match
        //
        int iStartChar = 0;
        int iEndChar = -1;

        if(!m_bReverseEntry) {
            // straight entry, we want to highlight the added text from
            // the end of the input line text to the end of the field
            iStartChar = line.GetLength();
        }
        else {
            // reverse entry, we want to highlight the added text
            // on the left - from char 0 to match less line length
            iEndChar = sMatchedText.GetLength() - line.GetLength();
        }
        PostMessage(CB_SETEDITSEL, 0, MAKELPARAM(iStartChar, iEndChar));

        // ?? remove
        // TRACE(_T("OnEditUpdate-Post: SetEditSel(%d, %d)\n")
        //  , iStartChar, iEndChar);

        // set the window text to the match
        line = sMatchedText;
    }
    // if this text won't match any string and we are emulating
    // a drop list so the selection is forced
    else if(sMatchedText.IsEmpty() && m_autoComboBoxType == ACB_DROP_LIST) {

        // alert the user to no match
        MessageBeep(MB_ICONEXCLAMATION);

        // if we are not entering text backwards
        if(!m_bReverseEntry) {
            // remove the last character typed in error. at the end
            line.Delete(line.GetLength() - 1);
        }
        else {
            // remove the last character typed in error (at the beginning)
            // which will be in the firt position
            line.Delete(0);
        }
        assert(iHiLightStart > 0);
        iHiLightStart -= 1;

        // restore the line since the closeup cleared the edit box
        SetWindowText(line);

        // ?? remove
        // TRACE(_T("OnEditUpdate: SetWindowText(%s)\n"), (LPCSTR) line);
    }
    else {
        // the list box will be closed to show that there is no match
        ShowDropDown(FALSE);

        // ?? remove
        // TRACE(_T("OnEditUpdate: ShowDropDown(false)\n"));

        // restore the line since the closeup cleared the edit box
        SetWindowText(line);

        // ?? remove
        // TRACE(_T("OnEditUpdate: SetWindowText(%s)\n"), (LPCSTR) line);
    }
    // if we are not entering text backwards
    if(!m_bReverseEntry) {
        // now set the selection to beyond the last character which
        // moves the caret to the end
        SetEditSel(iHiLightStart, -1);

        // ?? remove
        // TRACE(_T("OnEditUpdate: SetEditSel(%d, -1)\n"), iHiLightStart);
    }
    else {
        // now set the insertion caret to the beginning so we
        // build up the text from the right
        SetEditSel(0, 0);

        // ?? remove
        // TRACE(_T("OnEditUpdate: SetEditSel(0, 0)\n"));
    }
    // ?? remove
    // TRACE(_T("EditUpdate SelectedRow: %d Line: %s Matched Text: %s\n")
    // , m_iSelectedRow, (LPCTSTR) line, (LPCTSTR) sMatchedText);
}

本文转载于:http://www.diyabc.com/frontweb/news347.html

posted @ 2020-08-05 10:20  Dincat  阅读(201)  评论(0编辑  收藏  举报