大学IT网 - 最懂大学生的IT学习网站! QQ资料交流群:367606806
当前位置:大学IT网 > C++技巧 > 高手解答:关于RICHEDIT的两个问题

高手解答:关于RICHEDIT的两个问题

关键词:解答两个高手RICHEDIT  阅读(919) 赞(10)

[摘要]本文是对高手解答:关于RICHEDIT的两个问题的讲解,对学习C++编程技术有所帮助,与大家分享。

效果1:RichEditCtrl在用DDX中止数据交流的时分会发作数据丧生效果?为什么!

当我们在拖了一个控件到顺序里后,通常的做发是Ctrl+W,用类导游给控件关联一个变量,然后依托DDX/DDV中止数据交流,假定我们用异常的方法来给RICHEDIT关联一个CString类型的变量就会存在一个效果,就是假定我们的数据大于了64K,数据就会丧失。

经过查MSDN发现,WM_GETTEXT消息并没有设计在RICHEDIT的数据大于64K的时分怎样处置。而类导游生成的代码是用DDX_Text来交流控件和CString变量的数据。恰恰,DDX_Text函数是调用GetWindowText函数,而这个函数又会发出WM_GETTEXT消息到控件来前往控件里的数据。WM_GETTEXT消息不能接受跨越64K的数据,因此招致了RICHEDIT在数据交流的时分发作了丧失。

为了解决这个效果,我们要用到DDX_RichText函数。添加下面两个函数到工程

以下是引用片段:
  DWORD CALLBACK ES2MemCallBack(DWORD_PTR dwCookie,LPBYTE pbBuff, LONG cb, LONG *pcb)
  {
  LPTSTR& lpszStrFill = *(LPTSTR*)dwCookie;
  memcpy(lpszStrFill, pbBuff, *pcb = cb);
  lpszStrFill += cb;
  *lpszStrFill = TCHAR('\0');
  return 0;
  }
  void AFXAPI DDX_RichText(CDataExchange* pDX, int nIDC, CString& value)
  {
  extern void AFXAPI AfxSetWindowText(HWND hWndCtrl, LPCTSTR lpszNew);
  HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
  if (pDX->m_bSaveAndValidate)
  {
  int nLen = ::GetWindowTextLength(hWndCtrl);
  LPTSTR lpszStrFill = value.GetBufferSetLength(nLen);
  EDITSTREAM es = { (DWORD_PTR) &lpszStrFill, 0, ES2MemCallBack };
  ::SendMessage(hWndCtrl, EM_STREAMOUT, SF_TEXT, (LPARAM) &es);
  value.ReleaseBuffer();
  }
  else
  {
  AfxSetWindowText(hWndCtrl, value);
  }
  }

之后我们还需求修正工程的.clw文件,用文本方式翻开.clw文件。参考里面类的格式加下面两行代码:

以下是引用片段:
  ExtraDDXCount=1
  ExtraDDX1=7;;TextOver64KB;CString;;RichText;Retrieves text in excess of 64KB from RichEdit controls

假定没有采用上面的步骤,我们就需求手动修正代码,把一切的DDX_Text改为DDX_RichText。同时要把他们移到类导游控制代码的外面。也就是移出:

以下是引用片段:
  //{{AFX_DATA_INIT(...)
  //}}AFX_DATA_INIT
  //{{AFX_DATA_MAP(...)
  //}}AFX_DATA_MAP
  reference:
  Q280447 BUG: Text from a Rich Edit Control Is Truncated During Dialog Data Exchange (DDX)

效果2:当我们用类导游给richedit添加了EN_SETFOCUS, EN_KILLFOCUS的函数后却不能照应,我发现这个照应函数根本就没有被调用。即使是一个MessageBox()函数也不会调用。

原来是默许的消息映射添加错误了。

正确的消息影射和照应应该是:

以下是引用片段:
  ON_EN_SETFOCUS(IDC_RICHEDIT1,OnSetfocusRichedit1)
  ON_EN_KILLFOCUS(IDC_RICHEDIT1,OnKillfocusRichedit1)

照应函数方式为:

以下是引用片段:
  afx_msg void OnSetfocusRichedit1();
  afx_msg void OnKillfocusRichedit1();

但是假定我们用类导游来直接添加,生成的代码却是:

以下是引用片段:
  ON_NOTIFY(EN_SETFOCUS, IDC_RICHEDIT1, OnSetfocusRichedit1)
  ON_NOTIFY(EN_KILLFOCUS, IDC_RICHEDIT1, OnKillfocusRichedit1)

我们需求自己手动改为上面的方式。

还有一个效果就是RichEditCtrl有时分不会出现在类导游的控件ID列表里。这就需求我们自己添加DDX/DDV函数。自己动手啦!^_^

以下是引用片段:
  -- sampledlg.h --
  class CSampleDlg : public CDialog
  {
  public:
  CSampleDlg(CWnd* pParent = NULL);
  // Dialog Data
  //{{AFX_DATA(CSampleDlg)
  enum { IDD = IDD_SAMPLE_DIALOG };
  CString m_edit; // Added by ClassWizard for an edit control
  //}}AFX_DATA
  // Manually add member variables for the rich edit control
  CRichEditCtrl m_richEditCtrl;
  .......
  sampledlg.cpp --
  ......
  void CSampleDlg::DoDataExchange(CDataExchange* pDX)
  {
  CDialog::DoDataExchange(pDX);
  //{{AFX_DATA_MAP(CSampleDlg)
  DDX_Text(pDX, IDC_EDIT, m_edit);
  DDV_MaxChars(pDX, m_edit, 10);
  //}}AFX_DATA_MAP
  // Manually add DDX_Control, DDX_Text and DDV_MaxChars for the
  // rich edit control
  DDX_Control(pDX, IDC_RICHEDIT1, m_richEditCtrl);
  DDX_Text(pDX, IDC_RICHEDIT1, m_richedit);
  DDV_MaxChars(pDX, m_richedit, 10);
  }



相关评论