NET RichTextBox控件如何可以插入图像

时间:2022-02-11 07:16:00

本文介绍。NET RichTextBox控件如何可以插入图像,控制和ActiveX对象通过使用OLE方式,如在解释,。不幸的是,它涵盖了只用一个C源代码样本,所以我需要在托管代码(C#)实施类似的解决方案。
有一个RichTextBox插入图片和OLE对象的其他有关条款,但他们使用的是RTF代码,我需要一个更适合聊天使用专门的控制,并提供了一​​种方法插入表情符号,进度条和图像,最后,恢复他们得到他们的OLE处理或任何对象的属性。特别感谢
Khendys戈登的文章:""。背景
要实现的解决方案,我需要使用的P / Invoke平台调用方法。我从大量的信息。
插入RichTextBox的OLE对象的第一步是得到它IRichEditOle的接口。它可以做,通过发送消息EM_GETOLEINTERFACE控制:this.IRichEditOle = SendMessage消息(richEditHandle,EM_GETOLEINTERFACE,0);
有了这个接口,你可以通过REOBJECT结构中插入的对象。重要的是要注意您可以指定插入点,纵横dwUser变量来存储此对象的标志或任何相关的信息,因此你可以在任何时间更新恢复它.//------- ----------------REOBJECT reoObject =新REOBJECT();reoObject.cp = this._richEdit.TextLength;reoObject.clsid = GUID;reoObject.pstg = pStorage;reoObject.poleobj = Marshal.GetIUnknownForObject(对照组);reoObject.polesite = pOleClientSite;reoObject.dvAspect =(UINT)(DVASPECT.DVASPECT_CONTENT);reoObject.dwFlags =(UINT)(REOOBJECTFLAGS.REO_BELOWBASELINE);reoObject.dwUser = 1; this.IRicEditOle.InsertObject(reoObject);//-----------------------
插入图像,你需要实现接口IDataObject的,在这种情况下,我命名它myDataObject。这个类是一个OLE回调对象使用FORMATETC和STGMEDIUM结构,展示形象,告诉OLE容器,这个对象是一个GDI中位图的剪贴板格式(CF_BITMAP)(TYMED_GDI)公共类myDataObject:IDataObject的。{ 私营位图mBitmap; 公众的FORMATETC mpFormatetc; #地区的IDataObject* 私人常量UINT S_OK = 0; 私人常量UINT E_POINTER = 0x80004003; 私人常量UINT E_NOTIMPL = 0x80004001; 私人常量UINT的E_FAIL = 0X80004005;  ; 公共UINT的GetData(文献的FORMATETC pFormatetc &# 160; 文献的STGMEDIUM pMedium) { IntPtr的hDst = mBitmap.GetHbitmap(); pMedium.tymed =(INT)TYMED.TYMED_GDI; pMedium.unionmember = hDst; pMedium.pUnkForRelease = IntPtr.Zero; (UINT)返回S_OK; } ... ... #endregion 公共myDataObject() { mBitmap =新的Bitmap(16,16); mpFormatetc =新的FORMATETC(); } 公共无效SetImage(字符串strFilename) { 尝试   ; { mBitmap =(位图)Bitmap.FromFile(strFilename,TRUE);   ; / /剪贴板格式= CF_BITMAP & #160; mpFormatetc.cfFormat = CLIPFORMAT.CF_BITMAP; / /目标设备=屏幕 0; mpFormatetc.ptd = IntPtr.Zero; & #160; / /详细程度=全部内容   ; mpFormatetc.dwAspect = DVASPECT.DVASPECT_CONTENT; / /指数= applicaple & #160; mpFormatetc.lindex = -1; 0; / /存储介质的HBITMAP处理 60; mpFormatetc.tymed = TYMED.TYMED_GDI; } 捕获 { } } 公共无效SetImage(Image图像) { 尝试 0; { mBitmap =新的位图(图片); / /剪贴板格式= CF_BITMAP mpFormatetc.cfFormat = CLIPFORMAT.CF_BITMAP; 0; / /目标设备=屏幕 &# 160;mpFormatetc.ptd = IntPtr.Zero;   ; / /详细程度=全部内容 60;mpFormatetc.dwAspect = DVASPECT.DVASPECT_CONTENT; 0; / /指数= applicaple   ; mpFormatetc.lindex = -1; &# 160; / /存储介质的HBITMAP处理 & #160; mpFormatetc.tymed = TYMED.TYMED_GDI; 0; } 捕获 { 0; } }}
看一个成员方法SetImage如何创建一个Bitmap对象,利用其处理时调用的GetData。
现在,这里的对象是如何创建一个共享的全局内存和指针(IStorage)和使用IRichEditOle.public无效InsertMyDataObject(myDataObject MDO)OleClientSite接口插入到RichEditBox{ (MDO == NULL) 60; 回报; //----------------------- ILockBytes pLockBytes; INT SC = CreateILockBytesOnHGlobal(IntPtr.Zero,   ; 真实的,出pLockBytes); IStorage pStorage; SC = StgCreateDocfileOnILockBytes(pLockBytes,(UINT) (STGM.STGM_SHARE_EXCLUSIVE | STGM.STGM_CREATE | &# 160; STGM.STGM_READWRITE) 0,pStorage); 的IOleClientSite pOleClientSite; 60; this.IRichEditOle.GetClientSite(出pOleClientSite); 0; //----------------------- GUID(GUID = Marshal.GenerateGuidForType mdo.GetType()); GUID IID_IOleObject = 新的Guid("{00000112 - 0000 - 0000 - C000 - 000000000046}"); GUID IID_IDataObject = &# 160; 新的Guid("{0000010e - 0000 - 0000 - C000 - 000000000046}"); GUID IID_IUnknown = 新的Guid("{00000000 - 0000 - 0000 - C000 - 000000000046}"); 对象pOleObject; INT HR = OleCreateStaticFromData(MDO,楼盘IID_IOleObject (UINT)OLERENDER.OLERENDER_FORMAT,文献mdo.mpFormatetc pOleClientSite,pStorage出pOleObject); 如果(pOleObject == NULL) 回报; //----------------------- //----------------------- OleSetContainedObject(pOleObject,TRUE);&# 160; REOBJECT reoObject =新REOBJECT(); reoObject.cp = this._richEdit.TextLength; reoObject.clsid = GUID; reoObject.pstg = pStorage; reoObject.poleobj = Marshal.GetIUnknownForObject(pOleObject); 60; reoObject.polesite = pOleClientSite; reoObject.dvAspect =(UINT)(DVASPECT.DVASPECT_CONTENT); reoObject.dwFlags =(UINT)(REOOBJECTFLAGS.REO_BELOWBASELINE); &# 160;reoObject.dwUser = 0; this.IRichEditOle.InsertObject(reoObject); //----------------------- &# 160;//----------------------- Marshal.ReleaseComObject(pLockBytes); Marshal.ReleaseComObject(pOleClientSite); Marshal.ReleaseComObject(pStorage); Marshal.ReleaseComObject(pOleObject); //-----------------------
还有其他的方法来插入控件和ActiveX对象,他们看起来非常相似,上面的方法,所以请查看源代码。兴趣点
最后,如何控制更新?
这是诡计,你需要使用一个计时器,并调用该方法UpdateObjects。此方法执行搜索RichTextBox中的所有对象,如果他们作为特殊标记(在我来说,我使用的dwUser变量),他们将被更新:公共无效UpdateObjects(){ K = this.IRichEditOle.GetObjectCount(); 为(int i = 0;我LT; K,I) { & #160; REOBJECT reoObject =新REOBJECT(); this.IRichEditOle.GetObject(I,reoObject   ; GETOBJECTOPT​​IONS.REO_GETOBJ_ALL_INTERFACES); (reoObject.dwUser == 1) &# 160; { 点PT = this._richEdit.GetPositionFromCharIndex(reoObject.cp); 0; 矩形RECT =新的Rectangle(PT,reoObject.sizel); this._richEdit.Invalidate(RECT,FALSE); / /重绘 } }}
有很多优化这种控制,但现在需要的工作,任何建议表示赞赏。使用代码
要使用的代码,只需添加一个引用的控制,投入正常的RichTextBox的形式,然后更换MyExtRichTextBox类型:MyExtRichTextBox.MyExtRichTextBox richTextBox1的;提示
我通过创建一个控件(按钮和进度条)的阵列,并加入一个计时器的形式,然后调用这样的方法UpdateObjects更新的对象为:private void timer1_Tick(对象发件人,发送System.EventArgs){ 为(int i = 0;我LT; ar.Count;我) { itimer的;   ; 如果(itimer的GT 100) 60; itimer的= 0; 对象obj = AR [我]; 60; 如果(obj是按钮) { & #160; BT =按钮(Button)的OBJ; 0; (bt.Text ="Clickedquot;) & #160; bt.Text ="按钮quot; i.ToString() &# 160; " - "itimer.ToString(); } 其他 & #160; { ProgressBar的PB =(进度条)OBJ; 如果(1 pb.Value GT 100) pb.Value = 0; pb.Value = pb.Value 1; } } richTextBox1.UpdateObjects();}历史版本1.0 - 1 / 2005年11月