mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
efa86fff9f
Supporting TIPs... JIRA issue: CORE-19360 - Implement CicBridge::Notify method. - Implement CicBridge:: SetCompositionString method. - Add some CicInputContext methods (stub).
520 lines
10 KiB
C++
520 lines
10 KiB
C++
/*
|
|
* PROJECT: ReactOS msctfime.ime
|
|
* LICENSE: LGPL-2.1-or-later (https://spdx.org/licenses/LGPL-2.1-or-later)
|
|
* PURPOSE: Input Context of msctfime.ime
|
|
* COPYRIGHT: Copyright 2024 Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
|
|
*/
|
|
|
|
#include "msctfime.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msctfime);
|
|
|
|
/***********************************************************************
|
|
* CInputContextOwner
|
|
*/
|
|
|
|
/// @unimplemented
|
|
CInputContextOwner::CInputContextOwner(FN_IC_OWNER_CALLBACK fnCallback, LPVOID pCallbackPV)
|
|
{
|
|
m_dwCookie = -1;
|
|
m_fnCallback = fnCallback;
|
|
m_cRefs = 1;
|
|
m_pCallbackPV = pCallbackPV;
|
|
}
|
|
|
|
/// @implemented
|
|
CInputContextOwner::~CInputContextOwner()
|
|
{
|
|
}
|
|
|
|
/// @implemented
|
|
HRESULT CInputContextOwner::_Advise(IUnknown *pContext)
|
|
{
|
|
ITfSource *pSource = NULL;
|
|
|
|
m_pContext = NULL;
|
|
|
|
HRESULT hr = E_FAIL;
|
|
if (SUCCEEDED(m_pContext->QueryInterface(IID_ITfSource, (LPVOID*)&pSource)) &&
|
|
SUCCEEDED(pSource->AdviseSink(IID_ITfContextOwner,
|
|
static_cast<ITfContextOwner*>(this), &m_dwCookie)))
|
|
{
|
|
m_pContext = pContext;
|
|
m_pContext->AddRef();
|
|
hr = S_OK;
|
|
}
|
|
|
|
if (pSource)
|
|
pSource->Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
/// @implemented
|
|
HRESULT CInputContextOwner::_Unadvise()
|
|
{
|
|
ITfSource *pSource = NULL;
|
|
|
|
HRESULT hr = E_FAIL;
|
|
if (m_pContext)
|
|
{
|
|
if (SUCCEEDED(m_pContext->QueryInterface(IID_ITfSource, (LPVOID*)&pSource)) &&
|
|
SUCCEEDED(pSource->UnadviseSink(m_dwCookie)))
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
|
|
if (m_pContext)
|
|
{
|
|
m_pContext->Release();
|
|
m_pContext = NULL;
|
|
}
|
|
|
|
if (pSource)
|
|
pSource->Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CInputContextOwner::QueryInterface(REFIID riid, LPVOID* ppvObj)
|
|
{
|
|
*ppvObj = NULL;
|
|
|
|
if (IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_ITfContextOwner))
|
|
{
|
|
*ppvObj = this;
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
if (IsEqualIID(riid, IID_ITfMouseTrackerACP))
|
|
{
|
|
*ppvObj = static_cast<ITfMouseTrackerACP*>(this);
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP_(ULONG) CInputContextOwner::AddRef()
|
|
{
|
|
return ++m_cRefs;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP_(ULONG) CInputContextOwner::Release()
|
|
{
|
|
if (--m_cRefs == 0)
|
|
{
|
|
delete this;
|
|
return 0;
|
|
}
|
|
return m_cRefs;
|
|
}
|
|
|
|
/// @unimplemented
|
|
STDMETHODIMP
|
|
CInputContextOwner::GetACPFromPoint(
|
|
const POINT *ptScreen,
|
|
DWORD dwFlags,
|
|
LONG *pacp)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
STDMETHODIMP
|
|
CInputContextOwner::GetTextExt(
|
|
LONG acpStart,
|
|
LONG acpEnd,
|
|
RECT *prc,
|
|
BOOL *pfClipped)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CInputContextOwner::GetScreenExt(RECT *prc)
|
|
{
|
|
return m_fnCallback(2, &prc, m_pCallbackPV);
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CInputContextOwner::GetStatus(TF_STATUS *pdcs)
|
|
{
|
|
return m_fnCallback(6, &pdcs, m_pCallbackPV);
|
|
}
|
|
|
|
/// @unimplemented
|
|
STDMETHODIMP CInputContextOwner::GetWnd(HWND *phwnd)
|
|
{
|
|
return m_fnCallback(7, &phwnd, m_pCallbackPV);
|
|
}
|
|
|
|
/// @unimplemented
|
|
STDMETHODIMP CInputContextOwner::GetAttribute(REFGUID rguidAttribute, VARIANT *pvarValue)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
struct MOUSE_SINK_ARGS
|
|
{
|
|
ITfRangeACP *range;
|
|
ITfMouseSink *pSink;
|
|
DWORD *pdwCookie;
|
|
};
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CInputContextOwner::AdviseMouseSink(
|
|
ITfRangeACP *range,
|
|
ITfMouseSink *pSink,
|
|
DWORD *pdwCookie)
|
|
{
|
|
MOUSE_SINK_ARGS args = { range, pSink, pdwCookie };
|
|
return m_fnCallback(9, &args, m_pCallbackPV);
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CInputContextOwner::UnadviseMouseSink(DWORD dwCookie)
|
|
{
|
|
return m_fnCallback(10, &dwCookie, m_pCallbackPV);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CicInputContext
|
|
*/
|
|
|
|
/// @unimplemented
|
|
CicInputContext::CicInputContext(
|
|
_In_ TfClientId cliendId,
|
|
_Inout_ PCIC_LIBTHREAD pLibThread,
|
|
_In_ HIMC hIMC)
|
|
{
|
|
m_hIMC = hIMC;
|
|
m_dwQueryPos = 0;
|
|
m_cRefs = 1;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP CicInputContext::QueryInterface(REFIID riid, LPVOID* ppvObj)
|
|
{
|
|
*ppvObj = NULL;
|
|
|
|
if (IsEqualIID(riid, IID_ITfContextOwnerCompositionSink))
|
|
{
|
|
*ppvObj = static_cast<ITfContextOwnerCompositionSink*>(this);
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
if (IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_ITfCleanupContextSink))
|
|
{
|
|
*ppvObj = this;
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP_(ULONG) CicInputContext::AddRef()
|
|
{
|
|
return ::InterlockedIncrement(&m_cRefs);
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP_(ULONG) CicInputContext::Release()
|
|
{
|
|
if (::InterlockedDecrement(&m_cRefs) == 0)
|
|
{
|
|
delete this;
|
|
return 0;
|
|
}
|
|
return m_cRefs;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP
|
|
CicInputContext::OnStartComposition(
|
|
ITfCompositionView *pComposition,
|
|
BOOL *pfOk)
|
|
{
|
|
if ((m_cCompLocks <= 0) || m_bReconverting)
|
|
{
|
|
*pfOk = TRUE;
|
|
++m_cCompLocks;
|
|
}
|
|
else
|
|
{
|
|
*pfOk = FALSE;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP
|
|
CicInputContext::OnUpdateComposition(
|
|
ITfCompositionView *pComposition,
|
|
ITfRange *pRangeNew)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP
|
|
CicInputContext::OnEndComposition(
|
|
ITfCompositionView *pComposition)
|
|
{
|
|
--m_cCompLocks;
|
|
return S_OK;
|
|
}
|
|
|
|
/// @implemented
|
|
HRESULT
|
|
CicInputContext::GetGuidAtom(
|
|
_Inout_ CicIMCLock& imcLock,
|
|
_In_ BYTE iAtom,
|
|
_Out_opt_ LPDWORD pdwGuidAtom)
|
|
{
|
|
CicIMCCLock<CTFIMECONTEXT> imeContext(imcLock.get().hCompStr);
|
|
if (FAILED(imeContext.m_hr))
|
|
return imeContext.m_hr;
|
|
|
|
HRESULT hr = E_FAIL;
|
|
if (iAtom < m_cGuidAtoms)
|
|
{
|
|
*pdwGuidAtom = m_adwGuidAtoms[iAtom];
|
|
hr = S_OK;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT
|
|
CicInputContext::CreateInputContext(
|
|
_Inout_ ITfThreadMgr *pThreadMgr,
|
|
_Inout_ CicIMCLock& imcLock)
|
|
{
|
|
//FIXME
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT
|
|
CicInputContext::DestroyInputContext()
|
|
{
|
|
ITfSourceSingle *pSource = NULL;
|
|
|
|
if (m_pContext && m_pContext->QueryInterface(IID_ITfSourceSingle, (void **)&pSource) == S_OK)
|
|
pSource->UnadviseSingleSink(m_clientId, IID_ITfCleanupContextSink);
|
|
|
|
//FIXME: m_dwUnknown5
|
|
|
|
if (m_pTextEventSink)
|
|
{
|
|
m_pTextEventSink->_Unadvise();
|
|
m_pTextEventSink->Release();
|
|
m_pTextEventSink = NULL;
|
|
}
|
|
|
|
if (m_pCompEventSink2)
|
|
{
|
|
m_pCompEventSink2->_Unadvise();
|
|
m_pCompEventSink2->Release();
|
|
m_pCompEventSink2 = NULL;
|
|
}
|
|
|
|
if (m_pCompEventSink1)
|
|
{
|
|
m_pCompEventSink1->_Unadvise();
|
|
m_pCompEventSink1->Release();
|
|
m_pCompEventSink1 = NULL;
|
|
}
|
|
|
|
if (m_pInputContextOwner)
|
|
{
|
|
m_pInputContextOwner->_Unadvise();
|
|
m_pInputContextOwner->Release();
|
|
m_pInputContextOwner = NULL;
|
|
}
|
|
|
|
if (m_pDocumentMgr)
|
|
m_pDocumentMgr->Pop(1);
|
|
|
|
if (m_pContext)
|
|
{
|
|
ClearCompartment(m_clientId, m_pContext, GUID_COMPARTMENT_CTFIME_CICINPUTCONTEXT, 0);
|
|
m_pContext->Release();
|
|
m_pContext = NULL;
|
|
}
|
|
|
|
if (m_pContextOwnerServices)
|
|
{
|
|
m_pContextOwnerServices->Release();
|
|
m_pContextOwnerServices = NULL;
|
|
}
|
|
|
|
// FIXME: m_pICOwnerCallback
|
|
|
|
if (m_pDocumentMgr)
|
|
{
|
|
m_pDocumentMgr->Release();
|
|
m_pDocumentMgr = NULL;
|
|
}
|
|
|
|
if (pSource)
|
|
pSource->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP
|
|
CicInputContext::OnCompositionTerminated(TfEditCookie ecWrite, ITfComposition *pComposition)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
/// @implemented
|
|
STDMETHODIMP
|
|
CicInputContext::OnCleanupContext(
|
|
_In_ TfEditCookie ecWrite,
|
|
_Inout_ ITfContext *pic)
|
|
{
|
|
TLS *pTLS = TLS::PeekTLS();
|
|
if (!pTLS || !pTLS->m_pProfile)
|
|
return E_OUTOFMEMORY;
|
|
|
|
LANGID LangID;
|
|
pTLS->m_pProfile->GetLangId(&LangID);
|
|
|
|
IMEINFO IMEInfo;
|
|
WCHAR szPath[MAX_PATH];
|
|
if (Inquire(&IMEInfo, szPath, 0, (HKL)UlongToHandle(LangID)) != S_OK)
|
|
return E_FAIL;
|
|
|
|
ITfProperty *pProp = NULL;
|
|
if (!(IMEInfo.fdwProperty & IME_PROP_COMPLETE_ON_UNSELECT))
|
|
return S_OK;
|
|
|
|
HRESULT hr = pic->GetProperty(GUID_PROP_COMPOSING, &pProp);
|
|
if (FAILED(hr))
|
|
return S_OK;
|
|
|
|
IEnumTfRanges *pRanges = NULL;
|
|
hr = pProp->EnumRanges(ecWrite, &pRanges, NULL);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ITfRange *pRange = NULL;
|
|
while (pRanges->Next(1, &pRange, 0) == S_OK)
|
|
{
|
|
VARIANT vari;
|
|
V_VT(&vari) = VT_EMPTY;
|
|
pProp->GetValue(ecWrite, pRange, &vari);
|
|
if (V_VT(&vari) == VT_I4)
|
|
{
|
|
if (V_I4(&vari))
|
|
pProp->Clear(ecWrite, pRange);
|
|
}
|
|
pRange->Release();
|
|
pRange = NULL;
|
|
}
|
|
pRanges->Release();
|
|
}
|
|
pProp->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::SetupDocFeedString(CicIMCLock& imcLock, UINT uCodePage)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::EscbClearDocFeedBuffer(CicIMCLock& imcLock, BOOL bFlag)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::EscbCompComplete(CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::EscbCompCancel(CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::OnSetCandidatePos(TLS *pTLS, CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::DelayedReconvertFuncCall(CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT
|
|
CicInputContext::MsImeMouseHandler(
|
|
DWORD dwUnknown58,
|
|
DWORD dwUnknown59,
|
|
UINT keys,
|
|
CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT
|
|
CicInputContext::SetupReconvertString(
|
|
CicIMCLock& imcLock,
|
|
ITfThreadMgr_P *pThreadMgr,
|
|
UINT uCodePage,
|
|
UINT uMsg,
|
|
BOOL bUndo)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
void CicInputContext::ClearPrevCandidatePos()
|
|
{
|
|
m_dwUnknown8 = 0;
|
|
ZeroMemory(&m_rcCandidate1, sizeof(m_rcCandidate1));
|
|
ZeroMemory(&m_CandForm, sizeof(m_CandForm));
|
|
ZeroMemory(&m_rcCandidate2, sizeof(m_rcCandidate2));
|
|
m_dwQueryPos = 0;
|
|
}
|
|
|
|
/// @unimplemented
|
|
HRESULT CicInputContext::EndReconvertString(CicIMCLock& imcLock)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/// @unimplemented
|
|
BOOL CicInputContext::SetCompositionString(
|
|
CicIMCLock& imcLock,
|
|
ITfThreadMgr_P *pThreadMgr,
|
|
DWORD dwIndex,
|
|
LPCVOID lpComp,
|
|
DWORD dwCompLen,
|
|
LPCVOID lpRead,
|
|
DWORD dwReadLen,
|
|
UINT uCodePage)
|
|
{
|
|
return FALSE;
|
|
}
|