push e9c4c6cdd0babd7b2cb4288f191bb331b756eaf2
[wine/hacks.git] / dlls / msctf / documentmgr.c
blob7fd354aa1b8c1f37475ba0b637c7a77fc1e6304b
1 /*
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
21 #include "config.h"
23 #include <stdarg.h>
25 #define COBJMACROS
27 #include "wine/debug.h"
28 #include "windef.h"
29 #include "winbase.h"
30 #include "winreg.h"
31 #include "winuser.h"
32 #include "shlwapi.h"
33 #include "winerror.h"
34 #include "objbase.h"
36 #include "wine/unicode.h"
38 #include "msctf.h"
39 #include "msctf_internal.h"
41 WINE_DEFAULT_DEBUG_CHANNEL(msctf);
43 typedef struct tagDocumentMgr {
44 const ITfDocumentMgrVtbl *DocumentMgrVtbl;
45 const ITfSourceVtbl *SourceVtbl;
46 LONG refCount;
48 ITfContext* contextStack[2]; /* limit of 2 contexts */
49 ITfThreadMgrEventSink* ThreadMgrSink;
50 } DocumentMgr;
52 static inline DocumentMgr *impl_from_ITfSourceVtbl(ITfSource *iface)
54 return (DocumentMgr *)((char *)iface - FIELD_OFFSET(DocumentMgr,SourceVtbl));
57 static void DocumentMgr_Destructor(DocumentMgr *This)
59 TRACE("destroying %p\n", This);
60 if (This->contextStack[0])
61 ITfContext_Release(This->contextStack[0]);
62 if (This->contextStack[1])
63 ITfContext_Release(This->contextStack[1]);
64 HeapFree(GetProcessHeap(),0,This);
67 static HRESULT WINAPI DocumentMgr_QueryInterface(ITfDocumentMgr *iface, REFIID iid, LPVOID *ppvOut)
69 DocumentMgr *This = (DocumentMgr *)iface;
70 *ppvOut = NULL;
72 if (IsEqualIID(iid, &IID_IUnknown) || IsEqualIID(iid, &IID_ITfDocumentMgr))
74 *ppvOut = This;
76 else if (IsEqualIID(iid, &IID_ITfSource))
78 *ppvOut = &This->SourceVtbl;
81 if (*ppvOut)
83 IUnknown_AddRef(iface);
84 return S_OK;
87 WARN("unsupported interface: %s\n", debugstr_guid(iid));
88 return E_NOINTERFACE;
91 static ULONG WINAPI DocumentMgr_AddRef(ITfDocumentMgr *iface)
93 DocumentMgr *This = (DocumentMgr *)iface;
94 return InterlockedIncrement(&This->refCount);
97 static ULONG WINAPI DocumentMgr_Release(ITfDocumentMgr *iface)
99 DocumentMgr *This = (DocumentMgr *)iface;
100 ULONG ret;
102 ret = InterlockedDecrement(&This->refCount);
103 if (ret == 0)
104 DocumentMgr_Destructor(This);
105 return ret;
108 /*****************************************************
109 * ITfDocumentMgr functions
110 *****************************************************/
111 static HRESULT WINAPI DocumentMgr_CreateContext(ITfDocumentMgr *iface,
112 TfClientId tidOwner,
113 DWORD dwFlags, IUnknown *punk, ITfContext **ppic,
114 TfEditCookie *pecTextStore)
116 DocumentMgr *This = (DocumentMgr *)iface;
117 TRACE("(%p) 0x%x 0x%x %p %p %p\n",This,tidOwner,dwFlags,punk,ppic,pecTextStore);
118 return Context_Constructor(tidOwner, punk, ppic, pecTextStore);
121 static HRESULT WINAPI DocumentMgr_Push(ITfDocumentMgr *iface, ITfContext *pic)
123 DocumentMgr *This = (DocumentMgr *)iface;
124 ITfContext *check;
126 TRACE("(%p) %p\n",This,pic);
128 if (This->contextStack[1]) /* FUll */
129 return TF_E_STACKFULL;
131 if (!pic || FAILED(IUnknown_QueryInterface(pic,&IID_ITfContext,(LPVOID*) &check)))
132 return E_INVALIDARG;
134 if (This->contextStack[0] == NULL)
135 ITfThreadMgrEventSink_OnInitDocumentMgr(This->ThreadMgrSink,iface);
137 This->contextStack[1] = This->contextStack[0];
138 This->contextStack[0] = check;
140 ITfThreadMgrEventSink_OnPushContext(This->ThreadMgrSink,check);
141 Context_Initialize(check);
143 return S_OK;
146 static HRESULT WINAPI DocumentMgr_Pop(ITfDocumentMgr *iface, DWORD dwFlags)
148 DocumentMgr *This = (DocumentMgr *)iface;
149 TRACE("(%p) 0x%x\n",This,dwFlags);
151 if (dwFlags == TF_POPF_ALL)
153 if (This->contextStack[0])
155 ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[0]);
156 ITfContext_Release(This->contextStack[0]);
157 Context_Uninitialize(This->contextStack[0]);
159 if (This->contextStack[1])
161 ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[1]);
162 ITfContext_Release(This->contextStack[1]);
163 Context_Uninitialize(This->contextStack[1]);
165 This->contextStack[0] = This->contextStack[1] = NULL;
166 ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface);
167 return S_OK;
170 if (dwFlags)
171 return E_INVALIDARG;
173 if (This->contextStack[1] == NULL) /* Cannot pop last context */
174 return E_FAIL;
176 ITfThreadMgrEventSink_OnPopContext(This->ThreadMgrSink,This->contextStack[0]);
177 ITfContext_Release(This->contextStack[0]);
178 Context_Uninitialize(This->contextStack[0]);
179 This->contextStack[0] = This->contextStack[1];
180 This->contextStack[1] = NULL;
182 if (This->contextStack[0] == NULL)
183 ITfThreadMgrEventSink_OnUninitDocumentMgr(This->ThreadMgrSink, iface);
185 return S_OK;
188 static HRESULT WINAPI DocumentMgr_GetTop(ITfDocumentMgr *iface, ITfContext **ppic)
190 DocumentMgr *This = (DocumentMgr *)iface;
191 TRACE("(%p)\n",This);
192 if (!ppic)
193 return E_INVALIDARG;
195 if (This->contextStack[0])
196 ITfContext_AddRef(This->contextStack[0]);
198 *ppic = This->contextStack[0];
200 return S_OK;
203 static HRESULT WINAPI DocumentMgr_GetBase(ITfDocumentMgr *iface, ITfContext **ppic)
205 DocumentMgr *This = (DocumentMgr *)iface;
206 ITfContext *tgt;
208 TRACE("(%p)\n",This);
209 if (!ppic)
210 return E_INVALIDARG;
212 if (This->contextStack[1])
213 tgt = This->contextStack[1];
214 else
215 tgt = This->contextStack[0];
217 if (tgt)
218 ITfContext_AddRef(tgt);
220 *ppic = tgt;
222 return S_OK;
225 static HRESULT WINAPI DocumentMgr_EnumContexts(ITfDocumentMgr *iface, IEnumTfContexts **ppEnum)
227 DocumentMgr *This = (DocumentMgr *)iface;
228 FIXME("STUB:(%p)\n",This);
229 return E_NOTIMPL;
232 static const ITfDocumentMgrVtbl DocumentMgr_DocumentMgrVtbl =
234 DocumentMgr_QueryInterface,
235 DocumentMgr_AddRef,
236 DocumentMgr_Release,
238 DocumentMgr_CreateContext,
239 DocumentMgr_Push,
240 DocumentMgr_Pop,
241 DocumentMgr_GetTop,
242 DocumentMgr_GetBase,
243 DocumentMgr_EnumContexts
247 static HRESULT WINAPI Source_QueryInterface(ITfSource *iface, REFIID iid, LPVOID *ppvOut)
249 DocumentMgr *This = impl_from_ITfSourceVtbl(iface);
250 return DocumentMgr_QueryInterface((ITfDocumentMgr*)This, iid, *ppvOut);
253 static ULONG WINAPI Source_AddRef(ITfSource *iface)
255 DocumentMgr *This = impl_from_ITfSourceVtbl(iface);
256 return DocumentMgr_AddRef((ITfDocumentMgr*)This);
259 static ULONG WINAPI Source_Release(ITfSource *iface)
261 DocumentMgr *This = impl_from_ITfSourceVtbl(iface);
262 return DocumentMgr_Release((ITfDocumentMgr*)This);
265 /*****************************************************
266 * ITfSource functions
267 *****************************************************/
268 static WINAPI HRESULT DocumentMgrSource_AdviseSink(ITfSource *iface,
269 REFIID riid, IUnknown *punk, DWORD *pdwCookie)
271 DocumentMgr *This = impl_from_ITfSourceVtbl(iface);
272 FIXME("STUB:(%p)\n",This);
273 return E_NOTIMPL;
276 static WINAPI HRESULT DocumentMgrSource_UnadviseSink(ITfSource *iface, DWORD pdwCookie)
278 DocumentMgr *This = impl_from_ITfSourceVtbl(iface);
279 FIXME("STUB:(%p)\n",This);
280 return E_NOTIMPL;
283 static const ITfSourceVtbl DocumentMgr_SourceVtbl =
285 Source_QueryInterface,
286 Source_AddRef,
287 Source_Release,
289 DocumentMgrSource_AdviseSink,
290 DocumentMgrSource_UnadviseSink,
293 HRESULT DocumentMgr_Constructor(ITfThreadMgrEventSink *ThreadMgrSink, ITfDocumentMgr **ppOut)
295 DocumentMgr *This;
297 This = HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,sizeof(DocumentMgr));
298 if (This == NULL)
299 return E_OUTOFMEMORY;
301 This->DocumentMgrVtbl= &DocumentMgr_DocumentMgrVtbl;
302 This->SourceVtbl = &DocumentMgr_SourceVtbl;
303 This->refCount = 1;
304 This->ThreadMgrSink = ThreadMgrSink;
306 TRACE("returning %p\n", This);
307 *ppOut = (ITfDocumentMgr*)This;
308 return S_OK;