2 * RichEdit - functions and interfaces around CreateTextServices
4 * Copyright 2005, 2006, Maarten Lankhorst
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24 #ifdef __i386__ /* thiscall functions are i386-specific */
26 #define THISCALL(func) __thiscall_ ## func
27 #define DEFINE_THISCALL_WRAPPER(func,args) \
28 extern typeof(func) THISCALL(func); \
29 __ASM_STDCALL_FUNC(__thiscall_ ## func, args, \
33 "jmp " __ASM_NAME(#func) __ASM_STDCALL(args) )
36 #define THISCALL(func) func
37 #define DEFINE_THISCALL_WRAPPER(func,args) /* nothing */
41 WINE_DEFAULT_DEBUG_CHANNEL(richedit
);
43 typedef struct ITextServicesImpl
{
44 IUnknown IUnknown_inner
;
45 ITextServices ITextServices_iface
;
49 CRITICAL_SECTION csTxtSrv
;
50 ME_TextEditor
*editor
;
54 static inline ITextServicesImpl
*impl_from_IUnknown(IUnknown
*iface
)
56 return CONTAINING_RECORD(iface
, ITextServicesImpl
, IUnknown_inner
);
59 static HRESULT WINAPI
ITextServicesImpl_QueryInterface(IUnknown
*iface
, REFIID riid
, void **ppv
)
61 ITextServicesImpl
*This
= impl_from_IUnknown(iface
);
63 TRACE("(%p)->(%s, %p)\n", iface
, debugstr_guid(riid
), ppv
);
65 if (IsEqualIID(riid
, &IID_IUnknown
))
66 *ppv
= &This
->IUnknown_inner
;
67 else if (IsEqualIID(riid
, &IID_ITextServices
))
68 *ppv
= &This
->ITextServices_iface
;
69 else if (IsEqualIID(riid
, &IID_IRichEditOle
) || IsEqualIID(riid
, &IID_ITextDocument
)) {
70 if (!This
->editor
->reOle
)
71 if (!CreateIRichEditOle(This
->outer_unk
, This
->editor
, (void **)(&This
->editor
->reOle
)))
73 if (IsEqualIID(riid
, &IID_ITextDocument
))
74 ME_GetITextDocumentInterface(This
->editor
->reOle
, ppv
);
76 *ppv
= This
->editor
->reOle
;
79 FIXME("Unknown interface: %s\n", debugstr_guid(riid
));
83 IUnknown_AddRef((IUnknown
*)*ppv
);
87 static ULONG WINAPI
ITextServicesImpl_AddRef(IUnknown
*iface
)
89 ITextServicesImpl
*This
= impl_from_IUnknown(iface
);
90 LONG ref
= InterlockedIncrement(&This
->ref
);
92 TRACE("(%p) ref=%d\n", This
, ref
);
97 static ULONG WINAPI
ITextServicesImpl_Release(IUnknown
*iface
)
99 ITextServicesImpl
*This
= impl_from_IUnknown(iface
);
100 LONG ref
= InterlockedDecrement(&This
->ref
);
102 TRACE("(%p) ref=%d\n", This
, ref
);
106 ME_DestroyEditor(This
->editor
);
107 This
->csTxtSrv
.DebugInfo
->Spare
[0] = 0;
108 DeleteCriticalSection(&This
->csTxtSrv
);
114 static const IUnknownVtbl textservices_inner_vtbl
=
116 ITextServicesImpl_QueryInterface
,
117 ITextServicesImpl_AddRef
,
118 ITextServicesImpl_Release
121 static inline ITextServicesImpl
*impl_from_ITextServices(ITextServices
*iface
)
123 return CONTAINING_RECORD(iface
, ITextServicesImpl
, ITextServices_iface
);
126 static HRESULT WINAPI
fnTextSrv_QueryInterface(ITextServices
*iface
, REFIID riid
, void **ppv
)
128 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
129 return IUnknown_QueryInterface(This
->outer_unk
, riid
, ppv
);
132 static ULONG WINAPI
fnTextSrv_AddRef(ITextServices
*iface
)
134 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
135 return IUnknown_AddRef(This
->outer_unk
);
138 static ULONG WINAPI
fnTextSrv_Release(ITextServices
*iface
)
140 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
141 return IUnknown_Release(This
->outer_unk
);
144 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxSendMessage(ITextServices
*iface
, UINT msg
, WPARAM wparam
,
145 LPARAM lparam
, LRESULT
*plresult
)
147 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
151 lresult
= ME_HandleMessage(This
->editor
, msg
, wparam
, lparam
, TRUE
, &hresult
);
152 if (plresult
) *plresult
= lresult
;
156 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxDraw(ITextServices
*iface
, DWORD dwDrawAspect
, LONG lindex
,
157 void *pvAspect
, DVTARGETDEVICE
*ptd
, HDC hdcDraw
, HDC hdcTargetDev
,
158 LPCRECTL lprcBounds
, LPCRECTL lprcWBounds
, LPRECT lprcUpdate
,
159 BOOL (CALLBACK
* pfnContinue
)(DWORD
), DWORD dwContinue
,
162 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
164 FIXME("%p: STUB\n", This
);
168 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetHScroll(ITextServices
*iface
, LONG
*plMin
, LONG
*plMax
, LONG
*plPos
,
169 LONG
*plPage
, BOOL
*pfEnabled
)
171 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
173 *plMin
= This
->editor
->horz_si
.nMin
;
174 *plMax
= This
->editor
->horz_si
.nMax
;
175 *plPos
= This
->editor
->horz_si
.nPos
;
176 *plPage
= This
->editor
->horz_si
.nPage
;
177 *pfEnabled
= (This
->editor
->styleFlags
& WS_HSCROLL
) != 0;
181 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetVScroll(ITextServices
*iface
, LONG
*plMin
, LONG
*plMax
, LONG
*plPos
,
182 LONG
*plPage
, BOOL
*pfEnabled
)
184 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
186 *plMin
= This
->editor
->vert_si
.nMin
;
187 *plMax
= This
->editor
->vert_si
.nMax
;
188 *plPos
= This
->editor
->vert_si
.nPos
;
189 *plPage
= This
->editor
->vert_si
.nPage
;
190 *pfEnabled
= (This
->editor
->styleFlags
& WS_VSCROLL
) != 0;
194 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxSetCursor(ITextServices
*iface
, DWORD dwDrawAspect
, LONG lindex
,
195 void *pvAspect
, DVTARGETDEVICE
*ptd
, HDC hdcDraw
,
196 HDC hicTargetDev
, LPCRECT lprcClient
, INT x
, INT y
)
198 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
200 FIXME("%p: STUB\n", This
);
204 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxQueryHitPoint(ITextServices
*iface
, DWORD dwDrawAspect
, LONG lindex
,
205 void *pvAspect
, DVTARGETDEVICE
*ptd
, HDC hdcDraw
,
206 HDC hicTargetDev
, LPCRECT lprcClient
, INT x
, INT y
,
209 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
211 FIXME("%p: STUB\n", This
);
215 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxInplaceActivate(ITextServices
*iface
, LPCRECT prcClient
)
217 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
219 FIXME("%p: STUB\n", This
);
223 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxInplaceDeactivate(ITextServices
*iface
)
225 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
227 FIXME("%p: STUB\n", This
);
231 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxUIActivate(ITextServices
*iface
)
233 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
235 FIXME("%p: STUB\n", This
);
239 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxUIDeactivate(ITextServices
*iface
)
241 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
243 FIXME("%p: STUB\n", This
);
247 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetText(ITextServices
*iface
, BSTR
*pbstrText
)
249 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
252 length
= ME_GetTextLength(This
->editor
);
257 bstr
= SysAllocStringByteLen(NULL
, length
* sizeof(WCHAR
));
259 return E_OUTOFMEMORY
;
261 ME_CursorFromCharOfs(This
->editor
, 0, &start
);
262 ME_GetTextW(This
->editor
, bstr
, length
, &start
, INT_MAX
, FALSE
, FALSE
);
271 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxSetText(ITextServices
*iface
, LPCWSTR pszText
)
273 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
276 ME_SetCursorToStart(This
->editor
, &cursor
);
277 ME_InternalDeleteText(This
->editor
, &cursor
, ME_GetTextLength(This
->editor
), FALSE
);
279 ME_InsertTextFromCursor(This
->editor
, 0, pszText
, -1, This
->editor
->pBuffer
->pDefaultStyle
);
280 ME_SetSelection(This
->editor
, 0, 0);
281 This
->editor
->nModifyStep
= 0;
283 ME_EmptyUndoStack(This
->editor
);
284 ME_UpdateRepaint(This
->editor
, FALSE
);
289 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetCurTargetX(ITextServices
*iface
, LONG
*x
)
291 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
293 FIXME("%p: STUB\n", This
);
297 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetBaseLinePos(ITextServices
*iface
, LONG
*x
)
299 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
301 FIXME("%p: STUB\n", This
);
305 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetNaturalSize(ITextServices
*iface
, DWORD dwAspect
, HDC hdcDraw
,
306 HDC hicTargetDev
, DVTARGETDEVICE
*ptd
, DWORD dwMode
,
307 const SIZEL
*psizelExtent
, LONG
*pwidth
, LONG
*pheight
)
309 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
311 FIXME("%p: STUB\n", This
);
315 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetDropTarget(ITextServices
*iface
, IDropTarget
**ppDropTarget
)
317 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
319 FIXME("%p: STUB\n", This
);
323 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_OnTxPropertyBitsChange(ITextServices
*iface
, DWORD dwMask
, DWORD dwBits
)
325 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
327 FIXME("%p: STUB\n", This
);
331 DECLSPEC_HIDDEN HRESULT WINAPI
fnTextSrv_TxGetCachedSize(ITextServices
*iface
, DWORD
*pdwWidth
, DWORD
*pdwHeight
)
333 ITextServicesImpl
*This
= impl_from_ITextServices(iface
);
335 FIXME("%p: STUB\n", This
);
339 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxSendMessage
,20)
340 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxDraw
,52)
341 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetHScroll
,24)
342 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetVScroll
,24)
343 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxSetCursor
,40)
344 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxQueryHitPoint
,44)
345 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxInplaceActivate
,8)
346 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxInplaceDeactivate
,4)
347 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxUIActivate
,4)
348 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxUIDeactivate
,4)
349 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetText
,8)
350 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxSetText
,8)
351 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetCurTargetX
,8)
352 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetBaseLinePos
,8)
353 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetNaturalSize
,36)
354 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetDropTarget
,8)
355 DEFINE_THISCALL_WRAPPER(fnTextSrv_OnTxPropertyBitsChange
,12)
356 DEFINE_THISCALL_WRAPPER(fnTextSrv_TxGetCachedSize
,12)
358 static const ITextServicesVtbl textservices_vtbl
=
360 fnTextSrv_QueryInterface
,
363 THISCALL(fnTextSrv_TxSendMessage
),
364 THISCALL(fnTextSrv_TxDraw
),
365 THISCALL(fnTextSrv_TxGetHScroll
),
366 THISCALL(fnTextSrv_TxGetVScroll
),
367 THISCALL(fnTextSrv_OnTxSetCursor
),
368 THISCALL(fnTextSrv_TxQueryHitPoint
),
369 THISCALL(fnTextSrv_OnTxInplaceActivate
),
370 THISCALL(fnTextSrv_OnTxInplaceDeactivate
),
371 THISCALL(fnTextSrv_OnTxUIActivate
),
372 THISCALL(fnTextSrv_OnTxUIDeactivate
),
373 THISCALL(fnTextSrv_TxGetText
),
374 THISCALL(fnTextSrv_TxSetText
),
375 THISCALL(fnTextSrv_TxGetCurTargetX
),
376 THISCALL(fnTextSrv_TxGetBaseLinePos
),
377 THISCALL(fnTextSrv_TxGetNaturalSize
),
378 THISCALL(fnTextSrv_TxGetDropTarget
),
379 THISCALL(fnTextSrv_OnTxPropertyBitsChange
),
380 THISCALL(fnTextSrv_TxGetCachedSize
)
383 /******************************************************************
384 * CreateTextServices (RICHED20.4)
386 HRESULT WINAPI
CreateTextServices(IUnknown
*pUnkOuter
, ITextHost
*pITextHost
, IUnknown
**ppUnk
)
388 ITextServicesImpl
*ITextImpl
;
390 TRACE("%p %p --> %p\n", pUnkOuter
, pITextHost
, ppUnk
);
391 if (pITextHost
== NULL
)
394 ITextImpl
= CoTaskMemAlloc(sizeof(*ITextImpl
));
395 if (ITextImpl
== NULL
)
396 return E_OUTOFMEMORY
;
397 InitializeCriticalSection(&ITextImpl
->csTxtSrv
);
398 ITextImpl
->csTxtSrv
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": ITextServicesImpl.csTxtSrv");
400 ITextHost_AddRef(pITextHost
);
401 ITextImpl
->pMyHost
= pITextHost
;
402 ITextImpl
->IUnknown_inner
.lpVtbl
= &textservices_inner_vtbl
;
403 ITextImpl
->ITextServices_iface
.lpVtbl
= &textservices_vtbl
;
404 ITextImpl
->editor
= ME_MakeEditor(pITextHost
, FALSE
, ES_LEFT
);
405 ITextImpl
->editor
->exStyleFlags
= 0;
406 SetRectEmpty(&ITextImpl
->editor
->rcFormat
);
409 ITextImpl
->outer_unk
= pUnkOuter
;
411 ITextImpl
->outer_unk
= &ITextImpl
->IUnknown_inner
;
413 *ppUnk
= &ITextImpl
->IUnknown_inner
;