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

2008-02-23 05:40:02来源:互联网 阅读 ()

新老客户大回馈,云服务器低至5折

  问题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(...)
  /

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇: 专家讲解用.NET编写串口程式的一点心得

下一篇: 数学和程式 一道游戏题目的快速解法