|
|
//
// range.h
//
#ifndef RANGE_H
#define RANGE_H
#include "private.h"
#include "anchor.h"
#include "sink.h"
#include "ic.h"
#define IGNORE_LAST_LOCKRELEASED 0xffffffff
class CInputContext; class CEnumOwnedRanges;
extern const IID IID_PRIV_CRANGE;
inline TfGravity DCGToIMG(TsGravity dcg) { return dcg == TS_GR_FORWARD ? TF_GRAVITY_FORWARD : TF_GRAVITY_BACKWARD; }
typedef enum { RINIT_DEF_GRAVITY, RINIT_GRAVITY, RINIT_NO_GRAVITY } RInit;
class CRange : public ITfRangeACP, public ITfRangeAnchor, public ITfSource { public: // work around for new #define in mem.h
#undef new
DECLARE_CACHED_NEW; // retore mem.h trick
#ifdef DEBUG
#define new new(TEXT(__FILE__), __LINE__)
#endif // DEBUG
CRange() { Dbg_MemSetThisNameIDCounter(TEXT("CRange"), PERF_RANGE_COUNTER); _cRef = 1; } ~CRange();
// NB: caller must be certain that paStart <= paEnd before calling _InitWithDefaultGravity!
BOOL _InitWithDefaultGravity(CInputContext *pic, AnchorOwnership ao, IAnchor *paStart, IAnchor *paEnd) { return _Init(pic, ao, paStart, paEnd, RINIT_DEF_GRAVITY); } BOOL _InitWithAnchorGravity(CInputContext *pic, AnchorOwnership ao, IAnchor *paStart, IAnchor *paEnd) { return _Init(pic, ao, paStart, paEnd, RINIT_GRAVITY); }
static void _InitClass(); static void _UninitClass();
//
// IUnknown methods
//
STDMETHODIMP QueryInterface(REFIID riid, void **ppvObj); STDMETHODIMP_(ULONG) AddRef(void); STDMETHODIMP_(ULONG) Release(void);
// ITfRange
STDMETHODIMP GetText(TfEditCookie ec, DWORD dwFlags, WCHAR *pchText, ULONG cchMax, ULONG *pcch); STDMETHODIMP SetText(TfEditCookie ec, DWORD dwFlags, const WCHAR *pchText, LONG cch); STDMETHODIMP GetFormattedText(TfEditCookie ec, IDataObject **ppDataObject); STDMETHODIMP GetEmbedded(TfEditCookie ec, REFGUID rguidService, REFIID riid, IUnknown **ppunk); STDMETHODIMP InsertEmbedded(TfEditCookie ec, DWORD dwFlags, IDataObject *pDataObject); STDMETHODIMP ShiftStart(TfEditCookie ec, LONG cchReq, LONG *pcch, const TF_HALTCOND *pHalt); STDMETHODIMP ShiftEnd(TfEditCookie ec, LONG cchReq, LONG *pcch, const TF_HALTCOND *pHalt); STDMETHODIMP ShiftStartToRange(TfEditCookie ec, ITfRange *pRange, TfAnchor aPos); STDMETHODIMP ShiftEndToRange(TfEditCookie ec, ITfRange *pRange, TfAnchor aPos); STDMETHODIMP ShiftStartRegion(TfEditCookie ec, TfShiftDir dir, BOOL *pfNoRegion); STDMETHODIMP ShiftEndRegion(TfEditCookie ec, TfShiftDir dir, BOOL *pfNoRegion); STDMETHODIMP IsEmpty(TfEditCookie ec, BOOL *pfEmpty); STDMETHODIMP Collapse(TfEditCookie ec, TfAnchor aPos); STDMETHODIMP IsEqualStart(TfEditCookie ec, ITfRange *pWith, TfAnchor aPos, BOOL *pfEqual); STDMETHODIMP IsEqualEnd(TfEditCookie ec, ITfRange *pWith, TfAnchor aPos, BOOL *pfEqual); STDMETHODIMP CompareStart(TfEditCookie ec, ITfRange *pWith, TfAnchor aPos, LONG *plResult); STDMETHODIMP CompareEnd(TfEditCookie ec, ITfRange *pWith, TfAnchor aPos, LONG *plResult); STDMETHODIMP AdjustForInsert(TfEditCookie ec, ULONG cchInsert, BOOL *pfInsertOk); STDMETHODIMP GetGravity(TfGravity *pgStart, TfGravity *pgEnd); STDMETHODIMP SetGravity(TfEditCookie ec, TfGravity gStart, TfGravity gEnd); STDMETHODIMP Clone(ITfRange **ppClone); STDMETHODIMP GetContext(ITfContext **ppContext);
// ITfRangeACP
STDMETHODIMP GetExtent(LONG *pacpAnchor, LONG *pcch); STDMETHODIMP SetExtent(LONG acpAnchor, LONG cch);
// ITfRangeAnchor
STDMETHODIMP GetExtent(IAnchor **ppaStart, IAnchor **ppaEnd); STDMETHODIMP SetExtent(IAnchor *paStart, IAnchor *paEnd);
// ITfSource
STDMETHODIMP AdviseSink(REFIID riid, IUnknown *punk, DWORD *pdwCookie); STDMETHODIMP UnadviseSink(DWORD dwCookie);
CRange *_Clone() { CRange *rangeClone;
if ((rangeClone = new CRange) == NULL) return NULL;
if (!rangeClone->_Init(_pic, COPY_ANCHORS, _paStart, _paEnd, RINIT_NO_GRAVITY)) { rangeClone->Release(); return NULL; }
rangeClone->_dwLastLockReleaseID = _dwLastLockReleaseID;
return rangeClone; }
CRange *_GetNextOnChangeRangeInIcsub() { return _nextOnChangeRangeInIcsub; }
IAnchor *_GetStart() { return _paStart; } IAnchor *_GetEnd() { return _paEnd; }
CInputContext *_GetContext() { return _pic; }
CStructArray<GENERICSINK> *_GetChangeSinks() { return _prgChangeSinks; }
void _QuickCheckCrossedAnchors() { if (_dwLastLockReleaseID != IGNORE_LAST_LOCKRELEASED) { _CheckCrossedAnchors(); } }
#if 0
HRESULT _SnapToRegion(DWORD dwFlags); #endif
BOOL _IsDirty() { return _fDirty; } void _SetDirty() { _fDirty = TRUE; } void _ClearDirty() { _fDirty = FALSE; }
private: BOOL _Init(CInputContext *pic, AnchorOwnership ao, IAnchor *paStart, IAnchor *paEnd, RInit rinit);
HRESULT _SetGravity(TfGravity gStart, TfGravity gEnd, BOOL fCheckCrossedAnchors);
void _CheckCrossedAnchors();
HRESULT _PreEditCompositionCheck(TfEditCookie ec, CComposition **ppComposition, BOOL *pfNewComposition);
HRESULT _ShiftConditional(IAnchor *paStart, IAnchor *paLimit, LONG cchReq, LONG *pcch, const TF_HALTCOND *pHalt);
HRESULT _IsEqualX(TfEditCookie ec, TfAnchor aPosThisRange, ITfRange *pWith, TfAnchor aPos, BOOL *pfEqual); HRESULT _CompareX(TfEditCookie ec, TfAnchor aPosThisRange, ITfRange *pWith, TfAnchor aPos, LONG *plResult);
BOOL _IsValidEditCookie(TfEditCookie ec, DWORD dwFlags);
void _InitLastLockReleaseId(TsGravity gStart, TsGravity gEnd) { if (gStart == TF_GRAVITY_FORWARD && gEnd == TF_GRAVITY_BACKWARD) { // this range has the potential for crossed anchors, need to monitor
// since the range may have just been cloned from a crossed range, need to
// init _dwLastLockReleaseID with something that will guarantee a check
_dwLastLockReleaseID = _pic->_GetLastLockReleaseID() - 1; } else { // don't bother checking for crossed anchors, since it can't happen
_dwLastLockReleaseID = IGNORE_LAST_LOCKRELEASED; } }
CStructArray<GENERICSINK> *_prgChangeSinks; // ITfRangeChangeSink sinks
IAnchor *_paStart; IAnchor *_paEnd;
CInputContext *_pic; CRange *_nextOnChangeRangeInIcsub; // perf: could use an array in the pic to save space
DWORD _dwLastLockReleaseID;
BOOL _fDirty : 1;
long _cRef;
DBG_ID_DECLARE; };
// this call doesn't AddRef the object!
inline CRange *GetCRange_NA(IUnknown *range) { CRange *prange;
range->QueryInterface(IID_PRIV_CRANGE, (void **)&prange);
return prange; }
// returns TRUE if range is in the same context
inline BOOL VerifySameContext(CRange *pRange1, CRange *pRange2) { Assert((pRange1->_GetContext() == pRange2->_GetContext())); return (pRange1->_GetContext() == pRange2->_GetContext()); } // returns TRUE if range is in the same context
inline BOOL VerifySameContext(CInputContext *pContext, CRange *pRange) { Assert((pRange->_GetContext() == pContext)); return (pRange->_GetContext() == pContext); } // returns TRUE if range is in the same context
inline BOOL VerifySameContext(CInputContext *pContext, ITfRange *pTargetRange) { CRange *pRange = GetCRange_NA(pTargetRange);
Assert((pRange != NULL) && (pRange->_GetContext() == pContext)); return (pRange != NULL) && (pRange->_GetContext() == pContext); }
#endif // RANGE_H
|