我爱电脑技术论坛's Archiver

白雪公主 发表于 2008-3-30 09:21

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

 问题1:RichEditCtrl在用DDX进行数据交换的时候会发生数据丢失问题?为什么!
J5d%R(@Y,CX+Sc
7Ig~!b!lN5S!Rk   当我们在拖了一个控件到程序里后,通常的做发是Ctrl+W,用类向导给控件关联一个变量,然后依靠DDX/DDV进行数据交换,如果我们用同样的方法来给RICHEDIT关联一个CString类型的变量就会存在一个问题,就是如果我们的数据大于了64K,数据就会丢失。
u{ Ba7M3fM3v9r 1C.w)amHR
  通过查MSDN发现,WM_GETTEXT消息并没有设计在RICHEDIT的数据大于64K的时候怎样处理。而类向导生成的代码是用DDX_Text来交换控件和CString变量的数据。恰好,DDX_Text函数是调用GetWindowText函数,而这个函数又会发出WM_GETTEXT消息到控件来返回控件里的数据。WM_GETTEXT消息不能接受超过64K的数据,因此导致了RICHEDIT在数据交换的时候发生了丢失。f&YG[V)q
(c @ \-b u8Lu@-HZ O r
  为了解决这个问题,我们要用到DDX_RichText函数。添加下面两个函数到工程|I@8__}
L5v xLJN \#i7b
以下是引用片段:]rHrz#Ke'[.Y
  DWORD CALLBACK ES2MemCallBack(DWORD_PTR dwCookie,LPBYTE pbBuff, LONG cb, LONG *pcb)
n R9@.h a q   { :u u.r^XC0[VU
  LPTSTR& lpszStrFill = *(LPTSTR*)dwCookie;
}e hvx   memcpy(lpszStrFill, pbBuff, *pcb = cb); X]4t)Q*AGk
  lpszStrFill += cb;
2G4mx`0yI   *lpszStrFill = TCHAR('\0');
8U\\!A$f Q8I"Q   return 0; 9Y#``TX&q {7^*p
  }
,]"}K8oKP   void AFXAPI DDX_RichText(CDataExchange* pDX, int nIDC, CString& value)
&FeN%} D+wt~1Z   {
,hr|j(Y   extern void AFXAPI AfxSetWindowText(HWND hWndCtrl, LPCTSTR lpszNew); T9I X%M0L{twk
  HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
5A.|T;R,B*q   if (pDX->m_bSaveAndValidate) wD3X|0P pe
  { ngV$Qss VhU?
  int nLen = ::GetWindowTextLength(hWndCtrl); Q1v!d1Cx)x EY
  LPTSTR lpszStrFill = value.GetBufferSetLength(nLen);
'?` Hw&iF&a|jm   EDITSTREAM es = { (DWORD_PTR) &lpszStrFill, 0, ES2MemCallBack };
]8Bg$c]?   ::SendMessage(hWndCtrl, EM_STREAMOUT, SF_TEXT, (LPARAM) &es);
-s#S+E}~7b.N!p   value.ReleaseBuffer();
/_r#x-ff]   } 2LN1@1?}b
  else
^8j5r \1AG-f{   { K)cM*_HE2f?f
  AfxSetWindowText(hWndCtrl, value); 4Ji#GYdR/_
  }
{(c,k)ux6a H3S   }
?-w nG!Xn(H'a XY^BefK

-c3kFx,k   之后我们还需要修改工程的.clw文件,用文本方式打开.clw文件。参考里面类的格式加下面两行代码: Q @ct0Ob
OFti y:B&q
以下是引用片段:%N$G&e,|M4w:`K$Y
  ExtraDDXCount=1
ed7q}Y   ExtraDDX1=7;;TextOver64KB;CString;;RichText;Retrieves text in excess of 64KB from RichEdit controls fU!Ap?j.F

he+Mqk k
&fU%qG'SQ-M$@ ?(|   如果没有采用上面的步骤,我们就需要手动修改代码,把所有的DDX_Text改为DDX_RichText。同时要把他们移到类向导控制代码的外面。也就是移出:
i)BnxC"tU[
T7z5q5f1X8r,B 以下是引用片段:
:`F'|1L c7O u   //{{AFX_DATA_INIT(...) 7\^,b7`Hd;~-J@
  //}}AFX_DATA_INIT 2j$r/g K$[1W"h*`
  //{{AFX_DATA_MAP(...) T;O,S2N!H
  //}}AFX_DATA_MAP ]N2YbP;mPM7dmlZ
  reference:
sc*@[7Sp u   Q280447 BUG: Text from a Rich Edit Control Is Truncated During Dialog Data Exchange (DDX)
*t,I^K[f7]\;K F,y {w[4`%x7y.N
4CDu4o;|
  问题2:当我们用类向导给richedit添加了EN_SETFOCUS, EN_KILLFOCUS的函数后却不能响应,我发现这个响应函数根本就没有被调用。即使是一个MessageBox()函数也不会调用。4F epU bZJ
%]HGx:n$^L
  原来是默认的消息映射添加错误了。!?qgIg

IN)A5{+YUW?c   正确的消息影射和响应应该是::y\^H9Ryd
*Va4{ v W
以下是引用片段:i{:Gd!h#tp
  ON_EN_SETFOCUS(IDC_RICHEDIT1,OnSetfocusRichedit1)
8xB1FT{   ON_EN_KILLFOCUS(IDC_RICHEDIT1,OnKillfocusRichedit1)
.H n` ?{8N(_7w Q6UG0Svq3V
W-[4?)^Y6Gr
  响应函数形式为:
2bV1ay-B IV$~ Tbe4I~#[V
以下是引用片段:
c`"m^~0HL+~+Y2W   afx_msg void OnSetfocusRichedit1();
3B r:mkhS/agb   afx_msg void OnKillfocusRichedit1();
5k3Y2Hu6d#tQ 4mx4LL\(Pr6u

2P!MNC7ARg:t   但是如果我们用类向导来直接添加,生成的代码却是:j WTH0U#M_
2od j8E ~4c W
以下是引用片段:
,fl%f3x.US*@l z   ON_NOTIFY(EN_SETFOCUS, IDC_RICHEDIT1, OnSetfocusRichedit1) 7HF)S ~.Mii0g*q.J
  ON_NOTIFY(EN_KILLFOCUS, IDC_RICHEDIT1, OnKillfocusRichedit1) *b+o3c M0E\ @:m
u P8{5W0kET |

5fI j9Ki7V4Y LrS;C   我们需要自己手动改为上面的形式。
&i1RLm0uea}
3| ~7Utw {o   还有一个问题就是RichEditCtrl有时候不会出现在类向导的控件ID列表里。这就需要我们自己添加DDX/DDV函数。自己动手啦!^_^
R ?\O6@;_ f/rj{8tfgG {Bga'z!G \
以下是引用片段:!a:^h"zs
  -- sampledlg.h --
N E)G;Zn)J_z   class CSampleDlg : public CDialog %fN'L,WIt
  { m i#O`w5kx"`LC
  public: I%_+aA'g4Y
  CSampleDlg(CWnd* pParent = NULL);
|cm}Y `k3a#}   // Dialog Data
&oRL0l,H O"R7?+UKq   //{{AFX_DATA(CSampleDlg) \vt%a_7x {$r Q
  enum { IDD = IDD_SAMPLE_DIALOG }; U B)T+G0A,s;r:j
  CString m_edit; // Added by ClassWizard for an edit control 1_c1V{@2[#Ma^
  //}}AFX_DATA u Q'_@CF?Y
  // Manually add member variables for the rich edit control
b m,S$]7?;?k n c   CRichEditCtrl m_richEditCtrl; :{:J s'm&O7MNa:]
  ....... K^&Kd0Zwg
  sampledlg.cpp -- r6@&Lx|UQ q,p
  ......
bQC V7jFY   void CSampleDlg::DoDataExchange(CDataExchange* pDX)
*vBK | ySVLbF   {
U+h!qz%Z   CDialog::DoDataExchange(pDX);
W:r^%Y.a!j   //{{AFX_DATA_MAP(CSampleDlg)
p,@9A;ZP,P vs'T0a ?#g   DDX_Text(pDX, IDC_EDIT, m_edit);
2bEJ)O Y:\   DDV_MaxChars(pDX, m_edit, 10); Ax WQ;r}/nw
  //}}AFX_DATA_MAP B1m'Yc ||
  // Manually add DDX_Control, DDX_Text and DDV_MaxChars for the
T\cyP'W y   // rich edit control [%|2o|(Z;HC6bx
  DDX_Control(pDX, IDC_RICHEDIT1, m_richEditCtrl); |}L&@M(Q q_'c in5n$x a
  DDX_Text(pDX, IDC_RICHEDIT1, m_richedit);
d@LO(M&U/`j   DDV_MaxChars(pDX, m_richedit, 10); LvE;M%wX ydR&j
  }

页: [1]
   

Powered by Discuz! Archiver 6.1.0  © 2001-2007 Comsenz Inc.