2 * ITfDocumentMgr implementation
4 * Copyright 2009 Aric Stewart, CodeWeavers
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
25 #include "wine/debug.h"
35 #include "msctf_internal.h"
37 WINE_DEFAULT_DEBUG_CHANNEL(msctf
);
39 typedef struct tagDocumentMgr
{
40 ITfDocumentMgr ITfDocumentMgr_iface
;
41 ITfSource ITfSource_iface
;
45 ITfCompartmentMgr
*CompartmentMgr
;
47 ITfContext
* initialContext
;
48 ITfContext
* contextStack
[2]; /* limit of 2 contexts */
49 ITfThreadMgrEventSink
* ThreadMgrSink
;
51 struct list TransitoryExtensionSink
;
54 typedef struct tagEnumTfContext
{
55 IEnumTfContexts IEnumTfContexts_iface
;
62 static HRESULT
EnumTfContext_Constructor(DocumentMgr
* mgr
, IEnumTfContexts
**ppOut
);
64 static inline DocumentMgr
*impl_from_ITfDocumentMgr(ITfDocumentMgr
*iface
)
66 return CONTAINING_RECORD(iface
, DocumentMgr
, ITfDocumentMgr_iface
);
69 static inline DocumentMgr
*impl_from_ITfSource(ITfSource
*iface
)
71 return CONTAINING_RECORD(iface
, DocumentMgr
, ITfSource_iface
);
74 static inline EnumTfContext
*impl_from_IEnumTfContexts(IEnumTfContexts
*iface
)
76 return CONTAINING_RECORD(iface
, EnumTfContext
, IEnumTfContexts_iface
);
79 static void DocumentMgr_Destructor(DocumentMgr
*This
)
81 ITfThreadMgr
*tm
= NULL
;
82 TRACE("destroying %p\n", This
);
87 ThreadMgr_OnDocumentMgrDestruction(tm
, &This
->ITfDocumentMgr_iface
);
88 ITfThreadMgr_Release(tm
);
91 if (This
->initialContext
)
92 ITfContext_Release(This
->initialContext
);
93 if (This
->contextStack
[0])
94 ITfContext_Release(This
->contextStack
[0]);
95 if (This
->contextStack
[1])
96 ITfContext_Release(This
->contextStack
[1]);
97 free_sinks(&This
->TransitoryExtensionSink
);
98 CompartmentMgr_Destructor(This
->CompartmentMgr
);
102 static HRESULT WINAPI
DocumentMgr_QueryInterface(ITfDocumentMgr
*iface
, REFIID iid
, LPVOID
*ppvOut
)
104 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
107 if (IsEqualIID(iid
, &IID_IUnknown
) || IsEqualIID(iid
, &IID_ITfDocumentMgr
))
109 *ppvOut
= &This
->ITfDocumentMgr_iface
;
111 else if (IsEqualIID(iid
, &IID_ITfSource
))
113 *ppvOut
= &This
->ITfSource_iface
;
115 else if (IsEqualIID(iid
, &IID_ITfCompartmentMgr
))
117 *ppvOut
= This
->CompartmentMgr
;
122 ITfDocumentMgr_AddRef(iface
);
126 WARN("unsupported interface: %s\n", debugstr_guid(iid
));
127 return E_NOINTERFACE
;
130 static ULONG WINAPI
DocumentMgr_AddRef(ITfDocumentMgr
*iface
)
132 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
133 return InterlockedIncrement(&This
->refCount
);
136 static ULONG WINAPI
DocumentMgr_Release(ITfDocumentMgr
*iface
)
138 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
141 ret
= InterlockedDecrement(&This
->refCount
);
143 DocumentMgr_Destructor(This
);
147 /*****************************************************
148 * ITfDocumentMgr functions
149 *****************************************************/
150 static HRESULT WINAPI
DocumentMgr_CreateContext(ITfDocumentMgr
*iface
,
152 DWORD dwFlags
, IUnknown
*punk
, ITfContext
**ppic
,
153 TfEditCookie
*pecTextStore
)
155 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
156 TRACE("(%p) 0x%lx 0x%lx %p %p %p\n",This
,tidOwner
,dwFlags
,punk
,ppic
,pecTextStore
);
157 return Context_Constructor(tidOwner
, punk
, iface
, ppic
, pecTextStore
);
160 static HRESULT WINAPI
DocumentMgr_Push(ITfDocumentMgr
*iface
, ITfContext
*pic
)
162 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
165 TRACE("(%p) %p\n",This
,pic
);
167 if (This
->contextStack
[1]) /* FUll */
168 return TF_E_STACKFULL
;
170 if (!pic
|| FAILED(ITfContext_QueryInterface(pic
,&IID_ITfContext
,(LPVOID
*) &check
)))
173 if (This
->contextStack
[0] == NULL
)
174 ITfThreadMgrEventSink_OnInitDocumentMgr(This
->ThreadMgrSink
,iface
);
176 This
->contextStack
[1] = This
->contextStack
[0];
177 This
->contextStack
[0] = check
;
179 Context_Initialize(check
, iface
);
180 ITfThreadMgrEventSink_OnPushContext(This
->ThreadMgrSink
,check
);
185 static HRESULT WINAPI
DocumentMgr_Pop(ITfDocumentMgr
*iface
, DWORD dwFlags
)
187 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
188 TRACE("(%p) 0x%lx\n",This
,dwFlags
);
190 if (dwFlags
== TF_POPF_ALL
)
194 for (i
= 0; i
< ARRAY_SIZE(This
->contextStack
); i
++)
195 if (This
->contextStack
[i
])
197 ITfThreadMgrEventSink_OnPopContext(This
->ThreadMgrSink
, This
->contextStack
[i
]);
198 Context_Uninitialize(This
->contextStack
[i
]);
199 ITfContext_Release(This
->contextStack
[i
]);
200 This
->contextStack
[i
] = NULL
;
203 ITfThreadMgrEventSink_OnUninitDocumentMgr(This
->ThreadMgrSink
, iface
);
210 if (This
->contextStack
[1] == NULL
) /* Cannot pop last context */
213 ITfThreadMgrEventSink_OnPopContext(This
->ThreadMgrSink
,This
->contextStack
[0]);
214 Context_Uninitialize(This
->contextStack
[0]);
215 ITfContext_Release(This
->contextStack
[0]);
216 This
->contextStack
[0] = This
->contextStack
[1];
217 This
->contextStack
[1] = NULL
;
219 if (This
->contextStack
[0] == NULL
)
220 ITfThreadMgrEventSink_OnUninitDocumentMgr(This
->ThreadMgrSink
, iface
);
225 static HRESULT WINAPI
DocumentMgr_GetTop(ITfDocumentMgr
*iface
, ITfContext
**ppic
)
227 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
230 TRACE("(%p)\n",This
);
234 if (This
->contextStack
[0])
235 tgt
= This
->contextStack
[0];
237 tgt
= This
->initialContext
;
240 ITfContext_AddRef(tgt
);
247 static HRESULT WINAPI
DocumentMgr_GetBase(ITfDocumentMgr
*iface
, ITfContext
**ppic
)
249 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
252 TRACE("(%p)\n",This
);
256 if (This
->contextStack
[1])
257 tgt
= This
->contextStack
[1];
258 else if (This
->contextStack
[0])
259 tgt
= This
->contextStack
[0];
261 tgt
= This
->initialContext
;
264 ITfContext_AddRef(tgt
);
271 static HRESULT WINAPI
DocumentMgr_EnumContexts(ITfDocumentMgr
*iface
, IEnumTfContexts
**ppEnum
)
273 DocumentMgr
*This
= impl_from_ITfDocumentMgr(iface
);
274 TRACE("(%p) %p\n",This
,ppEnum
);
275 return EnumTfContext_Constructor(This
, ppEnum
);
278 static const ITfDocumentMgrVtbl DocumentMgrVtbl
=
280 DocumentMgr_QueryInterface
,
283 DocumentMgr_CreateContext
,
288 DocumentMgr_EnumContexts
291 static HRESULT WINAPI
DocumentMgrSource_QueryInterface(ITfSource
*iface
, REFIID iid
, LPVOID
*ppvOut
)
293 DocumentMgr
*This
= impl_from_ITfSource(iface
);
294 return ITfDocumentMgr_QueryInterface(&This
->ITfDocumentMgr_iface
, iid
, ppvOut
);
297 static ULONG WINAPI
DocumentMgrSource_AddRef(ITfSource
*iface
)
299 DocumentMgr
*This
= impl_from_ITfSource(iface
);
300 return ITfDocumentMgr_AddRef(&This
->ITfDocumentMgr_iface
);
303 static ULONG WINAPI
DocumentMgrSource_Release(ITfSource
*iface
)
305 DocumentMgr
*This
= impl_from_ITfSource(iface
);
306 return ITfDocumentMgr_Release(&This
->ITfDocumentMgr_iface
);
309 /*****************************************************
310 * ITfSource functions
311 *****************************************************/
312 static HRESULT WINAPI
DocumentMgrSource_AdviseSink(ITfSource
*iface
,
313 REFIID riid
, IUnknown
*punk
, DWORD
*pdwCookie
)
315 DocumentMgr
*This
= impl_from_ITfSource(iface
);
317 TRACE("(%p) %s %p %p\n", This
, debugstr_guid(riid
), punk
, pdwCookie
);
319 if (!riid
|| !punk
|| !pdwCookie
)
322 if (IsEqualIID(riid
, &IID_ITfTransitoryExtensionSink
))
324 WARN("semi-stub for ITfTransitoryExtensionSink: callback won't be used.\n");
325 return advise_sink(&This
->TransitoryExtensionSink
, &IID_ITfTransitoryExtensionSink
,
326 COOKIE_MAGIC_DMSINK
, punk
, pdwCookie
);
329 FIXME("(%p) Unhandled Sink: %s\n",This
,debugstr_guid(riid
));
333 static HRESULT WINAPI
DocumentMgrSource_UnadviseSink(ITfSource
*iface
, DWORD pdwCookie
)
335 DocumentMgr
*This
= impl_from_ITfSource(iface
);
337 TRACE("(%p) %lx\n",This
,pdwCookie
);
339 if (get_Cookie_magic(pdwCookie
)!=COOKIE_MAGIC_DMSINK
)
342 return unadvise_sink(pdwCookie
);
345 static const ITfSourceVtbl DocumentMgrSourceVtbl
=
347 DocumentMgrSource_QueryInterface
,
348 DocumentMgrSource_AddRef
,
349 DocumentMgrSource_Release
,
350 DocumentMgrSource_AdviseSink
,
351 DocumentMgrSource_UnadviseSink
,
354 HRESULT
DocumentMgr_Constructor(ITfThreadMgrEventSink
*ThreadMgrSink
, ITfDocumentMgr
**ppOut
)
359 This
= calloc(1, sizeof(DocumentMgr
));
361 return E_OUTOFMEMORY
;
363 This
->ITfDocumentMgr_iface
.lpVtbl
= &DocumentMgrVtbl
;
364 This
->ITfSource_iface
.lpVtbl
= &DocumentMgrSourceVtbl
;
366 This
->ThreadMgrSink
= ThreadMgrSink
;
367 list_init(&This
->TransitoryExtensionSink
);
369 CompartmentMgr_Constructor((IUnknown
*)&This
->ITfDocumentMgr_iface
, &IID_IUnknown
, (IUnknown
**)&This
->CompartmentMgr
);
370 Context_Constructor(processId
,NULL
,&This
->ITfDocumentMgr_iface
, &This
->initialContext
, &cookie
);
372 *ppOut
= &This
->ITfDocumentMgr_iface
;
373 TRACE("returning %p\n", *ppOut
);
377 /**************************************************
378 * IEnumTfContexts implementation
379 **************************************************/
380 static void EnumTfContext_Destructor(EnumTfContext
*This
)
382 TRACE("destroying %p\n", This
);
386 static HRESULT WINAPI
EnumTfContext_QueryInterface(IEnumTfContexts
*iface
, REFIID iid
, LPVOID
*ppvOut
)
388 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
391 if (IsEqualIID(iid
, &IID_IUnknown
) || IsEqualIID(iid
, &IID_IEnumTfContexts
))
393 *ppvOut
= &This
->IEnumTfContexts_iface
;
398 IEnumTfContexts_AddRef(iface
);
402 WARN("unsupported interface: %s\n", debugstr_guid(iid
));
403 return E_NOINTERFACE
;
406 static ULONG WINAPI
EnumTfContext_AddRef(IEnumTfContexts
*iface
)
408 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
409 return InterlockedIncrement(&This
->refCount
);
412 static ULONG WINAPI
EnumTfContext_Release(IEnumTfContexts
*iface
)
414 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
417 ret
= InterlockedDecrement(&This
->refCount
);
419 EnumTfContext_Destructor(This
);
423 static HRESULT WINAPI
EnumTfContext_Next(IEnumTfContexts
*iface
,
424 ULONG ulCount
, ITfContext
**rgContext
, ULONG
*pcFetched
)
426 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
429 TRACE("(%p)\n",This
);
431 if (rgContext
== NULL
) return E_POINTER
;
433 while (fetched
< ulCount
)
438 if (!This
->docmgr
->contextStack
[This
->index
])
441 *rgContext
= This
->docmgr
->contextStack
[This
->index
];
442 ITfContext_AddRef(*rgContext
);
449 if (pcFetched
) *pcFetched
= fetched
;
450 return fetched
== ulCount
? S_OK
: S_FALSE
;
453 static HRESULT WINAPI
EnumTfContext_Skip( IEnumTfContexts
* iface
, ULONG celt
)
455 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
456 TRACE("(%p)\n",This
);
461 static HRESULT WINAPI
EnumTfContext_Reset( IEnumTfContexts
* iface
)
463 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
464 TRACE("(%p)\n",This
);
469 static HRESULT WINAPI
EnumTfContext_Clone( IEnumTfContexts
*iface
,
470 IEnumTfContexts
**ppenum
)
472 EnumTfContext
*This
= impl_from_IEnumTfContexts(iface
);
475 TRACE("(%p)\n",This
);
477 if (ppenum
== NULL
) return E_POINTER
;
479 res
= EnumTfContext_Constructor(This
->docmgr
, ppenum
);
482 EnumTfContext
*new_This
= impl_from_IEnumTfContexts(*ppenum
);
483 new_This
->index
= This
->index
;
488 static const IEnumTfContextsVtbl IEnumTfContexts_Vtbl
={
489 EnumTfContext_QueryInterface
,
490 EnumTfContext_AddRef
,
491 EnumTfContext_Release
,
499 static HRESULT
EnumTfContext_Constructor(DocumentMgr
*mgr
, IEnumTfContexts
**ppOut
)
503 This
= calloc(1, sizeof(EnumTfContext
));
505 return E_OUTOFMEMORY
;
507 This
->IEnumTfContexts_iface
.lpVtbl
= &IEnumTfContexts_Vtbl
;
511 *ppOut
= &This
->IEnumTfContexts_iface
;
512 TRACE("returning %p\n", *ppOut
);