/* * RichEdit - functions and interfaces around CreateTextServices * * Copyright 2005, 2006, Maarten Lankhorst * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "config.h" #include "wine/port.h" #define COBJMACROS #include "editor.h" #include "ole2.h" #include "oleauto.h" #include "richole.h" #include "tom.h" #include "imm.h" #include "textserv.h" #include "wine/debug.h" #include "editstr.h" #ifdef __i386__ /* thiscall functions are i386-specific */ #define THISCALL(func) __thiscall_ ## func #define DEFINE_THISCALL_WRAPPER(func,args) \ extern typeof(func) THISCALL(func); \ __ASM_STDCALL_FUNC(__thiscall_ ## func, args, \ "popl %eax\n\t" \ "pushl %ecx\n\t" \ "pushl %eax\n\t" \ "jmp " __ASM_NAME(#func) __ASM_STDCALL(args) ) #else /* __i386__ */ #define THISCALL(func) func #define DEFINE_THISCALL_WRAPPER(func,args) /* nothing */ #endif /* __i386__ */ WINE_DEFAULT_DEBUG_CHANNEL(richedit); typedef struct ITextServicesImpl { IUnknown IUnknown_inner; ITextServices ITextServices_iface; IUnknown *outer_unk; LONG ref; ITextHost *pMyHost; CRITICAL_SECTION csTxtSrv; ME_TextEditor *editor; char spare[256]; } ITextServicesImpl; static inline ITextServicesImpl *impl_from_IUnknown(IUnknown *iface) { return CONTAINING_RECORD(iface, ITextServicesImpl, IUnknown_inner); } static HRESULT WINAPI ITextServicesImpl_QueryInterface(IUnknown *iface, REFIID riid, void **ppv) { ITextServicesImpl *This = impl_from_IUnknown(iface); TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), ppv); if (IsEqualIID(riid, &IID_IUnknown)) *ppv = &This->IUnknown_inner; else if (IsEqualIID(riid, &IID_ITextServices)) *ppv = &This->ITextServices_iface; else if (IsEqualIID(riid, &IID_IRichEditOle) || IsEqualIID(riid, &IID_ITextDocument)) { if (!This->editor->reOle) if (!CreateIRichEditOle(This->outer_unk, This->editor, (void **)(&This->editor->reOle))) return E_OUTOFMEMORY; if (IsEqualIID(riid, &IID_ITextDocument)) ME_GetITextDocumentInterface(This->editor->reOle, ppv); else *ppv = This->editor->reOle; } else { *ppv = NULL; FIXME("Unknown interface: %s\n", debugstr_guid(riid)); return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI ITextServicesImpl_AddRef(IUnknown *iface) { ITextServicesImpl *This = impl_from_IUnknown(iface); LONG ref = InterlockedIncrement(&This->ref); TRACE("(%p) ref=%d\n", This, ref); return ref; } static ULONG WINAPI ITextServicesImpl_Release(IUnknown *iface) { ITextServicesImpl *This = impl_from_IUnknown(iface); LONG ref = InterlockedDecrement(&This->ref); TRACE("(%p) ref=%d\n", This, ref); if (!ref) { ME_DestroyEditor(This->editor); This->csTxtSrv.DebugInfo->Spare[0] = 0; DeleteCriticalSection(&This->csTxtSrv); CoTaskMemFree(This); } return ref; } static const IUnknownVtbl textservices_inner_vtbl = { ITextServicesImpl_QueryInterface, ITextServicesImpl_AddRef, ITextServicesImpl_Release }; static inline ITextServicesImpl *impl_from_ITextServices(ITextServices *iface) { return CONTAINING_RECORD(iface, ITextServicesImpl, ITextServices_iface); } static HRESULT WINAPI fnTextSrv_QueryInterface(ITextServices *iface, REFIID riid, void **ppv) { ITextServicesImpl *This = impl_from_ITextServices(iface); return IUnknown_QueryInterface(This->outer_unk, riid, ppv); } static ULONG WINAPI fnTextSrv_AddRef(ITextServices *iface) { ITextServicesImpl *This = impl_from_ITextServices(iface); return IUnknown_AddRef(This->outer_unk); } static ULONG WINAPI fnTextSrv_Release(ITextServices *iface) { ITextServicesImpl *This = impl_from_ITextServices(iface); return IUnknown_Release(This->outer_unk); } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxSendMessage(ITextServices *iface, UINT msg, WPARAM wparam, LPARAM lparam, LRESULT *plresult) { ITextServicesImpl *This = impl_from_ITextServices(iface); HRESULT hresult; LRESULT lresult; lresult = ME_HandleMessage(This->editor, msg, wparam, lparam, TRUE, &hresult); if (plresult) *plresult = lresult; return hresult; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxDraw(ITextServices *iface, DWORD dwDrawAspect, LONG lindex, void *pvAspect, DVTARGETDEVICE *ptd, HDC hdcDraw, HDC hdcTargetDev, LPCRECTL lprcBounds, LPCRECTL lprcWBounds, LPRECT lprcUpdate, BOOL (CALLBACK * pfnContinue)(DWORD), DWORD dwContinue, LONG lViewId) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetHScroll(ITextServices *iface, LONG *plMin, LONG *plMax, LONG *plPos, LONG *plPage, BOOL *pfEnabled) { ITextServicesImpl *This = impl_from_ITextServices(iface); if (plMin) *plMin = This->editor->horz_si.nMin; if (plMax) *plMax = This->editor->horz_si.nMax; if (plPos) *plPos = This->editor->horz_si.nPos; if (plPage) *plPage = This->editor->horz_si.nPage; if (pfEnabled) *pfEnabled = (This->editor->styleFlags & WS_HSCROLL) != 0; return S_OK; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetVScroll(ITextServices *iface, LONG *plMin, LONG *plMax, LONG *plPos, LONG *plPage, BOOL *pfEnabled) { ITextServicesImpl *This = impl_from_ITextServices(iface); if (plMin) *plMin = This->editor->vert_si.nMin; if (plMax) *plMax = This->editor->vert_si.nMax; if (plPos) *plPos = This->editor->vert_si.nPos; if (plPage) *plPage = This->editor->vert_si.nPage; if (pfEnabled) *pfEnabled = (This->editor->styleFlags & WS_VSCROLL) != 0; return S_OK; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxSetCursor(ITextServices *iface, DWORD dwDrawAspect, LONG lindex, void *pvAspect, DVTARGETDEVICE *ptd, HDC hdcDraw, HDC hicTargetDev, LPCRECT lprcClient, INT x, INT y) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxQueryHitPoint(ITextServices *iface, DWORD dwDrawAspect, LONG lindex, void *pvAspect, DVTARGETDEVICE *ptd, HDC hdcDraw, HDC hicTargetDev, LPCRECT lprcClient, INT x, INT y, DWORD *pHitResult) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxInplaceActivate(ITextServices *iface, LPCRECT prcClient) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxInplaceDeactivate(ITextServices *iface) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxUIActivate(ITextServices *iface) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxUIDeactivate(ITextServices *iface) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetText(ITextServices *iface, BSTR *pbstrText) { ITextServicesImpl *This = impl_from_ITextServices(iface); int length; length = ME_GetTextLength(This->editor); if (length) { ME_Cursor start; BSTR bstr; bstr = SysAllocStringByteLen(NULL, length * sizeof(WCHAR)); if (bstr == NULL) return E_OUTOFMEMORY; ME_CursorFromCharOfs(This->editor, 0, &start); ME_GetTextW(This->editor, bstr, length, &start, INT_MAX, FALSE, FALSE); *pbstrText = bstr; } else { *pbstrText = NULL; } return S_OK; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxSetText(ITextServices *iface, LPCWSTR pszText) { ITextServicesImpl *This = impl_from_ITextServices(iface); ME_Cursor cursor; ME_SetCursorToStart(This->editor, &cursor); ME_InternalDeleteText(This->editor, &cursor, ME_GetTextLength(This->editor), FALSE); if(pszText) ME_InsertTextFromCursor(This->editor, 0, pszText, -1, This->editor->pBuffer->pDefaultStyle); ME_SetSelection(This->editor, 0, 0); This->editor->nModifyStep = 0; OleFlushClipboard(); ME_EmptyUndoStack(This->editor); ME_UpdateRepaint(This->editor, FALSE); return S_OK; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetCurTargetX(ITextServices *iface, LONG *x) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetBaseLinePos(ITextServices *iface, LONG *x) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetNaturalSize(ITextServices *iface, DWORD dwAspect, HDC hdcDraw, HDC hicTargetDev, DVTARGETDEVICE *ptd, DWORD dwMode, const SIZEL *psizelExtent, LONG *pwidth, LONG *pheight) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetDropTarget(ITextServices *iface, IDropTarget **ppDropTarget) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_OnTxPropertyBitsChange(ITextServices *iface, DWORD dwMask, DWORD dwBits) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DECLSPEC_HIDDEN HRESULT WINAPI fnTextSrv_TxGetCachedSize(ITextServices *iface, DWORD *pdwWidth, DWORD *pdwHeight) { ITextServicesImpl *This = impl_from_ITextServices(iface); FIXME("%p: STUB\n", This); return E_NOTIMPL; } DEFINE_THISCALL_WRAPPER(fnTextSrv_TxSendMessage,20) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxDraw,52) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetHScroll,24) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetVScroll,24) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxSetCursor,40) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxQueryHitPoint,44) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxInplaceActivate,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxInplaceDeactivate,4) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxUIActivate,4) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxUIDeactivate,4) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetText,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxSetText,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetCurTargetX,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetBaseLinePos,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetNaturalSize,36) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetDropTarget,8) DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxPropertyBitsChange,12) DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetCachedSize,12) static const ITextServicesVtbl textservices_vtbl = { fnTextSrv_QueryInterface, fnTextSrv_AddRef, fnTextSrv_Release, THISCALL(fnTextSrv_TxSendMessage), THISCALL(fnTextSrv_TxDraw), THISCALL(fnTextSrv_TxGetHScroll), THISCALL(fnTextSrv_TxGetVScroll), THISCALL(fnTextSrv_OnTxSetCursor), THISCALL(fnTextSrv_TxQueryHitPoint), THISCALL(fnTextSrv_OnTxInplaceActivate), THISCALL(fnTextSrv_OnTxInplaceDeactivate), THISCALL(fnTextSrv_OnTxUIActivate), THISCALL(fnTextSrv_OnTxUIDeactivate), THISCALL(fnTextSrv_TxGetText), THISCALL(fnTextSrv_TxSetText), THISCALL(fnTextSrv_TxGetCurTargetX), THISCALL(fnTextSrv_TxGetBaseLinePos), THISCALL(fnTextSrv_TxGetNaturalSize), THISCALL(fnTextSrv_TxGetDropTarget), THISCALL(fnTextSrv_OnTxPropertyBitsChange), THISCALL(fnTextSrv_TxGetCachedSize) }; /****************************************************************** * CreateTextServices (RICHED20.4) */ HRESULT WINAPI CreateTextServices(IUnknown *pUnkOuter, ITextHost *pITextHost, IUnknown **ppUnk) { ITextServicesImpl *ITextImpl; TRACE("%p %p --> %p\n", pUnkOuter, pITextHost, ppUnk); if (pITextHost == NULL) return E_POINTER; ITextImpl = CoTaskMemAlloc(sizeof(*ITextImpl)); if (ITextImpl == NULL) return E_OUTOFMEMORY; InitializeCriticalSection(&ITextImpl->csTxtSrv); ITextImpl->csTxtSrv.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": ITextServicesImpl.csTxtSrv"); ITextImpl->ref = 1; ITextHost_AddRef(pITextHost); ITextImpl->pMyHost = pITextHost; ITextImpl->IUnknown_inner.lpVtbl = &textservices_inner_vtbl; ITextImpl->ITextServices_iface.lpVtbl = &textservices_vtbl; ITextImpl->editor = ME_MakeEditor(pITextHost, FALSE); if (pUnkOuter) ITextImpl->outer_unk = pUnkOuter; else ITextImpl->outer_unk = &ITextImpl->IUnknown_inner; *ppUnk = &ITextImpl->IUnknown_inner; return S_OK; }