输入子类编辑控件的验证以仅接受浮点数

时间:2014-06-20 10:25:29

标签: c++ validation winapi subclassing editcontrol

在我目前的项目中,过去几天我一直在为子类化编辑框而苦苦挣扎。到现在为止,我成功地对我的编辑框进行了子类化并验证了输入,使其只接受数字,逗号,减号和键盘命令。

但是很长一段时间以来我一直坚持改进输入验证。我希望我的编辑框的行为如下:

  • 仅在第一个位置接受减号
  • 只接受一个前导零
  • 只接受一个逗号
  • 在前导零后强制逗号
  • 在删除单个字符或选择文字时管理这些案例,通过' back''删除',选择所有,然后粘贴某些内容

我当前形式的代码如下所示,几乎没有提供我上面指定的高级验证要求:

inline LRESULT CALLBACK decEditBoxProc(HWND hWnd,
                                       UINT msg, 
                                       WPARAM wParam, 
                                       LPARAM lParam,
                                       UINT_PTR uIdSubclass,
                                       DWORD_PTR dwRefData)
{
    if(msg == WM_CHAR)
    {
        decEditBoxData* data = reinterpret_cast<decEditBoxData*>(ULongToPtr(dwRefData));

        bool isDigit          = (wParam >= '0' && wParam <= '9');
        bool isZero           = ((wParam == '0') && !data->blockZero);
        bool isSign           = (wParam == '-');
        bool isComma          = ((wParam == '.' || wParam == ',') && !data->blockComma);
        bool isValidCommand   = (wParam == VK_RETURN  
                                || wParam == VK_DELETE 
                                || wParam == VK_BACK);


        // Restrict comma to one.
        if(isComma && data->nCommas > 0)
            return FALSE;
        else if(isComma && data->nCommas == 0)
            data->nCommas++;

        // Restrict trailing zeroes to one.
        if(isZero && data->nTrailingZeroes > 0)
            return FALSE;
        else if(isZero && data->nTrailingZeroes == 0)
            data->nTrailingZeroes++;

        // Filter everything but digits, commas and valid commands.
        if(!isDigit && !isValidCommand && !isComma)
            return FALSE;
    }
    return DefSubclassProc(hWnd, msg, wParam, lParam);
}

非常感谢任何有关如何通过算法解决此问题的想法。

更新

感谢David Heffernan和IInspectable的建议,我能够(几乎)解决我的问题,而无需对编辑控件进行子类化。

在对话框程序中(包含编辑控件):

switch(msg)
{
case WM_COMMAND:
   switch(LOWORD(wParam))
   {
      case IDC_IN_REAL:
         if(HIWORD(wParam)==EN_CHANGE) onEditChange(hDlg, IDC_IN_REAL);
         break;

      case IDC_IN_IMAG:
         if(HIWORD(wParam)==EN_CHANGE) onEditChange(hDlg, IDC_IN_IMAG);
         break;
    }
    break;
}

使用onEditChange:

void onEditChange(HWND hDlg, int ctrlID)
{
    HWND hEdit    = GetDlgItem(hDlg, ctrlID);
    size_t len    = GetWindowTextLength(hEdit)+1;
    wchar_t* cstr = new wchar_t[len];
    GetWindowText(hEdit, cstr, len);

    std::wstring wstr(cstr);

    if(!(tools::isFloat(wstr)))
    {
        EDITBALLOONTIP bln;
        bln.cbStruct = sizeof(EDITBALLOONTIP);
        bln.pszTitle = L"Error";
        bln.pszText  = L"Not a valid floating point character.\nUse '.' instead of ','";
        bln.ttiIcon  = TTI_ERROR;
        Edit_ShowBalloonTip(hEdit, &bln);
    }
    delete [] cstr;
}

和isFloat():

bool tools::isFloat(std::wstring str)
{
    std::wistringstream iss(str);
    float f;
    wchar_t wc;
    if(!(iss >> f) || iss.get(wc))
        return false;
    return true;
}

我可能会为用户添加更多视觉反馈,但现在这并不重要。

然而,这个问题尚未得到解答。我的意图是允许&#34;,&#34;作为一个可能的小数点。

1 个答案:

答案 0 :(得分:-1)

你需要一台状态机。首先宣布你的州。

enum InputState
{
  NoCharacters
  MinusSign
  LeadingZero
  PreDecimalPoint
  PostDecimalPoint
}
InputState mState = NoCharacters

每当用户输入一个字符时,根据使用开关块的mState值调用不同的验证函数。

bool ValidCharacter(char input)
{
  switch (mState)
  {
        case NoCharacters:
          return NoCharacters(input);
        case MinusSign:
          return MinusSign(input);
        /// etc
  }
}

因此,例如,当mState == NoCharacters时调用的函数将接受任何数字,小数点或减号。 然后,如果字符为减号,则将mState更改为MinusSign;如果为零则为LeadingZero,等等。