-// TSF のデバッグ表示を行うかどうか?\r
-//#define TSF_DEBUG_OUTPUT\r
-//#define TSF_DEBUG_OUTPUT_DISPLAY_ATTR\r
-using System;\r
-using System.Collections.Generic;\r
-using System.Linq;\r
-using System.Text;\r
-using System.Diagnostics;\r
-using System.Reflection;\r
-using System.Runtime.CompilerServices;\r
-using System.Runtime.InteropServices;\r
-using System.Globalization;\r
-\r
-using DotNetTextStore.UnmanagedAPI.TSF;\r
-using DotNetTextStore.UnmanagedAPI.TSF.TextStore;\r
-using DotNetTextStore.UnmanagedAPI.WinDef;\r
-using DotNetTextStore.UnmanagedAPI.WinError;\r
-\r
-namespace DotNetTextStore\r
-{\r
-#if TSF_DEBUG_OUTPUT\r
- /// <summary>コールスタックの階層にあわせてインデントしてデバッグ表示するクラス。</summary>\r
- public class DebugOut : IDisposable\r
- {\r
- public DebugOut(string i_string, params object[] i_params)\r
- {\r
- _text = string.Format(i_string, i_params);\r
-\r
- s_callCount++;\r
- Debug.WriteLine("");\r
- Debug.WriteLine(string.Format("{0, 4} : ↓↓↓ ", s_callCount) + _text);\r
- }\r
-\r
- public static void Print(string i_string, params object[] i_params)\r
- {\r
- Debug.WriteLine(i_string, i_params);\r
- }\r
-\r
- public void Dispose()\r
- {\r
- s_callCount++;\r
- Debug.WriteLine(string.Format("{0, 4} : ↑↑↑ ", s_callCount) + _text);\r
- }\r
-\r
- public static string GetCaller([CallerMemberName] string caller="")\r
- {\r
- return caller;\r
- }\r
-\r
- string _text;\r
- static int s_callCount = 0;\r
- }\r
-#endif\r
-\r
- //=============================================================================================\r
-\r
-\r
- public struct TextDisplayAttribute\r
- {\r
- public int startIndex;\r
- public int endIndex;\r
- public TF_DISPLAYATTRIBUTE attribute;\r
- }\r
-\r
- //========================================================================================\r
-\r
-\r
- /// <summary>Dispose() で TextStore のロック解除を行うクラス。</summary>\r
- public class Unlocker : IDisposable\r
- {\r
- /// <summary>コンストラクタ</summary>\r
- public Unlocker(TextStoreBase io_textStore)\r
- {\r
- _textStore = io_textStore;\r
- }\r
- /// <summary>ロックが成功したかどうか調べる。</summary>\r
- public bool IsLocked\r
- {\r
- get { return _textStore != null; }\r
- }\r
- /// <summary>アンロックを行う。</summary>\r
- void IDisposable.Dispose()\r
- {\r
- if (_textStore != null)\r
- {\r
- _textStore.UnlockDocument();\r
- _textStore = null;\r
- }\r
- }\r
-\r
- /// <summary>アンロックを行うテキストストア</summary>\r
- TextStoreBase _textStore;\r
- }\r
-\r
- public abstract class TextStoreBase\r
- {\r
- public delegate bool IsReadOnlyHandler();\r
- public event IsReadOnlyHandler IsReadOnly;\r
-\r
- public delegate bool IsLoadingHandler();\r
- public event IsLoadingHandler IsLoading;\r
-\r
- public delegate int GetStringLengthHandler();\r
- public event GetStringLengthHandler GetStringLength;\r
-\r
- public delegate void GetSelectionIndexHandler(out int o_start, out int o_end);\r
- public event GetSelectionIndexHandler GetSelectionIndex;\r
-\r
- public delegate void SetSelectionIndexHandler(int i_start, int i_end);\r
- public event SetSelectionIndexHandler SetSelectionIndex;\r
-\r
- public delegate string GetStringHandler(int start, int length);\r
- public event GetStringHandler GetString;\r
-\r
- public delegate void InsertAtSelectionHandler(string i_value,ref int o_startIndex,ref int o_endIndex);\r
- public event InsertAtSelectionHandler InsertAtSelection;\r
-\r
- public delegate void GetScreenExtentHandler(\r
- out POINT o_pointTopLeft,\r
- out POINT o_pointBottomRight\r
- );\r
- public event GetScreenExtentHandler GetScreenExtent;\r
-\r
- public delegate void GetStringExtentHandler(\r
- int i_startIndex,\r
- int i_endIndex,\r
- out POINT o_pointTopLeft,\r
- out POINT o_pointBottomRight\r
- );\r
- public event GetStringExtentHandler GetStringExtent;\r
-\r
- public delegate bool CompositionStartedHandler();\r
- public event CompositionStartedHandler CompositionStarted;\r
-\r
- public delegate void CompostionUpdateHandler(int start, int end);\r
- public event CompostionUpdateHandler CompositionUpdated;\r
-\r
- public delegate void CompositionEndedHandler();\r
- public event CompositionEndedHandler CompositionEnded;\r
-\r
- #region "生成と破棄"\r
- public TextStoreBase()\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- try\r
- {\r
- // スレッドマネージャ-の生成\r
- CreateThreadMgr();\r
- // カテゴリマネージャーの生成\r
- CreateCategoryMgr();\r
- // 表示属性マネージャーの生成\r
- CreateDisplayAttributeMgr();\r
-\r
- // ドキュメントマネージャーの生成\r
- _threadMgr.CreateDocumentMgr(out _documentMgr);\r
-\r
- // スレッドマネージャのアクティブ化\r
- int clientId = 0;\r
- _threadMgr.Activate(out clientId);\r
-\r
- // コンテキストの生成\r
- _documentMgr.CreateContext(clientId, 0, this, out _context, out _editCookie);\r
-\r
- // コンテキストの push\r
- _documentMgr.Push(_context);\r
-\r
- // ファンクションプロバイダーを取得する。\r
- Guid guid = TfDeclarations.GUID_SYSTEM_FUNCTIONPROVIDER;\r
- _threadMgr.GetFunctionProvider(ref guid, out _functionProvider);\r
-\r
- // ITfReconversion オブジェクトを取得する。\r
- var guidNull = new Guid();\r
- var guidReconversion = new Guid("4cea93c0-0a58-11d3-8df0-00105a2799b5"); //ITfFnReconversionの定義から\r
- object reconversion = null;\r
- _functionProvider.GetFunction(\r
- ref guidNull,\r
- ref guidReconversion,\r
- out reconversion\r
- );\r
- _reconversion = reconversion as ITfFnReconversion;\r
-\r
- // MODEBIAS の初期化\r
- uint guidAtom = 0;\r
- Guid guidModebiasNone = TfDeclarations.GUID_MODEBIAS_NONE;\r
- _categoryMgr.RegisterGUID(ref guidModebiasNone, out guidAtom);\r
- _attributeInfo[0].attrID = TfDeclarations.GUID_PROP_MODEBIAS;\r
- _attributeInfo[0].flags = AttributeInfoFlags.None;\r
- _attributeInfo[0].currentValue.vt = (short)VarEnum.VT_EMPTY;\r
- _attributeInfo[0].defaultValue.vt = (short)VarEnum.VT_I4;\r
- _attributeInfo[0].defaultValue.data1 = (IntPtr)guidAtom;\r
- }\r
- catch (Exception exception)\r
- {\r
- Debug.WriteLine(exception.Message);\r
- Dispose(false);\r
- }\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// オブジェクトの破棄を行う。このメソッドは必ず呼び出す必要があります\r
- /// </summary>\r
- /// <param name="flag"></param>\r
- protected void Dispose(bool flag)\r
- {\r
- if (flag)\r
- return;\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- ReleaseComObject("_reconversion", ref _reconversion);\r
- ReleaseComObject("_functionProvider", ref _functionProvider);\r
- ReleaseComObject("_context", ref _context);\r
- DestroyDocumentMgr();\r
- DestroyDisplayAttributeMgr();\r
- DestroyCategoryMgr();\r
- DestroyThreadMgr();\r
-\r
- GC.SuppressFinalize(this);\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// スレッドマネージャの生成。このメソッドの実装は必須です\r
- /// </summary>\r
- /// \r
- /// <exception cref="COMException">\r
- /// スレッドマネージャーの生成に失敗した場合。\r
- /// </exception>\r
- protected virtual void CreateThreadMgr()\r
- {\r
- throw new NotImplementedException();\r
- }\r
-\r
- /// <summary>\r
- /// スレッドマネージャーの解放。\r
- /// </summary>\r
- void DestroyThreadMgr()\r
- {\r
- if (_threadMgr != null)\r
- {\r
- try { _threadMgr.Deactivate(); }\r
- catch (Exception) { }\r
-\r
- ReleaseComObject("_threadMgr", ref _threadMgr);\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// カテゴリマネージャーの生成。\r
- /// </summary>\r
- /// \r
- /// <exception cref="COMException">\r
- /// カテゴリマネージャーの生成に失敗した場合。\r
- /// </exception>\r
- void CreateCategoryMgr()\r
- {\r
- if (_categoryMgr == null)\r
- {\r
- var clsid = Marshal.GetTypeFromCLSID(TfDeclarations.CLSID_TF_CategoryMgr);\r
- _categoryMgr = Activator.CreateInstance(clsid) as ITfCategoryMgr;\r
-\r
- if (_categoryMgr == null)\r
- {\r
- const string message = "カテゴリマネージャーの生成に失敗しました。";\r
- Debug.WriteLine(message);\r
- throw new COMException(message, HRESULT.E_NOTIMPL);\r
- }\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// カテゴリマネージャーの解放。\r
- /// </summary>\r
- void DestroyCategoryMgr()\r
- {\r
- ReleaseComObject("_categoryMgr", ref _categoryMgr);\r
- }\r
-\r
- /// <summary>\r
- /// 表示属性マネージャーの生成。\r
- /// </summary>\r
- /// \r
- /// <exception cref="COMException">\r
- /// 表示属性マネージャーの生成に失敗した場合。\r
- /// </exception>\r
- void CreateDisplayAttributeMgr()\r
- {\r
- if (_displayAttributeMgr == null)\r
- {\r
- var clsid = Marshal.GetTypeFromCLSID(TfDeclarations.CLSID_TF_DisplayAttributeMgr);\r
- _displayAttributeMgr = Activator.CreateInstance(clsid) as ITfDisplayAttributeMgr;\r
-\r
- if (_displayAttributeMgr == null)\r
- {\r
- const string message = "表示属性マネージャーの生成に失敗しました。";\r
- Debug.WriteLine(message);\r
- throw new COMException(message, HRESULT.E_NOTIMPL);\r
- }\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// 表示属性マネージャーの解放。\r
- /// </summary>\r
- void DestroyDisplayAttributeMgr()\r
- {\r
- ReleaseComObject("_displayAttributeMgr", ref _displayAttributeMgr);\r
- }\r
-\r
- /// <summary>\r
- /// ドキュメントマネージャーの解放\r
- /// </summary>\r
- void DestroyDocumentMgr()\r
- {\r
- if (_documentMgr != null)\r
- {\r
- try { _documentMgr.Pop(PopFlags.TF_POPF_ALL); }\r
- catch (Exception) { }\r
-\r
- ReleaseComObject("_documentMgr", ref _documentMgr);\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// COM オブジェクトのリリースとデバッグメッセージ出力。\r
- /// </summary>\r
- protected static void ReleaseComObject<ComObject>(\r
- string i_objectName,\r
- ref ComObject io_comObject\r
- )\r
- {\r
- if (io_comObject != null)\r
- {\r
- var refCount = Marshal.ReleaseComObject(io_comObject);\r
-#if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine(\r
- "Marshal.ReleaseComObject({0}) returns {1}.",\r
- i_objectName,\r
- refCount\r
- );\r
-#endif\r
-\r
- io_comObject = default(ComObject);\r
- }\r
- }\r
- #endregion\r
-\r
- #region "コントロール側が状況に応じて呼び出さなければいけない TSF に通知を送るメソッド"\r
- /// <summary>\r
- /// 選択領域が変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。\r
- /// </summary>\r
- public void NotifySelectionChanged()\r
- {\r
- if( _sink != null )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print(DebugOut.GetCaller());\r
-#endif\r
- if ((_adviseFlags & AdviseFlags.TS_AS_SEL_CHANGE) != 0)\r
- _sink.OnSelectionChange();\r
- }\r
- }\r
-\r
- \r
- //=========================================================================================\r
-\r
-\r
- /// <summary>\r
- /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。\r
- /// </summary>\r
- void NotifyTextChanged(TS_TEXTCHANGE textChange)\r
- {\r
- if( _sink != null )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print(DebugOut.GetCaller());\r
-#endif\r
- if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)\r
- _sink.OnTextChange(0, ref textChange);\r
- _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);\r
- }\r
- }\r
-\r
- \r
- //=========================================================================================\r
-\r
- /// <summary>\r
- /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。\r
- /// </summary>\r
- /// <param name="start">開始位置</param>\r
- /// <param name="oldend">更新前の終了位置</param>\r
- /// <param name="newend">更新後の終了位置</param>\r
- /// <remarks>\r
- /// 詳しいことはITextStoreACPSink::OnTextChangeを参照してください\r
- /// </remarks>\r
- public void NotifyTextChanged(int start,int oldend,int newend)\r
- {\r
- if (_sink != null)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print(DebugOut.GetCaller());\r
-#endif\r
- if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)\r
- {\r
- var textChange = new TS_TEXTCHANGE();\r
- textChange.start = start;\r
- textChange.oldEnd = oldend;\r
- textChange.newEnd = newend;\r
-\r
- _sink.OnTextChange(0, ref textChange);\r
- }\r
- _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。\r
- /// </summary>\r
- /// <param name="i_oldLength"></param>\r
- /// <param name="i_newLength"></param>\r
- public void NotifyTextChanged(int i_oldLength, int i_newLength)\r
- {\r
- if( _sink != null )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print(DebugOut.GetCaller());\r
-#endif\r
- if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)\r
- {\r
- var textChange = new TS_TEXTCHANGE();\r
- textChange.start = 0;\r
- textChange.oldEnd = i_oldLength;\r
- textChange.newEnd = i_newLength;\r
-\r
- _sink.OnTextChange(0, ref textChange);\r
- }\r
- _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);\r
- }\r
- }\r
-\r
- \r
- \r
- //=========================================================================================\r
-\r
-\r
- /// <summary>コントロールがフォーカスを取得した時に呼び出さなければいけない。</summary>\r
- public void SetFocus()\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print(DebugOut.GetCaller());\r
-#endif\r
- if (_threadMgr != null)\r
- _threadMgr.SetFocus(_documentMgr);\r
- }\r
- #endregion "コントロール側が状況に応じて呼び出さなければいけない TSF に通知を送るメソッド"\r
-\r
- #region ロック関連\r
- /// <summary>\r
- /// ドキュメントのロックを行う。\r
- /// </summary>\r
- /// <param name="i_writable">読み書き両用ロックか?false の場合、読み取り専用。</param>\r
- /// <returns>Unlocker のインスタンスを返す。失敗した場合 null を返す。</returns>\r
- public Unlocker LockDocument(bool i_writable)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}({1})", DebugOut.GetCaller(),\r
- i_writable) )\r
- #endif\r
- {\r
- lock(this)\r
- {\r
- if( this._lockFlags == 0 )\r
- {\r
- if( i_writable )\r
- this._lockFlags = LockFlags.TS_LF_READWRITE;\r
- else\r
- this._lockFlags = LockFlags.TS_LF_READ;\r
-\r
- #if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine("LockDocument is succeeded.");\r
- #endif\r
-\r
- return new Unlocker(this);\r
- }\r
- else\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine("LockDocument is failed. {0}", _lockFlags);\r
- #endif\r
-\r
- return null;\r
- }\r
- }\r
- }\r
- }\r
-\r
-\r
- //========================================================================================\r
-\r
- \r
- /// <summary>\r
- /// ドキュメントのロックを行う。\r
- /// </summary>\r
- /// <param name="i_writable">読み書き両用ロックか?false の場合、読み取り専用。</param>\r
- /// <returns>Unlocker のインスタンスを返す。失敗した場合 null を返す。</returns>\r
- public Unlocker LockDocument(LockFlags i_flags)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}({1})", DebugOut.GetCaller(),\r
- i_flags) )\r
- #endif\r
- {\r
- lock(this)\r
- {\r
- if( this._lockFlags == 0 )\r
- {\r
- this._lockFlags = i_flags;\r
-\r
- #if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine("LockDocument is succeeded.");\r
- #endif\r
-\r
- return new Unlocker(this);\r
- }\r
- else\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine("LockDocument is failed. {0}", _lockFlags);\r
- #endif\r
-\r
- return null;\r
- }\r
- }\r
- }\r
- }\r
-\r
-\r
- //========================================================================================\r
-\r
- \r
- /// <summary>\r
- /// ドキュメントのアンロックを行う。\r
- /// </summary>\r
- public void UnlockDocument()\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
- #endif\r
- {\r
- lock(this)\r
- {\r
- _lockFlags = 0;\r
- }\r
-\r
- if( _pendingLockUpgrade )\r
- {\r
- _pendingLockUpgrade = false;\r
- int sessionResult;\r
- RequestLock(LockFlags.TS_LF_READWRITE, out sessionResult);\r
- }\r
- }\r
- }\r
-\r
-\r
- //========================================================================================\r
-\r
- \r
- /// <summary>\r
- /// 指定されたフラグでロックしている状態かどうか調べる。\r
- /// </summary>\r
- /// <param name="i_lockFlags"></param>\r
- /// <returns>ロックされている場合は true, されていない場合は false を返す。</returns>\r
- public bool IsLocked(LockFlags i_lockFlags)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}({1})",\r
- DebugOut.GetCaller(), i_lockFlags) )\r
- #endif\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine(\r
- "IsLocked() returns " + ((this._lockFlags & i_lockFlags) == i_lockFlags)\r
- );\r
- #endif\r
- return (this._lockFlags & i_lockFlags) == i_lockFlags;\r
- }\r
- }\r
-\r
- /// <summary>\r
- /// ロックされているか調べる\r
- /// </summary>\r
- /// <returns>ロックされているなら真、そうでなければ偽を返す</returns>\r
- public bool IsLocked()\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}",\r
- DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- bool retval = this._lockFlags != 0;\r
-#if TSF_DEBUG_OUTPUT\r
- Debug.WriteLine(\r
- "IsLocked() returns " + retval\r
- );\r
-#endif\r
- return retval;\r
- }\r
- }\r
- #endregion\r
-\r
- #region ITextStroeACP,ITextStoreACP2の共通部分\r
-\r
- /// <summary>\r
- /// 文字列を挿入する。\r
- /// </summary>\r
- public void InsertTextAtSelection(string s)\r
- {\r
- TS_TEXTCHANGE textChange = new TS_TEXTCHANGE();\r
-\r
- using (var unlocker = LockDocument(true))\r
- {\r
- if (unlocker != null)\r
- {\r
- int startIndex, endIndex;\r
-\r
- InsertTextAtSelection(\r
- UnmanagedAPI.TSF.TextStore.InsertAtSelectionFlags.TF_IAS_NOQUERY,\r
- s.ToCharArray(),\r
- s.Length,\r
- out startIndex,\r
- out endIndex,\r
- out textChange\r
- );\r
- }\r
- }\r
-\r
- // シンクの OnSelectionChange() をコール。\r
- NotifySelectionChanged();\r
- NotifyTextChanged(textChange);\r
- }\r
-\r
- public void InsertEmbeddedAtSelection(\r
- InsertAtSelectionFlags flags,\r
- object obj,\r
- out int start,\r
- out int end,\r
- out TS_TEXTCHANGE change\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void AdviseSink(ref Guid i_riid, object i_unknown, AdviseFlags i_mask)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (i_riid != new Guid("22d44c94-a419-4542-a272-ae26093ececf")) //ITextStoreACPSinkの定義より\r
- {\r
- throw new COMException(\r
- "ITextStoreACPSink 以外のIIDが渡されました。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
-\r
- // 既存のシンクのマスクのみを更新\r
- if (_sink == i_unknown)\r
- {\r
- _adviseFlags = i_mask;\r
- }\r
- // シンクを複数登録しようとした\r
- else if (_sink != null)\r
- {\r
- throw new COMException(\r
- "既にシンクを登録済みです。",\r
- UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_ADVISELIMIT\r
- );\r
- }\r
- else\r
- {\r
- // 各種値を保存\r
- _services = (ITextStoreACPServices)i_unknown;\r
- _sink = (ITextStoreACPSink)i_unknown;\r
- _adviseFlags = i_mask;\r
- }\r
- }\r
- }\r
-\r
- public void UnadviseSink(object i_unknown)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (_sink == null || _sink != i_unknown)\r
- {\r
- throw new COMException(\r
- "シンクは登録されていません。",\r
- UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_NOCONNECTION\r
- );\r
- }\r
-\r
- _services = null;\r
- _sink = null;\r
- _adviseFlags = 0;\r
- }\r
- }\r
-\r
- public void RequestLock(LockFlags i_lockFlags, out int o_sessionResult)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- o_sessionResult = UnmanagedAPI.WinError.HRESULT.E_FAIL;\r
-\r
- if (_sink == null)\r
- {\r
- throw new COMException(\r
- "シンクが登録されていません。",\r
- UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_NOCONNECTION\r
- );\r
- }\r
-\r
- if (_lockFlags != 0) // すでにロックされた状態の場合。\r
- {\r
- if ((i_lockFlags & LockFlags.TS_LF_SYNC) != 0)\r
- {\r
- o_sessionResult = TsResult.TS_E_SYNCHRONOUS;\r
- return;\r
- }\r
- else\r
- if ((_lockFlags & LockFlags.TS_LF_READWRITE) == LockFlags.TS_LF_READ\r
- && (i_lockFlags & LockFlags.TS_LF_READWRITE) == LockFlags.TS_LF_READWRITE)\r
- {\r
- _pendingLockUpgrade = true;\r
- o_sessionResult = TsResult.TS_S_ASYNC;\r
- return;\r
- }\r
-\r
- throw new COMException();\r
- }\r
-\r
- using (var unlocker = LockDocument(i_lockFlags))\r
- {\r
- // ロックに失敗した場合は TS_E_SYNCHRONOUS をセットして S_OK を返す。\r
- if (unlocker == null)\r
- {\r
- o_sessionResult = TsResult.TS_E_SYNCHRONOUS;\r
- }\r
- // ロックに成功した場合は OnLockGranted() を呼び出す。\r
- else\r
- {\r
- try\r
- {\r
- o_sessionResult = _sink.OnLockGranted(i_lockFlags);\r
- }\r
- catch (COMException comException)\r
- {\r
- Debug.WriteLine("OnLockGranted() 呼び出し中に例外が発生。");\r
- Debug.WriteLine(" " + comException.Message);\r
- o_sessionResult = comException.HResult;\r
- }\r
- catch (Exception exception)\r
- {\r
- Debug.WriteLine("OnLockGranted() 呼び出し中に例外が発生。");\r
- Debug.WriteLine(" " + exception.Message);\r
- }\r
- }\r
- }\r
- }\r
- }\r
-\r
- public void GetStatus(out TS_STATUS o_documentStatus)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (IsReadOnly != null && IsReadOnly())\r
- o_documentStatus.dynamicFlags = DynamicStatusFlags.TS_SD_READONLY;\r
- if (IsLoading != null && IsLoading())\r
- o_documentStatus.dynamicFlags = DynamicStatusFlags.TS_SD_LOADING;\r
- else\r
- o_documentStatus.dynamicFlags = 0;\r
- o_documentStatus.staticFlags = StaticStatusFlags.TS_SS_REGIONS;\r
- }\r
- }\r
-\r
- public void QueryInsert(\r
- int i_startIndex,\r
- int i_endIndex,\r
- int i_length,\r
- out int o_startIndex,\r
- out int o_endIndex\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetStringLength == null)\r
- throw new NotImplementedException();\r
-\r
- int documentLength = GetStringLength();\r
-\r
- if (i_startIndex < 0\r
- || i_startIndex > i_endIndex\r
- || i_endIndex > documentLength)\r
- {\r
- throw new COMException(\r
- "インデックスが無効です。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
- o_startIndex = i_startIndex;\r
- o_endIndex = i_endIndex;\r
-\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("o_startIndex:{0} o_endIndex:{1}", i_startIndex, i_endIndex);\r
-#endif\r
- }\r
- }\r
-\r
- public void GetSelection(\r
- int i_index,\r
- int i_selectionBufferLength,\r
- TS_SELECTION_ACP[] o_selections,\r
- out int o_fetchedLength\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetSelectionIndex == null)\r
- throw new NotImplementedException();\r
-\r
- o_fetchedLength = 0;\r
-\r
- if (IsLocked(LockFlags.TS_LF_READ) == false)\r
- {\r
- throw new COMException(\r
- "読取用ロックがされていません。",\r
- TsResult.TS_E_NOLOCK\r
- );\r
- }\r
-\r
- // -1 は TF_DEFAULT_SELECTION。選択は1つだけしかサポートしないので、\r
- // TF_DEFAULT_SELECTION でもなく、0 を超える数値が指定された場合はエラー。\r
- if (i_index != -1 && i_index > 0)\r
- {\r
- throw new COMException(\r
- "選択は1つだけしかサポートしていません。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
-\r
- if (i_selectionBufferLength > 0)\r
- {\r
- int start = 0, end = 0;\r
- GetSelectionIndex(out start, out end);\r
- if (start <= end)\r
- {\r
- o_selections[0].start = start;\r
- o_selections[0].end = end;\r
- o_selections[0].style.ase = TsActiveSelEnd.TS_AE_END;\r
- o_selections[0].style.interimChar = false;\r
- }\r
- else\r
- {\r
- o_selections[0].start = end;\r
- o_selections[0].end = start;\r
- o_selections[0].style.ase = TsActiveSelEnd.TS_AE_START;\r
- o_selections[0].style.interimChar = false;\r
- }\r
-\r
- o_fetchedLength = 1;\r
-\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("sel start:{0} end:{1}", start, end);\r
-#endif\r
- }\r
- }\r
- }\r
-\r
- public void SetSelection(int i_count, TS_SELECTION_ACP[] i_selections)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}({1}, {2})",\r
- DebugOut.GetCaller(),\r
- i_selections[0].start,\r
- i_selections[0].end))\r
-#endif\r
- {\r
- if (SetSelectionIndex == null)\r
- throw new NotImplementedException();\r
-\r
- if (i_count != 1)\r
- {\r
- throw new COMException(\r
- "選択は1つだけしかサポートしていません。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
-\r
- if (IsLocked(LockFlags.TS_LF_READWRITE) == false)\r
- {\r
- throw new COMException(\r
- "ロックされていません。",\r
- TsResult.TS_E_NOLOCK\r
- );\r
- }\r
-\r
- SetSelectionIndex(i_selections[0].start, i_selections[0].end);\r
-\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("set selection startIndex:{0} endIndex:{1}", i_selections[0].start, i_selections[0].end);\r
-#endif\r
- }\r
- }\r
-\r
- public void GetText(\r
- int i_startIndex,\r
- int i_endIndex,\r
- char[] o_plainText,\r
- int i_plainTextLength,\r
- out int o_plainTextLength,\r
- TS_RUNINFO[] o_runInfos,\r
- int i_runInfoLength,\r
- out int o_runInfoLength,\r
- out int o_nextUnreadCharPos\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetStringLength == null || GetString == null)\r
- throw new NotImplementedException();\r
-\r
- if (IsLocked(LockFlags.TS_LF_READ) == false)\r
- {\r
- throw new COMException(\r
- "読取用ロックがされていません。",\r
- TsResult.TS_E_NOLOCK\r
- );\r
- }\r
-\r
- if ((i_endIndex != -1 && i_startIndex > i_endIndex)\r
- || i_startIndex < 0 || i_startIndex > GetStringLength()\r
- || i_endIndex > GetStringLength())\r
- {\r
- throw new COMException(\r
- "インデックスが無効です。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
-\r
- var textLength = 0;\r
- var copyLength = 0;\r
-\r
- if (i_endIndex == -1)\r
- textLength = GetStringLength() - i_startIndex;\r
- else\r
- textLength = i_endIndex - i_startIndex;\r
- copyLength = Math.Min(i_plainTextLength, textLength);\r
-\r
- // 文字列を格納。\r
- var text = GetString(i_startIndex, copyLength);\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("got text:{0} from {1} length {2}", text,i_startIndex,copyLength);\r
-#endif\r
- for (int i = 0; i < copyLength; i++)\r
- {\r
- o_plainText[i] = text[i];\r
- }\r
-\r
- // 文字数を格納。\r
- o_plainTextLength = copyLength;\r
- // RUNINFO を格納。\r
- if (i_runInfoLength > 0)\r
- {\r
- o_runInfos[0].type = TsRunType.TS_RT_PLAIN;\r
- o_runInfos[0].length = copyLength;\r
- }\r
- o_runInfoLength = 1;\r
-\r
- // 次の文字の位置を格納。\r
- o_nextUnreadCharPos = i_startIndex + copyLength;\r
- }\r
- }\r
-\r
- public void SetText(\r
- SetTextFlags i_flags,\r
- int i_startIndex,\r
- int i_endIndex,\r
- char[] i_text,\r
- int i_length,\r
- out TS_TEXTCHANGE o_textChange\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}({1}, {2})",\r
- DebugOut.GetCaller(),\r
- i_startIndex, i_endIndex) )\r
-#endif\r
- {\r
- var selections = new TS_SELECTION_ACP[]\r
- {\r
- new TS_SELECTION_ACP\r
- {\r
- start = i_startIndex,\r
- end = i_endIndex,\r
- style = new TS_SELECTIONSTYLE\r
- {\r
- ase = TsActiveSelEnd.TS_AE_END,\r
- interimChar = false\r
- }\r
- }\r
- };\r
-\r
- int startIndex = 0, endIndex = 0;\r
- SetSelection(1, selections);\r
- InsertTextAtSelection(\r
- InsertAtSelectionFlags.TF_IAS_NOQUERY,\r
- i_text,\r
- i_length,\r
- out startIndex,\r
- out endIndex,\r
- out o_textChange\r
- );\r
- }\r
- }\r
-\r
- public void GetFormattedText(int i_start, int i_end, out object o_obj)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void GetEmbedded(\r
- int i_position,\r
- ref Guid i_guidService,\r
- ref Guid i_riid,\r
- out object o_obj\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void QueryInsertEmbedded(\r
- ref Guid i_guidService,\r
- int i_formatEtc,\r
- out bool o_insertable\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void InsertEmbedded(\r
- InsertEmbeddedFlags i_flags,\r
- int i_start,\r
- int i_end,\r
- object i_obj,\r
- out TS_TEXTCHANGE o_textChange\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void InsertTextAtSelection(\r
- InsertAtSelectionFlags i_flags,\r
- char[] i_text,\r
- int i_length,\r
- out int o_startIndex,\r
- out int o_endIndex,\r
- out TS_TEXTCHANGE o_textChange\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetSelectionIndex == null || InsertAtSelection == null)\r
- throw new NotImplementedException();\r
-\r
- int selectionStart = 0, selectionEnd = 0;\r
- GetSelectionIndex(out selectionStart, out selectionEnd);\r
-\r
- // 問い合わせのみで実際には操作を行わない\r
- if ((i_flags & InsertAtSelectionFlags.TF_IAS_QUERYONLY) != 0)\r
- {\r
- o_startIndex = Math.Min(selectionStart, selectionEnd);\r
- o_endIndex = Math.Max(selectionStart, selectionEnd);//o_startIndex + i_length;\r
-\r
- o_textChange.start = o_startIndex;\r
- o_textChange.oldEnd = o_endIndex;\r
- o_textChange.newEnd = o_startIndex + i_length;\r
- }\r
- else\r
- {\r
- var start = Math.Min(selectionStart, selectionEnd);\r
- var end = Math.Max(selectionStart, selectionEnd);\r
-\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("start: {0}, end: {1}, text: {2}", start, end, new string(i_text));\r
-#endif\r
-\r
- o_startIndex = start;\r
- o_endIndex = start + i_length;\r
-\r
- InsertAtSelection(new string(i_text), ref o_startIndex, ref o_endIndex);\r
-\r
- o_textChange.start = start;\r
- o_textChange.oldEnd = end;\r
- o_textChange.newEnd = o_endIndex;\r
- // InsertAtSelection() 内でカーソル位置を更新しているため、ここでは不要。\r
- // 改行した時に位置が狂う。\r
- // SetSelectionIndex(start, start + i_length);\r
- }\r
- }\r
- }\r
-\r
- public void GetEndACP(out int o_length)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetStringLength == null)\r
- throw new NotImplementedException();\r
-\r
- if (IsLocked(LockFlags.TS_LF_READ) == false)\r
- {\r
- throw new COMException(\r
- "読取用ロックがされていません。",\r
- TsResult.TS_E_NOLOCK\r
- );\r
- }\r
-\r
- o_length = GetStringLength();\r
- }\r
- }\r
-\r
- public void GetActiveView(out int o_viewCookie)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- o_viewCookie = 1;\r
- }\r
- }\r
-\r
- public void GetACPFromPoint(\r
- int i_viewCookie,\r
- ref POINT i_point,\r
- GetPositionFromPointFlags i_flags,\r
- out int o_index\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using (var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void GetTextExt(\r
- int i_viewCookie,\r
- int i_startIndex,\r
- int i_endIndex,\r
- out RECT o_rect,\r
- out bool o_isClipped\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetStringLength == null)\r
- throw new NotImplementedException();\r
-\r
- // 読取用ロックの確認。\r
- if (IsLocked(LockFlags.TS_LF_READ) == false)\r
- {\r
- throw new COMException(\r
- "読取用ロックがされていません。",\r
- TsResult.TS_E_NOLOCK\r
- );\r
- }\r
-\r
- if ((i_endIndex != -1 && i_startIndex > i_endIndex)\r
- || i_startIndex < 0 || i_startIndex > GetStringLength()\r
- || i_endIndex > GetStringLength())\r
- {\r
- throw new COMException(\r
- "インデックスが無効です。",\r
- UnmanagedAPI.WinError.HRESULT.E_INVALIDARG\r
- );\r
- }\r
-\r
- if (i_endIndex == -1)\r
- i_endIndex = GetStringLength();\r
-\r
- var pointTopLeft = new POINT();\r
- var pointBotttomRight = new POINT();\r
- GetStringExtent(i_startIndex, i_endIndex, out pointTopLeft, out pointBotttomRight);\r
-\r
- o_rect.left = (int)(pointTopLeft.x);\r
- o_rect.top = (int)(pointTopLeft.y);\r
- o_rect.bottom = (int)(pointBotttomRight.y);\r
- o_rect.right = (int)(pointBotttomRight.x);\r
- o_isClipped = false;\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("rect left:{0} top:{1} bottom:{2} right:{3}", o_rect.left, o_rect.top, o_rect.bottom, o_rect.right);\r
-#endif\r
- }\r
- }\r
-\r
- public void GetScreenExt(int i_viewCookie, out RECT o_rect)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- if (GetScreenExtent == null)\r
- throw new NotImplementedException();\r
-\r
- POINT pointTopLeft, pointBottomRight;\r
-\r
- GetScreenExtent(out pointTopLeft, out pointBottomRight);\r
-\r
- o_rect.left = (int)(pointTopLeft.x);\r
- o_rect.top = (int)(pointTopLeft.y);\r
- o_rect.bottom = (int)(pointBottomRight.y);\r
- o_rect.right = (int)(pointBottomRight.x);\r
-#if TSF_DEBUG_OUTPUT\r
- DebugOut.Print("rect left:{0} top:{1} bottom:{2} right:{3}", o_rect.left, o_rect.top, o_rect.bottom, o_rect.right);\r
-#endif\r
- }\r
- }\r
-\r
- public void RequestSupportedAttrs(\r
- AttributeFlags i_flags,\r
- int i_length,\r
- Guid[] i_filterAttributes\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- for (int i = 0; i < i_length; i++)\r
- {\r
- if (i_filterAttributes[i].Equals(_attributeInfo[0].attrID))\r
- {\r
- _attributeInfo[0].flags = AttributeInfoFlags.Requested;\r
- if ((i_flags & AttributeFlags.TS_ATTR_FIND_WANT_VALUE) != 0)\r
- {\r
- _attributeInfo[0].flags |= AttributeInfoFlags.Default;\r
- }\r
- else\r
- {\r
- _attributeInfo[0].currentValue = _attributeInfo[0].defaultValue;\r
- }\r
- }\r
- }\r
- }\r
- }\r
-\r
- public void RequestAttrsAtPosition(\r
- int i_position,\r
- int i_length,\r
- Guid[] i_filterAttributes,\r
- AttributeFlags i_flags\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- for (int i = 0; i < i_length; i++)\r
- {\r
- if (i_filterAttributes[i].Equals(_attributeInfo[0].attrID))\r
- {\r
- _attributeInfo[0].flags = AttributeInfoFlags.Requested;\r
- if ((i_flags & AttributeFlags.TS_ATTR_FIND_WANT_VALUE) != 0)\r
- {\r
- _attributeInfo[0].flags |= AttributeInfoFlags.Default;\r
- }\r
- else\r
- {\r
- _attributeInfo[0].currentValue = _attributeInfo[0].defaultValue;\r
- }\r
- }\r
- }\r
- }\r
- }\r
-\r
- public void RequestAttrsTransitioningAtPosition(\r
- int i_position,\r
- int i_length,\r
- Guid[] i_filterAttributes,\r
- AttributeFlags i_flags\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))\r
-#endif\r
- {\r
- // 何もしない。\r
- }\r
- }\r
-\r
- public void FindNextAttrTransition(\r
- int i_start,\r
- int i_halt,\r
- int i_length,\r
- Guid[] i_filterAttributes,\r
- AttributeFlags i_flags,\r
- out int o_nextIndex,\r
- out bool o_found,\r
- out int o_foundOffset\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- throw new NotImplementedException();\r
- }\r
- }\r
-\r
- public void RetrieveRequestedAttrs(\r
- int i_length,\r
- TS_ATTRVAL[] o_attributeVals,\r
- out int o_fetchedLength\r
- )\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
-#endif\r
- {\r
- o_fetchedLength = 0;\r
- for (int i = 0; i < _attributeInfo.Length && o_fetchedLength < i_length; i++)\r
- {\r
- if ((_attributeInfo[i].flags & AttributeInfoFlags.Requested) != 0)\r
- {\r
- o_attributeVals[o_fetchedLength].overlappedId = 0;\r
- o_attributeVals[o_fetchedLength].attributeId = _attributeInfo[i].attrID;\r
- o_attributeVals[o_fetchedLength].val = _attributeInfo[i].currentValue;\r
-\r
- if ((_attributeInfo[i].flags & AttributeInfoFlags.Default) != 0)\r
- _attributeInfo[i].currentValue = _attributeInfo[i].defaultValue;\r
-\r
- o_fetchedLength++;\r
- _attributeInfo[i].flags = AttributeInfoFlags.None;\r
- }\r
- }\r
- }\r
- }\r
- #endregion\r
-\r
- #region "ITfContextOwnerCompositionSink インターフェイスの実装"\r
- /// <summary>コンポジション入力が開始された時の処理。</summary>\r
- public void OnStartComposition(ITfCompositionView view, out bool ok)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
- #endif\r
- {\r
- if( CompositionStarted != null )\r
- ok = CompositionStarted();\r
- else\r
- ok = true;\r
- }\r
- }\r
-\r
-\r
- //========================================================================================\r
-\r
-\r
- /// <summary>コンポジションが変更された時の処理。</summary>\r
- public void OnUpdateComposition(ITfCompositionView view, ITfRange rangeNew)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
- #endif\r
- {\r
- if (rangeNew == null)\r
- return;\r
-\r
- int start, count;\r
- \r
- ITfRangeACP rangeAcp = (ITfRangeACP)rangeNew;\r
- \r
- rangeAcp.GetExtent(out start, out count);\r
- \r
- if (this.CompositionUpdated != null)\r
- this.CompositionUpdated(start, start + count);\r
- }\r
- }\r
-\r
- \r
- //========================================================================================\r
-\r
-\r
- /// <summary>コンポジション入力が終了した時の処理。</summary>\r
- public void OnEndComposition(ITfCompositionView view)\r
- {\r
- #if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )\r
- #endif\r
- {\r
- if( CompositionEnded != null )\r
- CompositionEnded();\r
- }\r
- }\r
- #endregion "ITfContextOwnerCompositionSink インターフェイスの実装"\r
-\r
- #region 表示属性\r
- /// <summary>表示属性の取得</summary>\r
- public IEnumerable<TextDisplayAttribute> EnumAttributes(int start, int end)\r
- {\r
- ITfRangeACP allRange;\r
- _services.CreateRange(start, end, out allRange);\r
-\r
- foreach (TextDisplayAttribute attr in this.EnumAttributes((ITfRange)allRange))\r
- yield return attr;\r
-\r
- ReleaseComObject("allRange", ref allRange);\r
- }\r
-\r
- IEnumerable<TextDisplayAttribute> EnumAttributes(ITfRange range)\r
- {\r
-#if TSF_DEBUG_OUTPUT\r
- using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))\r
-#endif\r
- {\r
- ITfProperty property = null; // プロパティインターフェイス\r
- IEnumTfRanges enumRanges; // 範囲の列挙子\r
- Guid guidPropAttribute = TfDeclarations.GUID_PROP_ATTRIBUTE;\r
-\r
- if (_context == null || _services == null)\r
- yield break;\r
-\r
- // GUID_PROP_ATTRIBUTE プロパティを取得。\r
- _context.GetProperty(ref guidPropAttribute, out property);\r
- if (property == null)\r
- yield break;\r
-\r
- // 全範囲の中で表示属性プロパティをもつ範囲を列挙する。\r
- property.EnumRanges((int)_editCookie, out enumRanges, range);\r
-\r
- ITfRange[] ranges = new ITfRange[1];\r
- int fetchedLength = 0;\r
- while (HRESULT.Succeeded(enumRanges.Next(1, ranges, out fetchedLength))\r
- && fetchedLength > 0)\r
- {\r
- // ItfRange から ItfRangeACP を取得。\r
- ITfRangeACP rangeACP = ranges[0] as ITfRangeACP;\r
- if (rangeACP == null)\r
- continue; // 普通はあり得ない\r
-\r
- // 範囲の開始位置と文字数を取得。\r
- int start, count;\r
- rangeACP.GetExtent(out start, out count);\r
-\r
- // VARIANT 値としてプロパティ値を取得。VT_I4 の GUID ATOM がとれる。\r
- VARIANT value = new VARIANT();\r
- property.GetValue((int)_editCookie, ranges[0], ref value);\r
- if (value.vt == (short)VarEnum.VT_I4)\r
- {\r
- Guid guid, clsid;\r
- ITfDisplayAttributeInfo info;\r
- TF_DISPLAYATTRIBUTE attribute;\r
-\r
- // GUID ATOM から GUID を取得する。\r
- _categoryMgr.GetGUID((int)value.data1, out guid);\r
- // その GUID から IDisplayAttributeInfo インターフェイスを取得。\r
- _displayAttributeMgr.GetDisplayAttributeInfo(\r
- ref guid,\r
- out info,\r
- out clsid\r
- );\r
- // さらに IDisplayAttributeInfo インターフェイスから表示属性を取得する。\r
- info.GetAttributeInfo(out attribute);\r
- ReleaseComObject("info", ref info);\r
-\r
- yield return new TextDisplayAttribute\r
- {\r
- startIndex = start,\r
- endIndex = start + count,\r
- attribute = attribute\r
- };\r
-\r
-#if TSF_DEBUG_OUTPUT_DISPLAY_ATTR\r
- Debug.WriteLine(\r
- "*******:::: DisplayAttribute: {0} ~ {1} :::::: *********",\r
- start, start + count\r
- );\r
- Debug.WriteLine(attribute.bAttr);\r
- Debug.WriteLine(\r
- "LineColorType: {0}, {1}",\r
- attribute.crLine.type, attribute.crLine.indexOrColorRef\r
- );\r
- Debug.WriteLine(\r
- "TextColorType: {0}, {1}",\r
- attribute.crText.type, attribute.crText.indexOrColorRef\r
- );\r
- Debug.WriteLine(\r
- "BackColorType: {0}, {1}",\r
- attribute.crBk.type, attribute.crBk.indexOrColorRef\r
- );\r
- Debug.WriteLine(\r
- "Bold, Style : {0}, {1}",\r
- attribute.fBoldLine, attribute.lsStyle\r
- );\r
-#endif\r
- }\r
-\r
- ReleaseComObject("rangeACP", ref rangeACP);\r
- }\r
-\r
- ReleaseComObject("ranges[0]", ref ranges[0]);\r
- ReleaseComObject("enumRanges", ref enumRanges);\r
- ReleaseComObject("property", ref property);\r
- }\r
- }\r
- #endregion\r
-\r
-#if METRO\r
- protected ITfThreadMgr2 _threadMgr;\r
-#else\r
- protected ITfThreadMgr _threadMgr;\r
-#endif\r
- protected ITfDocumentMgr _documentMgr;\r
- protected ITfFunctionProvider _functionProvider;\r
- protected ITfFnReconversion _reconversion;\r
- protected ITfContext _context;\r
- protected ITfCategoryMgr _categoryMgr;\r
- protected ITfDisplayAttributeMgr _displayAttributeMgr;\r
- protected ITextStoreACPSink _sink;\r
- protected uint _editCookie = 0;\r
- protected ITextStoreACPServices _services;\r
- protected AdviseFlags _adviseFlags = 0;\r
- protected LockFlags _lockFlags = 0;\r
- protected bool _pendingLockUpgrade = false;\r
-\r
- /// <summary>\r
- /// AttributeInfo で使用されるフラグ。各属性の状態を示す。\r
- /// </summary>\r
- protected enum AttributeInfoFlags\r
- {\r
- /// <summary>何もない。</summary>\r
- None = 0,\r
- /// <summary>デフォルト値の要求。</summary>\r
- Default = 1 << 0,\r
- /// <summary>要求された。</summary>\r
- Requested = 1 << 1\r
- }\r
-\r
- protected struct AttributeInfo\r
- {\r
- public Guid attrID;\r
- public AttributeInfoFlags flags;\r
- public VARIANT currentValue;\r
- public VARIANT defaultValue;\r
- }\r
-\r
- protected AttributeInfo[] _attributeInfo = new AttributeInfo[1];\r
-\r
- }\r
-}\r
+// TSF のデバッグ表示を行うかどうか?
+//#define TSF_DEBUG_OUTPUT
+//#define TSF_DEBUG_OUTPUT_DISPLAY_ATTR
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Diagnostics;
+using System.Reflection;
+using System.Runtime.CompilerServices;
+using System.Runtime.InteropServices;
+using System.Globalization;
+
+using DotNetTextStore.UnmanagedAPI.TSF;
+using DotNetTextStore.UnmanagedAPI.TSF.TextStore;
+using DotNetTextStore.UnmanagedAPI.WinDef;
+using DotNetTextStore.UnmanagedAPI.WinError;
+
+namespace DotNetTextStore
+{
+#if TSF_DEBUG_OUTPUT
+ /// <summary>コールスタックの階層にあわせてインデントしてデバッグ表示するクラス。</summary>
+ public class DebugOut : IDisposable
+ {
+ public DebugOut(string i_string, params object[] i_params)
+ {
+ _text = string.Format(i_string, i_params);
+
+ s_callCount++;
+ Debug.WriteLine("");
+ Debug.WriteLine(string.Format("{0, 4} : ↓↓↓ ", s_callCount) + _text);
+ }
+
+ public static void Print(string i_string, params object[] i_params)
+ {
+ Debug.WriteLine(i_string, i_params);
+ }
+
+ public void Dispose()
+ {
+ s_callCount++;
+ Debug.WriteLine(string.Format("{0, 4} : ↑↑↑ ", s_callCount) + _text);
+ }
+
+ public static string GetCaller([CallerMemberName] string caller="")
+ {
+ return caller;
+ }
+
+ string _text;
+ static int s_callCount = 0;
+ }
+#endif
+
+ //=============================================================================================
+
+
+ public struct TextDisplayAttribute
+ {
+ public int startIndex;
+ public int endIndex;
+ public TF_DISPLAYATTRIBUTE attribute;
+ }
+
+ //========================================================================================
+
+
+ /// <summary>Dispose() で TextStore のロック解除を行うクラス。</summary>
+ public class Unlocker : IDisposable
+ {
+ /// <summary>コンストラクタ</summary>
+ public Unlocker(TextStoreBase io_textStore)
+ {
+ _textStore = io_textStore;
+ }
+ /// <summary>ロックが成功したかどうか調べる。</summary>
+ public bool IsLocked
+ {
+ get { return _textStore != null; }
+ }
+ /// <summary>アンロックを行う。</summary>
+ void IDisposable.Dispose()
+ {
+ if (_textStore != null)
+ {
+ _textStore.UnlockDocument();
+ _textStore = null;
+ }
+ }
+
+ /// <summary>アンロックを行うテキストストア</summary>
+ TextStoreBase _textStore;
+ }
+
+ public abstract class TextStoreBase
+ {
+ public delegate bool IsReadOnlyHandler();
+ public event IsReadOnlyHandler IsReadOnly;
+
+ public delegate bool IsLoadingHandler();
+ public event IsLoadingHandler IsLoading;
+
+ public delegate int GetStringLengthHandler();
+ public event GetStringLengthHandler GetStringLength;
+
+ public delegate void GetSelectionIndexHandler(out int o_start, out int o_end);
+ public event GetSelectionIndexHandler GetSelectionIndex;
+
+ public delegate void SetSelectionIndexHandler(int i_start, int i_end);
+ public event SetSelectionIndexHandler SetSelectionIndex;
+
+ public delegate string GetStringHandler(int start, int length);
+ public event GetStringHandler GetString;
+
+ public delegate void InsertAtSelectionHandler(string i_value,ref int o_startIndex,ref int o_endIndex);
+ public event InsertAtSelectionHandler InsertAtSelection;
+
+ public delegate void GetScreenExtentHandler(
+ out POINT o_pointTopLeft,
+ out POINT o_pointBottomRight
+ );
+ public event GetScreenExtentHandler GetScreenExtent;
+
+ public delegate void GetStringExtentHandler(
+ int i_startIndex,
+ int i_endIndex,
+ out POINT o_pointTopLeft,
+ out POINT o_pointBottomRight
+ );
+ public event GetStringExtentHandler GetStringExtent;
+
+ public delegate bool CompositionStartedHandler();
+ public event CompositionStartedHandler CompositionStarted;
+
+ public delegate void CompostionUpdateHandler(int start, int end);
+ public event CompostionUpdateHandler CompositionUpdated;
+
+ public delegate void CompositionEndedHandler();
+ public event CompositionEndedHandler CompositionEnded;
+
+ #region "生成と破棄"
+ public TextStoreBase()
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ try
+ {
+ // スレッドマネージャ-の生成
+ CreateThreadMgr();
+ // カテゴリマネージャーの生成
+ CreateCategoryMgr();
+ // 表示属性マネージャーの生成
+ CreateDisplayAttributeMgr();
+
+ // ドキュメントマネージャーの生成
+ _threadMgr.CreateDocumentMgr(out _documentMgr);
+
+ // スレッドマネージャのアクティブ化
+ int clientId = 0;
+ _threadMgr.Activate(out clientId);
+
+ // コンテキストの生成
+ _documentMgr.CreateContext(clientId, 0, this, out _context, out _editCookie);
+
+ // コンテキストの push
+ _documentMgr.Push(_context);
+
+ // ファンクションプロバイダーを取得する。
+ Guid guid = TfDeclarations.GUID_SYSTEM_FUNCTIONPROVIDER;
+ _threadMgr.GetFunctionProvider(ref guid, out _functionProvider);
+
+ // ITfReconversion オブジェクトを取得する。
+ var guidNull = new Guid();
+ var guidReconversion = new Guid("4cea93c0-0a58-11d3-8df0-00105a2799b5"); //ITfFnReconversionの定義から
+ object reconversion = null;
+ _functionProvider.GetFunction(
+ ref guidNull,
+ ref guidReconversion,
+ out reconversion
+ );
+ _reconversion = reconversion as ITfFnReconversion;
+
+ // MODEBIAS の初期化
+ uint guidAtom = 0;
+ Guid guidModebiasNone = TfDeclarations.GUID_MODEBIAS_NONE;
+ _categoryMgr.RegisterGUID(ref guidModebiasNone, out guidAtom);
+ _attributeInfo[0].attrID = TfDeclarations.GUID_PROP_MODEBIAS;
+ _attributeInfo[0].flags = AttributeInfoFlags.None;
+ _attributeInfo[0].currentValue.vt = (short)VarEnum.VT_EMPTY;
+ _attributeInfo[0].defaultValue.vt = (short)VarEnum.VT_I4;
+ _attributeInfo[0].defaultValue.data1 = (IntPtr)guidAtom;
+ }
+ catch (Exception exception)
+ {
+ Debug.WriteLine(exception.Message);
+ Dispose(false);
+ }
+ }
+ }
+
+ /// <summary>
+ /// オブジェクトの破棄を行う。このメソッドは必ず呼び出す必要があります
+ /// </summary>
+ /// <param name="flag"></param>
+ protected void Dispose(bool flag)
+ {
+ if (flag)
+ return;
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ ReleaseComObject("_reconversion", ref _reconversion);
+ ReleaseComObject("_functionProvider", ref _functionProvider);
+ ReleaseComObject("_context", ref _context);
+ DestroyDocumentMgr();
+ DestroyDisplayAttributeMgr();
+ DestroyCategoryMgr();
+ DestroyThreadMgr();
+
+ GC.SuppressFinalize(this);
+ }
+ }
+
+ /// <summary>
+ /// スレッドマネージャの生成。このメソッドの実装は必須です
+ /// </summary>
+ ///
+ /// <exception cref="COMException">
+ /// スレッドマネージャーの生成に失敗した場合。
+ /// </exception>
+ protected virtual void CreateThreadMgr()
+ {
+ throw new NotImplementedException();
+ }
+
+ /// <summary>
+ /// スレッドマネージャーの解放。
+ /// </summary>
+ void DestroyThreadMgr()
+ {
+ if (_threadMgr != null)
+ {
+ try { _threadMgr.Deactivate(); }
+ catch (Exception) { }
+
+ ReleaseComObject("_threadMgr", ref _threadMgr);
+ }
+ }
+
+ /// <summary>
+ /// カテゴリマネージャーの生成。
+ /// </summary>
+ ///
+ /// <exception cref="COMException">
+ /// カテゴリマネージャーの生成に失敗した場合。
+ /// </exception>
+ void CreateCategoryMgr()
+ {
+ if (_categoryMgr == null)
+ {
+ var clsid = Marshal.GetTypeFromCLSID(TfDeclarations.CLSID_TF_CategoryMgr);
+ _categoryMgr = Activator.CreateInstance(clsid) as ITfCategoryMgr;
+
+ if (_categoryMgr == null)
+ {
+ const string message = "カテゴリマネージャーの生成に失敗しました。";
+ Debug.WriteLine(message);
+ throw new COMException(message, HRESULT.E_NOTIMPL);
+ }
+ }
+ }
+
+ /// <summary>
+ /// カテゴリマネージャーの解放。
+ /// </summary>
+ void DestroyCategoryMgr()
+ {
+ ReleaseComObject("_categoryMgr", ref _categoryMgr);
+ }
+
+ /// <summary>
+ /// 表示属性マネージャーの生成。
+ /// </summary>
+ ///
+ /// <exception cref="COMException">
+ /// 表示属性マネージャーの生成に失敗した場合。
+ /// </exception>
+ void CreateDisplayAttributeMgr()
+ {
+ if (_displayAttributeMgr == null)
+ {
+ var clsid = Marshal.GetTypeFromCLSID(TfDeclarations.CLSID_TF_DisplayAttributeMgr);
+ _displayAttributeMgr = Activator.CreateInstance(clsid) as ITfDisplayAttributeMgr;
+
+ if (_displayAttributeMgr == null)
+ {
+ const string message = "表示属性マネージャーの生成に失敗しました。";
+ Debug.WriteLine(message);
+ throw new COMException(message, HRESULT.E_NOTIMPL);
+ }
+ }
+ }
+
+ /// <summary>
+ /// 表示属性マネージャーの解放。
+ /// </summary>
+ void DestroyDisplayAttributeMgr()
+ {
+ ReleaseComObject("_displayAttributeMgr", ref _displayAttributeMgr);
+ }
+
+ /// <summary>
+ /// ドキュメントマネージャーの解放
+ /// </summary>
+ void DestroyDocumentMgr()
+ {
+ if (_documentMgr != null)
+ {
+ try { _documentMgr.Pop(PopFlags.TF_POPF_ALL); }
+ catch (Exception) { }
+
+ ReleaseComObject("_documentMgr", ref _documentMgr);
+ }
+ }
+
+ /// <summary>
+ /// COM オブジェクトのリリースとデバッグメッセージ出力。
+ /// </summary>
+ protected static void ReleaseComObject<ComObject>(
+ string i_objectName,
+ ref ComObject io_comObject
+ )
+ {
+ if (io_comObject != null)
+ {
+ var refCount = Marshal.ReleaseComObject(io_comObject);
+#if TSF_DEBUG_OUTPUT
+ Debug.WriteLine(
+ "Marshal.ReleaseComObject({0}) returns {1}.",
+ i_objectName,
+ refCount
+ );
+#endif
+
+ io_comObject = default(ComObject);
+ }
+ }
+ #endregion
+
+ #region "コントロール側が状況に応じて呼び出さなければいけない TSF に通知を送るメソッド"
+ /// <summary>
+ /// 選択領域が変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。
+ /// </summary>
+ public void NotifySelectionChanged()
+ {
+ if( _sink != null )
+ {
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print(DebugOut.GetCaller());
+#endif
+ if ((_adviseFlags & AdviseFlags.TS_AS_SEL_CHANGE) != 0)
+ _sink.OnSelectionChange();
+ }
+ }
+
+
+ //=========================================================================================
+
+
+ /// <summary>
+ /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。
+ /// </summary>
+ void NotifyTextChanged(TS_TEXTCHANGE textChange)
+ {
+ if( _sink != null )
+ {
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print(DebugOut.GetCaller());
+#endif
+ if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)
+ _sink.OnTextChange(0, ref textChange);
+ _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);
+ }
+ }
+
+
+ //=========================================================================================
+
+ /// <summary>
+ /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。
+ /// </summary>
+ /// <param name="start">開始位置</param>
+ /// <param name="oldend">更新前の終了位置</param>
+ /// <param name="newend">更新後の終了位置</param>
+ /// <remarks>
+ /// 詳しいことはITextStoreACPSink::OnTextChangeを参照してください
+ /// </remarks>
+ public void NotifyTextChanged(int start,int oldend,int newend)
+ {
+ if (_sink != null)
+ {
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print(DebugOut.GetCaller());
+#endif
+ if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)
+ {
+ var textChange = new TS_TEXTCHANGE();
+ textChange.start = start;
+ textChange.oldEnd = oldend;
+ textChange.newEnd = newend;
+
+ _sink.OnTextChange(0, ref textChange);
+ }
+ _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);
+ }
+ }
+
+ /// <summary>
+ /// テキストが変更されたことをTSFに伝える。各種ハンドラ内からコールしてはいけない。
+ /// </summary>
+ /// <param name="i_oldLength"></param>
+ /// <param name="i_newLength"></param>
+ public void NotifyTextChanged(int i_oldLength, int i_newLength)
+ {
+ if( _sink != null )
+ {
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print(DebugOut.GetCaller());
+#endif
+ if ((_adviseFlags & AdviseFlags.TS_AS_TEXT_CHANGE) != 0)
+ {
+ var textChange = new TS_TEXTCHANGE();
+ textChange.start = 0;
+ textChange.oldEnd = i_oldLength;
+ textChange.newEnd = i_newLength;
+
+ _sink.OnTextChange(0, ref textChange);
+ }
+ _sink.OnLayoutChange(TsLayoutCode.TS_LC_CHANGE, 1);
+ }
+ }
+
+
+
+ //=========================================================================================
+
+
+ /// <summary>コントロールがフォーカスを取得した時に呼び出さなければいけない。</summary>
+ public void SetFocus()
+ {
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print(DebugOut.GetCaller());
+#endif
+ if (_threadMgr != null)
+ _threadMgr.SetFocus(_documentMgr);
+ }
+ #endregion "コントロール側が状況に応じて呼び出さなければいけない TSF に通知を送るメソッド"
+
+ #region ロック関連
+ /// <summary>
+ /// ドキュメントのロックを行う。
+ /// </summary>
+ /// <param name="i_writable">読み書き両用ロックか?false の場合、読み取り専用。</param>
+ /// <returns>Unlocker のインスタンスを返す。失敗した場合 null を返す。</returns>
+ public Unlocker LockDocument(bool i_writable)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}({1})", DebugOut.GetCaller(),
+ i_writable) )
+ #endif
+ {
+ lock(this)
+ {
+ if( this._lockFlags == 0 )
+ {
+ if( i_writable )
+ this._lockFlags = LockFlags.TS_LF_READWRITE;
+ else
+ this._lockFlags = LockFlags.TS_LF_READ;
+
+ #if TSF_DEBUG_OUTPUT
+ Debug.WriteLine("LockDocument is succeeded.");
+ #endif
+
+ return new Unlocker(this);
+ }
+ else
+ {
+ #if TSF_DEBUG_OUTPUT
+ Debug.WriteLine("LockDocument is failed. {0}", _lockFlags);
+ #endif
+
+ return null;
+ }
+ }
+ }
+ }
+
+
+ //========================================================================================
+
+
+ /// <summary>
+ /// ドキュメントのロックを行う。
+ /// </summary>
+ /// <param name="i_writable">読み書き両用ロックか?false の場合、読み取り専用。</param>
+ /// <returns>Unlocker のインスタンスを返す。失敗した場合 null を返す。</returns>
+ public Unlocker LockDocument(LockFlags i_flags)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}({1})", DebugOut.GetCaller(),
+ i_flags) )
+ #endif
+ {
+ lock(this)
+ {
+ if( this._lockFlags == 0 )
+ {
+ this._lockFlags = i_flags;
+
+ #if TSF_DEBUG_OUTPUT
+ Debug.WriteLine("LockDocument is succeeded.");
+ #endif
+
+ return new Unlocker(this);
+ }
+ else
+ {
+ #if TSF_DEBUG_OUTPUT
+ Debug.WriteLine("LockDocument is failed. {0}", _lockFlags);
+ #endif
+
+ return null;
+ }
+ }
+ }
+ }
+
+
+ //========================================================================================
+
+
+ /// <summary>
+ /// ドキュメントのアンロックを行う。
+ /// </summary>
+ public void UnlockDocument()
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+ #endif
+ {
+ lock(this)
+ {
+ _lockFlags = 0;
+ }
+
+ if( _pendingLockUpgrade )
+ {
+ _pendingLockUpgrade = false;
+ int sessionResult;
+ RequestLock(LockFlags.TS_LF_READWRITE, out sessionResult);
+ }
+ }
+ }
+
+
+ //========================================================================================
+
+
+ /// <summary>
+ /// 指定されたフラグでロックしている状態かどうか調べる。
+ /// </summary>
+ /// <param name="i_lockFlags"></param>
+ /// <returns>ロックされている場合は true, されていない場合は false を返す。</returns>
+ public bool IsLocked(LockFlags i_lockFlags)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}({1})",
+ DebugOut.GetCaller(), i_lockFlags) )
+ #endif
+ {
+ #if TSF_DEBUG_OUTPUT
+ Debug.WriteLine(
+ "IsLocked() returns " + ((this._lockFlags & i_lockFlags) == i_lockFlags)
+ );
+ #endif
+ return (this._lockFlags & i_lockFlags) == i_lockFlags;
+ }
+ }
+
+ /// <summary>
+ /// ロックされているか調べる
+ /// </summary>
+ /// <returns>ロックされているなら真、そうでなければ偽を返す</returns>
+ public bool IsLocked()
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}",
+ DebugOut.GetCaller()) )
+#endif
+ {
+ bool retval = this._lockFlags != 0;
+#if TSF_DEBUG_OUTPUT
+ Debug.WriteLine(
+ "IsLocked() returns " + retval
+ );
+#endif
+ return retval;
+ }
+ }
+ #endregion
+
+ #region ITextStroeACP,ITextStoreACP2の共通部分
+
+ /// <summary>
+ /// 文字列を挿入する。
+ /// </summary>
+ public void InsertTextAtSelection(string s)
+ {
+ TS_TEXTCHANGE textChange = new TS_TEXTCHANGE();
+
+ using (var unlocker = LockDocument(true))
+ {
+ if (unlocker != null)
+ {
+ int startIndex, endIndex;
+
+ InsertTextAtSelection(
+ UnmanagedAPI.TSF.TextStore.InsertAtSelectionFlags.TF_IAS_NOQUERY,
+ s.ToCharArray(),
+ s.Length,
+ out startIndex,
+ out endIndex,
+ out textChange
+ );
+ }
+ }
+
+ // シンクの OnSelectionChange() をコール。
+ NotifySelectionChanged();
+ NotifyTextChanged(textChange);
+ }
+
+ public void InsertEmbeddedAtSelection(
+ InsertAtSelectionFlags flags,
+ object obj,
+ out int start,
+ out int end,
+ out TS_TEXTCHANGE change
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void AdviseSink(ref Guid i_riid, object i_unknown, AdviseFlags i_mask)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (i_riid != new Guid("22d44c94-a419-4542-a272-ae26093ececf")) //ITextStoreACPSinkの定義より
+ {
+ throw new COMException(
+ "ITextStoreACPSink 以外のIIDが渡されました。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+
+ // 既存のシンクのマスクのみを更新
+ if (_sink == i_unknown)
+ {
+ _adviseFlags = i_mask;
+ }
+ // シンクを複数登録しようとした
+ else if (_sink != null)
+ {
+ throw new COMException(
+ "既にシンクを登録済みです。",
+ UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_ADVISELIMIT
+ );
+ }
+ else
+ {
+ // 各種値を保存
+ _services = (ITextStoreACPServices)i_unknown;
+ _sink = (ITextStoreACPSink)i_unknown;
+ _adviseFlags = i_mask;
+ }
+ }
+ }
+
+ public void UnadviseSink(object i_unknown)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (_sink == null || _sink != i_unknown)
+ {
+ throw new COMException(
+ "シンクは登録されていません。",
+ UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_NOCONNECTION
+ );
+ }
+
+ _services = null;
+ _sink = null;
+ _adviseFlags = 0;
+ }
+ }
+
+ public void RequestLock(LockFlags i_lockFlags, out int o_sessionResult)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ o_sessionResult = UnmanagedAPI.WinError.HRESULT.E_FAIL;
+
+ if (_sink == null)
+ {
+ throw new COMException(
+ "シンクが登録されていません。",
+ UnmanagedAPI.TSF.TextStore.TsResult.CONNECT_E_NOCONNECTION
+ );
+ }
+
+ if (_lockFlags != 0) // すでにロックされた状態の場合。
+ {
+ if ((i_lockFlags & LockFlags.TS_LF_SYNC) != 0)
+ {
+ o_sessionResult = TsResult.TS_E_SYNCHRONOUS;
+ return;
+ }
+ else
+ if ((_lockFlags & LockFlags.TS_LF_READWRITE) == LockFlags.TS_LF_READ
+ && (i_lockFlags & LockFlags.TS_LF_READWRITE) == LockFlags.TS_LF_READWRITE)
+ {
+ _pendingLockUpgrade = true;
+ o_sessionResult = TsResult.TS_S_ASYNC;
+ return;
+ }
+
+ throw new COMException();
+ }
+
+ using (var unlocker = LockDocument(i_lockFlags))
+ {
+ // ロックに失敗した場合は TS_E_SYNCHRONOUS をセットして S_OK を返す。
+ if (unlocker == null)
+ {
+ o_sessionResult = TsResult.TS_E_SYNCHRONOUS;
+ }
+ // ロックに成功した場合は OnLockGranted() を呼び出す。
+ else
+ {
+ try
+ {
+ o_sessionResult = _sink.OnLockGranted(i_lockFlags);
+ }
+ catch (COMException comException)
+ {
+ Debug.WriteLine("OnLockGranted() 呼び出し中に例外が発生。");
+ Debug.WriteLine(" " + comException.Message);
+ o_sessionResult = comException.HResult;
+ }
+ catch (Exception exception)
+ {
+ Debug.WriteLine("OnLockGranted() 呼び出し中に例外が発生。");
+ Debug.WriteLine(" " + exception.Message);
+ }
+ }
+ }
+ }
+ }
+
+ public void GetStatus(out TS_STATUS o_documentStatus)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (IsReadOnly != null && IsReadOnly())
+ o_documentStatus.dynamicFlags = DynamicStatusFlags.TS_SD_READONLY;
+ if (IsLoading != null && IsLoading())
+ o_documentStatus.dynamicFlags = DynamicStatusFlags.TS_SD_LOADING;
+ else
+ o_documentStatus.dynamicFlags = 0;
+ o_documentStatus.staticFlags = StaticStatusFlags.TS_SS_REGIONS;
+ }
+ }
+
+ public void QueryInsert(
+ int i_startIndex,
+ int i_endIndex,
+ int i_length,
+ out int o_startIndex,
+ out int o_endIndex
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetStringLength == null)
+ throw new NotImplementedException();
+
+ int documentLength = GetStringLength();
+
+ if (i_startIndex < 0
+ || i_startIndex > i_endIndex
+ || i_endIndex > documentLength)
+ {
+ throw new COMException(
+ "インデックスが無効です。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+ o_startIndex = i_startIndex;
+ o_endIndex = i_endIndex;
+
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("o_startIndex:{0} o_endIndex:{1}", i_startIndex, i_endIndex);
+#endif
+ }
+ }
+
+ public void GetSelection(
+ int i_index,
+ int i_selectionBufferLength,
+ TS_SELECTION_ACP[] o_selections,
+ out int o_fetchedLength
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetSelectionIndex == null)
+ throw new NotImplementedException();
+
+ o_fetchedLength = 0;
+
+ if (IsLocked(LockFlags.TS_LF_READ) == false)
+ {
+ throw new COMException(
+ "読取用ロックがされていません。",
+ TsResult.TS_E_NOLOCK
+ );
+ }
+
+ // -1 は TF_DEFAULT_SELECTION。選択は1つだけしかサポートしないので、
+ // TF_DEFAULT_SELECTION でもなく、0 を超える数値が指定された場合はエラー。
+ if (i_index != -1 && i_index > 0)
+ {
+ throw new COMException(
+ "選択は1つだけしかサポートしていません。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+
+ if (i_selectionBufferLength > 0)
+ {
+ int start = 0, end = 0;
+ GetSelectionIndex(out start, out end);
+ if (start <= end)
+ {
+ o_selections[0].start = start;
+ o_selections[0].end = end;
+ o_selections[0].style.ase = TsActiveSelEnd.TS_AE_END;
+ o_selections[0].style.interimChar = false;
+ }
+ else
+ {
+ o_selections[0].start = end;
+ o_selections[0].end = start;
+ o_selections[0].style.ase = TsActiveSelEnd.TS_AE_START;
+ o_selections[0].style.interimChar = false;
+ }
+
+ o_fetchedLength = 1;
+
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("sel start:{0} end:{1}", start, end);
+#endif
+ }
+ }
+ }
+
+ public void SetSelection(int i_count, TS_SELECTION_ACP[] i_selections)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}({1}, {2})",
+ DebugOut.GetCaller(),
+ i_selections[0].start,
+ i_selections[0].end))
+#endif
+ {
+ if (SetSelectionIndex == null)
+ throw new NotImplementedException();
+
+ if (i_count != 1)
+ {
+ throw new COMException(
+ "選択は1つだけしかサポートしていません。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+
+ if (IsLocked(LockFlags.TS_LF_READWRITE) == false)
+ {
+ throw new COMException(
+ "ロックされていません。",
+ TsResult.TS_E_NOLOCK
+ );
+ }
+
+ SetSelectionIndex(i_selections[0].start, i_selections[0].end);
+
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("set selection startIndex:{0} endIndex:{1}", i_selections[0].start, i_selections[0].end);
+#endif
+ }
+ }
+
+ public void GetText(
+ int i_startIndex,
+ int i_endIndex,
+ char[] o_plainText,
+ int i_plainTextLength,
+ out int o_plainTextLength,
+ TS_RUNINFO[] o_runInfos,
+ int i_runInfoLength,
+ out int o_runInfoLength,
+ out int o_nextUnreadCharPos
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetStringLength == null || GetString == null)
+ throw new NotImplementedException();
+
+ if (IsLocked(LockFlags.TS_LF_READ) == false)
+ {
+ throw new COMException(
+ "読取用ロックがされていません。",
+ TsResult.TS_E_NOLOCK
+ );
+ }
+
+ if ((i_endIndex != -1 && i_startIndex > i_endIndex)
+ || i_startIndex < 0 || i_startIndex > GetStringLength()
+ || i_endIndex > GetStringLength())
+ {
+ throw new COMException(
+ "インデックスが無効です。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+
+ var textLength = 0;
+ var copyLength = 0;
+
+ if (i_endIndex == -1)
+ textLength = GetStringLength() - i_startIndex;
+ else
+ textLength = i_endIndex - i_startIndex;
+ copyLength = Math.Min(i_plainTextLength, textLength);
+
+ // 文字列を格納。
+ var text = GetString(i_startIndex, copyLength);
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("got text:{0} from {1} length {2}", text,i_startIndex,copyLength);
+#endif
+ for (int i = 0; i < copyLength; i++)
+ {
+ o_plainText[i] = text[i];
+ }
+
+ // 文字数を格納。
+ o_plainTextLength = copyLength;
+ // RUNINFO を格納。
+ if (i_runInfoLength > 0)
+ {
+ o_runInfos[0].type = TsRunType.TS_RT_PLAIN;
+ o_runInfos[0].length = copyLength;
+ }
+ o_runInfoLength = 1;
+
+ // 次の文字の位置を格納。
+ o_nextUnreadCharPos = i_startIndex + copyLength;
+ }
+ }
+
+ public void SetText(
+ SetTextFlags i_flags,
+ int i_startIndex,
+ int i_endIndex,
+ char[] i_text,
+ int i_length,
+ out TS_TEXTCHANGE o_textChange
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}({1}, {2})",
+ DebugOut.GetCaller(),
+ i_startIndex, i_endIndex) )
+#endif
+ {
+ var selections = new TS_SELECTION_ACP[]
+ {
+ new TS_SELECTION_ACP
+ {
+ start = i_startIndex,
+ end = i_endIndex,
+ style = new TS_SELECTIONSTYLE
+ {
+ ase = TsActiveSelEnd.TS_AE_END,
+ interimChar = false
+ }
+ }
+ };
+
+ int startIndex = 0, endIndex = 0;
+ SetSelection(1, selections);
+ InsertTextAtSelection(
+ InsertAtSelectionFlags.TF_IAS_NOQUERY,
+ i_text,
+ i_length,
+ out startIndex,
+ out endIndex,
+ out o_textChange
+ );
+ }
+ }
+
+ public void GetFormattedText(int i_start, int i_end, out object o_obj)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void GetEmbedded(
+ int i_position,
+ ref Guid i_guidService,
+ ref Guid i_riid,
+ out object o_obj
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void QueryInsertEmbedded(
+ ref Guid i_guidService,
+ int i_formatEtc,
+ out bool o_insertable
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void InsertEmbedded(
+ InsertEmbeddedFlags i_flags,
+ int i_start,
+ int i_end,
+ object i_obj,
+ out TS_TEXTCHANGE o_textChange
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void InsertTextAtSelection(
+ InsertAtSelectionFlags i_flags,
+ char[] i_text,
+ int i_length,
+ out int o_startIndex,
+ out int o_endIndex,
+ out TS_TEXTCHANGE o_textChange
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetSelectionIndex == null || InsertAtSelection == null)
+ throw new NotImplementedException();
+
+ int selectionStart = 0, selectionEnd = 0;
+ GetSelectionIndex(out selectionStart, out selectionEnd);
+
+ // 問い合わせのみで実際には操作を行わない
+ if ((i_flags & InsertAtSelectionFlags.TF_IAS_QUERYONLY) != 0)
+ {
+ o_startIndex = Math.Min(selectionStart, selectionEnd);
+ o_endIndex = Math.Max(selectionStart, selectionEnd);//o_startIndex + i_length;
+
+ o_textChange.start = o_startIndex;
+ o_textChange.oldEnd = o_endIndex;
+ o_textChange.newEnd = o_startIndex + i_length;
+ }
+ else
+ {
+ var start = Math.Min(selectionStart, selectionEnd);
+ var end = Math.Max(selectionStart, selectionEnd);
+
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("start: {0}, end: {1}, text: {2}", start, end, new string(i_text));
+#endif
+
+ o_startIndex = start;
+ o_endIndex = start + i_length;
+
+ InsertAtSelection(new string(i_text), ref o_startIndex, ref o_endIndex);
+
+ o_textChange.start = start;
+ o_textChange.oldEnd = end;
+ o_textChange.newEnd = o_endIndex;
+ // InsertAtSelection() 内でカーソル位置を更新しているため、ここでは不要。
+ // 改行した時に位置が狂う。
+ // SetSelectionIndex(start, start + i_length);
+ }
+ }
+ }
+
+ public void GetEndACP(out int o_length)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetStringLength == null)
+ throw new NotImplementedException();
+
+ if (IsLocked(LockFlags.TS_LF_READ) == false)
+ {
+ throw new COMException(
+ "読取用ロックがされていません。",
+ TsResult.TS_E_NOLOCK
+ );
+ }
+
+ o_length = GetStringLength();
+ }
+ }
+
+ public void GetActiveView(out int o_viewCookie)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ o_viewCookie = 1;
+ }
+ }
+
+ public void GetACPFromPoint(
+ int i_viewCookie,
+ ref POINT i_point,
+ GetPositionFromPointFlags i_flags,
+ out int o_index
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using (var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void GetTextExt(
+ int i_viewCookie,
+ int i_startIndex,
+ int i_endIndex,
+ out RECT o_rect,
+ out bool o_isClipped
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetStringLength == null)
+ throw new NotImplementedException();
+
+ // 読取用ロックの確認。
+ if (IsLocked(LockFlags.TS_LF_READ) == false)
+ {
+ throw new COMException(
+ "読取用ロックがされていません。",
+ TsResult.TS_E_NOLOCK
+ );
+ }
+
+ if ((i_endIndex != -1 && i_startIndex > i_endIndex)
+ || i_startIndex < 0 || i_startIndex > GetStringLength()
+ || i_endIndex > GetStringLength())
+ {
+ throw new COMException(
+ "インデックスが無効です。",
+ UnmanagedAPI.WinError.HRESULT.E_INVALIDARG
+ );
+ }
+
+ if (i_endIndex == -1)
+ i_endIndex = GetStringLength();
+
+ var pointTopLeft = new POINT();
+ var pointBotttomRight = new POINT();
+ GetStringExtent(i_startIndex, i_endIndex, out pointTopLeft, out pointBotttomRight);
+
+ o_rect.left = (int)(pointTopLeft.x);
+ o_rect.top = (int)(pointTopLeft.y);
+ o_rect.bottom = (int)(pointBotttomRight.y);
+ o_rect.right = (int)(pointBotttomRight.x);
+ o_isClipped = false;
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("rect left:{0} top:{1} bottom:{2} right:{3}", o_rect.left, o_rect.top, o_rect.bottom, o_rect.right);
+#endif
+ }
+ }
+
+ public void GetScreenExt(int i_viewCookie, out RECT o_rect)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ if (GetScreenExtent == null)
+ throw new NotImplementedException();
+
+ POINT pointTopLeft, pointBottomRight;
+
+ GetScreenExtent(out pointTopLeft, out pointBottomRight);
+
+ o_rect.left = (int)(pointTopLeft.x);
+ o_rect.top = (int)(pointTopLeft.y);
+ o_rect.bottom = (int)(pointBottomRight.y);
+ o_rect.right = (int)(pointBottomRight.x);
+#if TSF_DEBUG_OUTPUT
+ DebugOut.Print("rect left:{0} top:{1} bottom:{2} right:{3}", o_rect.left, o_rect.top, o_rect.bottom, o_rect.right);
+#endif
+ }
+ }
+
+ public void RequestSupportedAttrs(
+ AttributeFlags i_flags,
+ int i_length,
+ Guid[] i_filterAttributes
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ for (int i = 0; i < i_length; i++)
+ {
+ if (i_filterAttributes[i].Equals(_attributeInfo[0].attrID))
+ {
+ _attributeInfo[0].flags = AttributeInfoFlags.Requested;
+ if ((i_flags & AttributeFlags.TS_ATTR_FIND_WANT_VALUE) != 0)
+ {
+ _attributeInfo[0].flags |= AttributeInfoFlags.Default;
+ }
+ else
+ {
+ _attributeInfo[0].currentValue = _attributeInfo[0].defaultValue;
+ }
+ }
+ }
+ }
+ }
+
+ public void RequestAttrsAtPosition(
+ int i_position,
+ int i_length,
+ Guid[] i_filterAttributes,
+ AttributeFlags i_flags
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ for (int i = 0; i < i_length; i++)
+ {
+ if (i_filterAttributes[i].Equals(_attributeInfo[0].attrID))
+ {
+ _attributeInfo[0].flags = AttributeInfoFlags.Requested;
+ if ((i_flags & AttributeFlags.TS_ATTR_FIND_WANT_VALUE) != 0)
+ {
+ _attributeInfo[0].flags |= AttributeInfoFlags.Default;
+ }
+ else
+ {
+ _attributeInfo[0].currentValue = _attributeInfo[0].defaultValue;
+ }
+ }
+ }
+ }
+ }
+
+ public void RequestAttrsTransitioningAtPosition(
+ int i_position,
+ int i_length,
+ Guid[] i_filterAttributes,
+ AttributeFlags i_flags
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))
+#endif
+ {
+ // 何もしない。
+ }
+ }
+
+ public void FindNextAttrTransition(
+ int i_start,
+ int i_halt,
+ int i_length,
+ Guid[] i_filterAttributes,
+ AttributeFlags i_flags,
+ out int o_nextIndex,
+ out bool o_found,
+ out int o_foundOffset
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ throw new NotImplementedException();
+ }
+ }
+
+ public void RetrieveRequestedAttrs(
+ int i_length,
+ TS_ATTRVAL[] o_attributeVals,
+ out int o_fetchedLength
+ )
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+#endif
+ {
+ o_fetchedLength = 0;
+ for (int i = 0; i < _attributeInfo.Length && o_fetchedLength < i_length; i++)
+ {
+ if ((_attributeInfo[i].flags & AttributeInfoFlags.Requested) != 0)
+ {
+ o_attributeVals[o_fetchedLength].overlappedId = 0;
+ o_attributeVals[o_fetchedLength].attributeId = _attributeInfo[i].attrID;
+ o_attributeVals[o_fetchedLength].val = _attributeInfo[i].currentValue;
+
+ if ((_attributeInfo[i].flags & AttributeInfoFlags.Default) != 0)
+ _attributeInfo[i].currentValue = _attributeInfo[i].defaultValue;
+
+ o_fetchedLength++;
+ _attributeInfo[i].flags = AttributeInfoFlags.None;
+ }
+ }
+ }
+ }
+ #endregion
+
+ #region "ITfContextOwnerCompositionSink インターフェイスの実装"
+ /// <summary>コンポジション入力が開始された時の処理。</summary>
+ public void OnStartComposition(ITfCompositionView view, out bool ok)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+ #endif
+ {
+ if( CompositionStarted != null )
+ ok = CompositionStarted();
+ else
+ ok = true;
+ }
+ }
+
+
+ //========================================================================================
+
+
+ /// <summary>コンポジションが変更された時の処理。</summary>
+ public void OnUpdateComposition(ITfCompositionView view, ITfRange rangeNew)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+ #endif
+ {
+ if (rangeNew == null)
+ return;
+
+ int start, count;
+
+ ITfRangeACP rangeAcp = (ITfRangeACP)rangeNew;
+
+ rangeAcp.GetExtent(out start, out count);
+
+ if (this.CompositionUpdated != null)
+ this.CompositionUpdated(start, start + count);
+ }
+ }
+
+
+ //========================================================================================
+
+
+ /// <summary>コンポジション入力が終了した時の処理。</summary>
+ public void OnEndComposition(ITfCompositionView view)
+ {
+ #if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()) )
+ #endif
+ {
+ if( CompositionEnded != null )
+ CompositionEnded();
+ }
+ }
+ #endregion "ITfContextOwnerCompositionSink インターフェイスの実装"
+
+ #region 表示属性
+ /// <summary>表示属性の取得</summary>
+ public IEnumerable<TextDisplayAttribute> EnumAttributes(int start, int end)
+ {
+ ITfRangeACP allRange;
+ _services.CreateRange(start, end, out allRange);
+
+ foreach (TextDisplayAttribute attr in this.EnumAttributes((ITfRange)allRange))
+ yield return attr;
+
+ ReleaseComObject("allRange", ref allRange);
+ }
+
+ IEnumerable<TextDisplayAttribute> EnumAttributes(ITfRange range)
+ {
+#if TSF_DEBUG_OUTPUT
+ using(var dbgout = new DebugOut("{0}()", DebugOut.GetCaller()))
+#endif
+ {
+ ITfProperty property = null; // プロパティインターフェイス
+ IEnumTfRanges enumRanges; // 範囲の列挙子
+ Guid guidPropAttribute = TfDeclarations.GUID_PROP_ATTRIBUTE;
+
+ if (_context == null || _services == null)
+ yield break;
+
+ // GUID_PROP_ATTRIBUTE プロパティを取得。
+ _context.GetProperty(ref guidPropAttribute, out property);
+ if (property == null)
+ yield break;
+
+ // 全範囲の中で表示属性プロパティをもつ範囲を列挙する。
+ property.EnumRanges((int)_editCookie, out enumRanges, range);
+
+ ITfRange[] ranges = new ITfRange[1];
+ int fetchedLength = 0;
+ while (HRESULT.Succeeded(enumRanges.Next(1, ranges, out fetchedLength))
+ && fetchedLength > 0)
+ {
+ // ItfRange から ItfRangeACP を取得。
+ ITfRangeACP rangeACP = ranges[0] as ITfRangeACP;
+ if (rangeACP == null)
+ continue; // 普通はあり得ない
+
+ // 範囲の開始位置と文字数を取得。
+ int start, count;
+ rangeACP.GetExtent(out start, out count);
+
+ // VARIANT 値としてプロパティ値を取得。VT_I4 の GUID ATOM がとれる。
+ VARIANT value = new VARIANT();
+ property.GetValue((int)_editCookie, ranges[0], ref value);
+ if (value.vt == (short)VarEnum.VT_I4)
+ {
+ Guid guid, clsid;
+ ITfDisplayAttributeInfo info;
+ TF_DISPLAYATTRIBUTE attribute;
+
+ // GUID ATOM から GUID を取得する。
+ _categoryMgr.GetGUID((int)value.data1, out guid);
+ // その GUID から IDisplayAttributeInfo インターフェイスを取得。
+ _displayAttributeMgr.GetDisplayAttributeInfo(
+ ref guid,
+ out info,
+ out clsid
+ );
+ // さらに IDisplayAttributeInfo インターフェイスから表示属性を取得する。
+ info.GetAttributeInfo(out attribute);
+ ReleaseComObject("info", ref info);
+
+ yield return new TextDisplayAttribute
+ {
+ startIndex = start,
+ endIndex = start + count,
+ attribute = attribute
+ };
+
+#if TSF_DEBUG_OUTPUT_DISPLAY_ATTR
+ Debug.WriteLine(
+ "*******:::: DisplayAttribute: {0} ~ {1} :::::: *********",
+ start, start + count
+ );
+ Debug.WriteLine(attribute.bAttr);
+ Debug.WriteLine(
+ "LineColorType: {0}, {1}",
+ attribute.crLine.type, attribute.crLine.indexOrColorRef
+ );
+ Debug.WriteLine(
+ "TextColorType: {0}, {1}",
+ attribute.crText.type, attribute.crText.indexOrColorRef
+ );
+ Debug.WriteLine(
+ "BackColorType: {0}, {1}",
+ attribute.crBk.type, attribute.crBk.indexOrColorRef
+ );
+ Debug.WriteLine(
+ "Bold, Style : {0}, {1}",
+ attribute.fBoldLine, attribute.lsStyle
+ );
+#endif
+ }
+
+ ReleaseComObject("rangeACP", ref rangeACP);
+ }
+
+ ReleaseComObject("ranges[0]", ref ranges[0]);
+ ReleaseComObject("enumRanges", ref enumRanges);
+ ReleaseComObject("property", ref property);
+ }
+ }
+ #endregion
+
+#if METRO
+ protected ITfThreadMgr2 _threadMgr;
+#else
+ protected ITfThreadMgr _threadMgr;
+#endif
+ protected ITfDocumentMgr _documentMgr;
+ protected ITfFunctionProvider _functionProvider;
+ protected ITfFnReconversion _reconversion;
+ protected ITfContext _context;
+ protected ITfCategoryMgr _categoryMgr;
+ protected ITfDisplayAttributeMgr _displayAttributeMgr;
+ protected ITextStoreACPSink _sink;
+ protected uint _editCookie = 0;
+ protected ITextStoreACPServices _services;
+ protected AdviseFlags _adviseFlags = 0;
+ protected LockFlags _lockFlags = 0;
+ protected bool _pendingLockUpgrade = false;
+
+ /// <summary>
+ /// AttributeInfo で使用されるフラグ。各属性の状態を示す。
+ /// </summary>
+ protected enum AttributeInfoFlags
+ {
+ /// <summary>何もない。</summary>
+ None = 0,
+ /// <summary>デフォルト値の要求。</summary>
+ Default = 1 << 0,
+ /// <summary>要求された。</summary>
+ Requested = 1 << 1
+ }
+
+ protected struct AttributeInfo
+ {
+ public Guid attrID;
+ public AttributeInfoFlags flags;
+ public VARIANT currentValue;
+ public VARIANT defaultValue;
+ }
+
+ protected AttributeInfo[] _attributeInfo = new AttributeInfo[1];
+
+ }
+}