xref: /reactos/dll/win32/msctf/documentmgr.c (revision b8dd046e)
1 /*
2  *  ITfDocumentMgr implementation
3  *
4  *  Copyright 2009 Aric Stewart, CodeWeavers
5  *
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.
10  *
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.
15  *
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
19  */
20 
21 #include <stdarg.h>
22 
23 #define COBJMACROS
24 
25 #include "wine/debug.h"
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winreg.h"
29 #include "winuser.h"
30 #include "shlwapi.h"
31 #include "winerror.h"
32 #include "objbase.h"
33 
34 #include "msctf.h"
35 #include "msctf_internal.h"
36 
37 WINE_DEFAULT_DEBUG_CHANNEL(msctf);
38 
39 typedef struct tagDocumentMgr {
40     ITfDocumentMgr ITfDocumentMgr_iface;
41     ITfSource ITfSource_iface;
42     LONG refCount;
43 
44     /* Aggregation */
45     ITfCompartmentMgr  *CompartmentMgr;
46 
47     ITfContext*  contextStack[2]; /* limit of 2 contexts */
48     ITfThreadMgrEventSink* ThreadMgrSink;
49 
50     struct list TransitoryExtensionSink;
51 } DocumentMgr;
52 
53 typedef struct tagEnumTfContext {
54     IEnumTfContexts IEnumTfContexts_iface;
55     LONG refCount;
56 
57     DWORD   index;
58     DocumentMgr *docmgr;
59 } EnumTfContext;
60 
61 static HRESULT EnumTfContext_Constructor(DocumentMgr* mgr, IEnumTfContexts **ppOut);
62 
63 static inline DocumentMgr *impl_from_ITfDocumentMgr(ITfDocumentMgr *iface)
64 {
65     return CONTAINING_RECORD(iface, DocumentMgr, ITfDocumentMgr_iface);
66 }
67 
68 static inline DocumentMgr *impl_from_ITfSource(ITfSource *iface)
69 {
70     return CONTAINING_RECORD(iface, DocumentMgr, ITfSource_iface);
71 }
72 
73 static inline EnumTfContext *impl_from_IEnumTfContexts(IEnumTfContexts *iface)
74 {
75     return CONTAINING_RECORD(iface, EnumTfContext, IEnumTfContexts_iface);
76 }
77 
78 static void DocumentMgr_Destructor(DocumentMgr *This)
79 {
80     ITfThreadMgr *tm = NULL;
81     TRACE("destroying %p\n", This);
82 
83     TF_GetThreadMgr(&tm);
84     if (tm)
85     {
86         ThreadMgr_OnDocumentMgrDestruction(tm, &This->ITfDocumentMgr_iface);
87         ITfThreadMgr_Release(tm);
88     }
89 
90     if (This->contextStack[0])
91         ITfContext_Release(This->contextStack[0]);
92     if (This->contextStack[1])
93         ITfContext_Release(This->contextStack[1]);
94     free_sinks(&This->TransitoryExtensionSink);
95     CompartmentMgr_Destructor(This->CompartmentMgr);
96     HeapFree(GetProcessHeap(),0,This);
97 }
98 
99 static HRESULT WINAPI DocumentMgr_QueryInterface(ITfDocumentMgr *iface, REFIID iid, LPVOID *ppvOut)
100 {
101     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
102     *ppvOut = NULL;
103 
104     if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfDocumentMgr))
105     {
106         *ppvOut = &This->ITfDocumentMgr_iface;
107     }
108     else if (IsEqualIID(iid, &IID_ITfSource))
109     {
110         *ppvOut = &This->ITfSource_iface;
111     }
112     else if (IsEqualIID(iid, &IID_ITfCompartmentMgr))
113     {
114         *ppvOut = This->CompartmentMgr;
115     }
116 
117     if (*ppvOut)
118     {
119         ITfDocumentMgr_AddRef(iface);
120         return S_OK;
121     }
122 
123     WARN("unsupported interface: %s\n", debugstr_guid(iid));
124     return E_NOINTERFACE;
125 }
126 
127 static ULONG WINAPI DocumentMgr_AddRef(ITfDocumentMgr *iface)
128 {
129     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
130     return InterlockedIncrement(&This->refCount);
131 }
132 
133 static ULONG WINAPI DocumentMgr_Release(ITfDocumentMgr *iface)
134 {
135     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
136     ULONG ret;
137 
138     ret = InterlockedDecrement(&This->refCount);
139     if (ret == 0)
140         DocumentMgr_Destructor(This);
141     return ret;
142 }
143 
144 /*****************************************************
145  * ITfDocumentMgr functions
146  *****************************************************/
147 static HRESULT WINAPI DocumentMgr_CreateContext(ITfDocumentMgr *iface,
148         TfClientId tidOwner,
149         DWORD dwFlags, IUnknown *punk, ITfContext **ppic,
150         TfEditCookie *pecTextStore)
151 {
152     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
153     TRACE("(%p) 0x%x 0x%x %p %p %p\n",This,tidOwner,dwFlags,punk,ppic,pecTextStore);
154     return Context_Constructor(tidOwner, punk, iface, ppic, pecTextStore);
155 }
156 
157 static HRESULT WINAPI DocumentMgr_Push(ITfDocumentMgr *iface, ITfContext *pic)
158 {
159     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
160     ITfContext *check;
161 
162     TRACE("(%p) %p\n",This,pic);
163 
164     if (This->contextStack[1])  /* FUll */
165         return TF_E_STACKFULL;
166 
167     if (!pic || FAILED(ITfContext_QueryInterface(pic,&IID_ITfContext,(LPVOID*) &check)))
168         return E_INVALIDARG;
169 
170     if (This->contextStack[0] == NULL)
171         ITfThreadMgrEventSink_OnInitDocumentMgr(This->ThreadMgrSink,iface);
172 
173     This->contextStack[1] = This->contextStack[0];
174     This->contextStack[0] = check;
175 
176     Context_Initialize(check, iface);
177     ITfThreadMgrEventSink_OnPushContext(This->ThreadMgrSink,check);
178 
179     return S_OK;
180 }
181 
182 static HRESULT WINAPI DocumentMgr_Pop(ITfDocumentMgr *iface, DWORD dwFlags)
183 {
184     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
185     TRACE("(%p) 0x%x\n",This,dwFlags);
186 
187     if (dwFlags == TF_POPF_ALL)
188     {
189         int i;
190 
191         for (i = 0; i < ARRAY_SIZE(This->contextStack); i++)
192             if (This->contextStack[i])
193             {
194                 ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink, This->contextStack[i]);
195                 Context_Uninitialize(This->contextStack[i]);
196                 ITfContext_Release(This->contextStack[i]);
197                 This->contextStack[i] = NULL;
198             }
199 
200         ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface);
201         return S_OK;
202     }
203 
204     if (dwFlags)
205         return E_INVALIDARG;
206 
207     if (This->contextStack[1] == NULL) /* Cannot pop last context */
208         return E_FAIL;
209 
210     ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[0]);
211     Context_Uninitialize(This->contextStack[0]);
212     ITfContext_Release(This->contextStack[0]);
213     This->contextStack[0] = This->contextStack[1];
214     This->contextStack[1] = NULL;
215 
216     if (This->contextStack[0] == NULL)
217         ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface);
218 
219     return S_OK;
220 }
221 
222 static HRESULT WINAPI DocumentMgr_GetTop(ITfDocumentMgr *iface, ITfContext **ppic)
223 {
224     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
225     TRACE("(%p)\n",This);
226     if (!ppic)
227         return E_INVALIDARG;
228 
229     if (This->contextStack[0])
230         ITfContext_AddRef(This->contextStack[0]);
231 
232     *ppic = This->contextStack[0];
233 
234     return S_OK;
235 }
236 
237 static HRESULT WINAPI DocumentMgr_GetBase(ITfDocumentMgr *iface, ITfContext **ppic)
238 {
239     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
240     ITfContext *tgt;
241 
242     TRACE("(%p)\n",This);
243     if (!ppic)
244         return E_INVALIDARG;
245 
246     if (This->contextStack[1])
247         tgt = This->contextStack[1];
248     else
249         tgt = This->contextStack[0];
250 
251     if (tgt)
252         ITfContext_AddRef(tgt);
253 
254     *ppic = tgt;
255 
256     return S_OK;
257 }
258 
259 static HRESULT WINAPI DocumentMgr_EnumContexts(ITfDocumentMgr *iface, IEnumTfContexts **ppEnum)
260 {
261     DocumentMgr *This = impl_from_ITfDocumentMgr(iface);
262     TRACE("(%p) %p\n",This,ppEnum);
263     return EnumTfContext_Constructor(This, ppEnum);
264 }
265 
266 static const ITfDocumentMgrVtbl DocumentMgrVtbl =
267 {
268     DocumentMgr_QueryInterface,
269     DocumentMgr_AddRef,
270     DocumentMgr_Release,
271     DocumentMgr_CreateContext,
272     DocumentMgr_Push,
273     DocumentMgr_Pop,
274     DocumentMgr_GetTop,
275     DocumentMgr_GetBase,
276     DocumentMgr_EnumContexts
277 };
278 
279 static HRESULT WINAPI DocumentMgrSource_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut)
280 {
281     DocumentMgr *This = impl_from_ITfSource(iface);
282     return ITfDocumentMgr_QueryInterface(&This->ITfDocumentMgr_iface, iid, ppvOut);
283 }
284 
285 static ULONG WINAPI DocumentMgrSource_AddRef(ITfSource *iface)
286 {
287     DocumentMgr *This = impl_from_ITfSource(iface);
288     return ITfDocumentMgr_AddRef(&This->ITfDocumentMgr_iface);
289 }
290 
291 static ULONG WINAPI DocumentMgrSource_Release(ITfSource *iface)
292 {
293     DocumentMgr *This = impl_from_ITfSource(iface);
294     return ITfDocumentMgr_Release(&This->ITfDocumentMgr_iface);
295 }
296 
297 /*****************************************************
298  * ITfSource functions
299  *****************************************************/
300 static HRESULT WINAPI DocumentMgrSource_AdviseSink(ITfSource *iface,
301         REFIID riid, IUnknown *punk, DWORD *pdwCookie)
302 {
303     DocumentMgr *This = impl_from_ITfSource(iface);
304 
305     TRACE("(%p) %s %p %p\n", This, debugstr_guid(riid), punk, pdwCookie);
306 
307     if (!riid || !punk || !pdwCookie)
308         return E_INVALIDARG;
309 
310     if (IsEqualIID(riid, &IID_ITfTransitoryExtensionSink))
311     {
312         WARN("semi-stub for ITfTransitoryExtensionSink: callback won't be used.\n");
313         return advise_sink(&This->TransitoryExtensionSink, &IID_ITfTransitoryExtensionSink,
314                            COOKIE_MAGIC_DMSINK, punk, pdwCookie);
315     }
316 
317     FIXME("(%p) Unhandled Sink: %s\n",This,debugstr_guid(riid));
318     return E_NOTIMPL;
319 }
320 
321 static HRESULT WINAPI DocumentMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie)
322 {
323     DocumentMgr *This = impl_from_ITfSource(iface);
324 
325     TRACE("(%p) %x\n",This,pdwCookie);
326 
327     if (get_Cookie_magic(pdwCookie)!=COOKIE_MAGIC_DMSINK)
328         return E_INVALIDARG;
329 
330     return unadvise_sink(pdwCookie);
331 }
332 
333 static const ITfSourceVtbl DocumentMgrSourceVtbl =
334 {
335     DocumentMgrSource_QueryInterface,
336     DocumentMgrSource_AddRef,
337     DocumentMgrSource_Release,
338     DocumentMgrSource_AdviseSink,
339     DocumentMgrSource_UnadviseSink,
340 };
341 
342 HRESULT DocumentMgr_Constructor(ITfThreadMgrEventSink *ThreadMgrSink, ITfDocumentMgr **ppOut)
343 {
344     DocumentMgr *This;
345 
346     This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(DocumentMgr));
347     if (This == NULL)
348         return E_OUTOFMEMORY;
349 
350     This->ITfDocumentMgr_iface.lpVtbl = &DocumentMgrVtbl;
351     This->ITfSource_iface.lpVtbl = &DocumentMgrSourceVtbl;
352     This->refCount = 1;
353     This->ThreadMgrSink = ThreadMgrSink;
354     list_init(&This->TransitoryExtensionSink);
355 
356     CompartmentMgr_Constructor((IUnknown*)&This->ITfDocumentMgr_iface, &IID_IUnknown, (IUnknown**)&This->CompartmentMgr);
357 
358     *ppOut = &This->ITfDocumentMgr_iface;
359     TRACE("returning %p\n", *ppOut);
360     return S_OK;
361 }
362 
363 /**************************************************
364  * IEnumTfContexts implementation
365  **************************************************/
366 static void EnumTfContext_Destructor(EnumTfContext *This)
367 {
368     TRACE("destroying %p\n", This);
369     HeapFree(GetProcessHeap(),0,This);
370 }
371 
372 static HRESULT WINAPI EnumTfContext_QueryInterface(IEnumTfContexts *iface, REFIID iid, LPVOID *ppvOut)
373 {
374     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
375     *ppvOut = NULL;
376 
377     if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_IEnumTfContexts))
378     {
379         *ppvOut = &This->IEnumTfContexts_iface;
380     }
381 
382     if (*ppvOut)
383     {
384         IEnumTfContexts_AddRef(iface);
385         return S_OK;
386     }
387 
388     WARN("unsupported interface: %s\n", debugstr_guid(iid));
389     return E_NOINTERFACE;
390 }
391 
392 static ULONG WINAPI EnumTfContext_AddRef(IEnumTfContexts *iface)
393 {
394     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
395     return InterlockedIncrement(&This->refCount);
396 }
397 
398 static ULONG WINAPI EnumTfContext_Release(IEnumTfContexts *iface)
399 {
400     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
401     ULONG ret;
402 
403     ret = InterlockedDecrement(&This->refCount);
404     if (ret == 0)
405         EnumTfContext_Destructor(This);
406     return ret;
407 }
408 
409 static HRESULT WINAPI EnumTfContext_Next(IEnumTfContexts *iface,
410     ULONG ulCount, ITfContext **rgContext, ULONG *pcFetched)
411 {
412     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
413     ULONG fetched = 0;
414 
415     TRACE("(%p)\n",This);
416 
417     if (rgContext == NULL) return E_POINTER;
418 
419     while (fetched < ulCount)
420     {
421         if (This->index > 1)
422             break;
423 
424         if (!This->docmgr->contextStack[This->index])
425             break;
426 
427         *rgContext = This->docmgr->contextStack[This->index];
428         ITfContext_AddRef(*rgContext);
429 
430         ++This->index;
431         ++fetched;
432         ++rgContext;
433     }
434 
435     if (pcFetched) *pcFetched = fetched;
436     return fetched == ulCount ? S_OK : S_FALSE;
437 }
438 
439 static HRESULT WINAPI EnumTfContext_Skip( IEnumTfContexts* iface, ULONG celt)
440 {
441     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
442     TRACE("(%p)\n",This);
443     This->index += celt;
444     return S_OK;
445 }
446 
447 static HRESULT WINAPI EnumTfContext_Reset( IEnumTfContexts* iface)
448 {
449     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
450     TRACE("(%p)\n",This);
451     This->index = 0;
452     return S_OK;
453 }
454 
455 static HRESULT WINAPI EnumTfContext_Clone( IEnumTfContexts *iface,
456     IEnumTfContexts **ppenum)
457 {
458     EnumTfContext *This = impl_from_IEnumTfContexts(iface);
459     HRESULT res;
460 
461     TRACE("(%p)\n",This);
462 
463     if (ppenum == NULL) return E_POINTER;
464 
465     res = EnumTfContext_Constructor(This->docmgr, ppenum);
466     if (SUCCEEDED(res))
467     {
468         EnumTfContext *new_This = impl_from_IEnumTfContexts(*ppenum);
469         new_This->index = This->index;
470     }
471     return res;
472 }
473 
474 static const IEnumTfContextsVtbl IEnumTfContexts_Vtbl ={
475     EnumTfContext_QueryInterface,
476     EnumTfContext_AddRef,
477     EnumTfContext_Release,
478 
479     EnumTfContext_Clone,
480     EnumTfContext_Next,
481     EnumTfContext_Reset,
482     EnumTfContext_Skip
483 };
484 
485 static HRESULT EnumTfContext_Constructor(DocumentMgr *mgr, IEnumTfContexts **ppOut)
486 {
487     EnumTfContext *This;
488 
489     This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(EnumTfContext));
490     if (This == NULL)
491         return E_OUTOFMEMORY;
492 
493     This->IEnumTfContexts_iface.lpVtbl = &IEnumTfContexts_Vtbl;
494     This->refCount = 1;
495     This->docmgr = mgr;
496 
497     *ppOut = &This->IEnumTfContexts_iface;
498     TRACE("returning %p\n", *ppOut);
499     return S_OK;
500 }
501