// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef UI_BASE_IME_WIN_TSF_TEXT_STORE_H_
#define UI_BASE_IME_WIN_TSF_TEXT_STORE_H_
#include <msctf.h>
#include <wrl/client.h>
#include <deque>
#include <string>
#include "base/component_export.h"
#include "base/memory/raw_ptr.h"
#include "ui/base/ime/ime_key_event_dispatcher.h"
#include "ui/base/ime/ime_text_span.h"
#include "ui/events/event_utils.h"
#include "ui/gfx/range/range.h"
namespace ui {
class TextInputClient;
// d5138268-a1bf-4308-bcbf-2e739398e234
// Bootstrap the definition of the GUID for the URL property which
// will be defined in the windows SDK.
// https://docs.microsoft.com/en-us/windows/win32/tsf/predefined-properties
const GUID GUID_PROP_URL = {0xd5138268,
0xa1bf,
0x4308,
{0xbc, 0xbf, 0x2e, 0x73, 0x93, 0x98, 0xe2, 0x34}};
// TSFTextStore is used to interact with the input method via TSF manager.
// TSFTextStore have a string buffer which is manipulated by TSF manager through
// ITextStoreACP interface methods such as SetText().
// When the input method updates the composition, TSFTextStore calls
// TextInputClient::SetCompositionText(). And when the input method finishes the
// composition, TSFTextStore calls TextInputClient::InsertText().
//
// How TSFTextStore works:
// - Assume the document is empty and in focus.
// - The user enters "a".
// - The input method set composition as "a".
// - TSF manager calls TSFTextStore::RequestLock().
// - TSFTextStore callbacks ITextStoreACPSink::OnLockGranted().
// - In OnLockGranted(), TSF manager calls
// - TSFTextStore::OnStartComposition()
// - TSFTextStore::SetText()
// The pending string buffer is set as "a".
// The document whole buffer is set as "a".
// - TSFTextStore::OnUpdateComposition()
// - TSFTextStore::OnEndEdit()
// TSFTextStore can get the composition information such as underlines.
// - TSFTextStore calls TextInputClient::SetCompositionText().
// "a" is shown with an underline as composition string.
// - The user enters 'b'.
// - The input method set composition as "ab".
// - TSF manager calls TSFTextStore::RequestLock().
// - TSFTextStore callbacks ITextStoreACPSink::OnLockGranted().
// - In OnLockGranted(), TSF manager calls
// - TSFTextStore::SetText()
// The pending string buffer is set as "b".
// The document whole buffer is changed to "ab".
// - TSFTextStore::OnUpdateComposition()
// - TSFTextStore::OnEndEdit()
// - TSFTextStore calls TextInputClient::SetCompositionText().
// "ab" is shown with an underline as composition string.
// - The user enters <space>.
// - The input method set composition as "aB".
// - TSF manager calls TSFTextStore::RequestLock().
// - TSFTextStore callbacks ITextStoreACPSink::OnLockGranted().
// - In OnLockGranted(), TSF manager calls
// - TSFTextStore::SetText()
// The pending string buffer is set as "B".
// The document whole buffer is changed to "aB".
// - TSFTextStore::OnUpdateComposition()
// - TSFTextStore::OnEndEdit()
// - TSFTextStore calls TextInputClient::SetCompositionText().
// "aB" is shown with an underline as composition string.
// - The user enters <enter>.
// - The input method commits "aB".
// - TSF manager calls TSFTextStore::RequestLock().
// - TSFTextStore callbacks ITextStoreACPSink::OnLockGranted().
// - In OnLockGranted(), TSF manager calls
// - TSFTextStore::OnEndComposition()
// - TSFTextStore::OnEndEdit()
// TSFTextStore knows "aB" is committed.
// - TSFTextStore calls TextInputClient::InsertText().
// "aB" is shown as committed string.
// - TSFTextStore clears the pending string buffer.
// - TSFTextStore verified if the document whole buffer is the same as the
// buffer returned from TextInputClient. If the buffer is different, then
// call OnSelectionChange(), OnLayoutChange() and
// OnTextChange() of ITextStoreACPSink to let TSF manager know that the
// string buffer has been changed other than IME.
//
// About the locking scheme:
// When TSF manager manipulates the string buffer it calls RequestLock() to get
// the lock of the document. If TSFTextStore can grant the lock request, it
// callbacks ITextStoreACPSink::OnLockGranted().
// RequestLock() is called from only one thread, but called recursively in
// OnLockGranted() or OnSelectionChange() or OnLayoutChange() or OnTextChange().
// If the document is locked and the lock request is asynchronous, TSFTextStore
// queues the request. The queued requests will be handled after the current
// lock is removed.
// More information about document locks can be found here:
// http://msdn.microsoft.com/en-us/library/ms538064
//
// More information about TSF can be found here:
// http://msdn.microsoft.com/en-us/library/ms629032
class COMPONENT_EXPORT(UI_BASE_IME_WIN) TSFTextStore
: public ITextStoreACP,
public ITfContextOwnerCompositionSink,
public ITfLanguageProfileNotifySink,
public ITfKeyTraceEventSink,
public ITfTextEditSink {
public:
TSFTextStore();
TSFTextStore(const TSFTextStore&) = delete;
TSFTextStore& operator=(const TSFTextStore&) = delete;
virtual ~TSFTextStore();
HRESULT Initialize();
// ITextStoreACP:
IFACEMETHODIMP_(ULONG) AddRef() override;
IFACEMETHODIMP_(ULONG) Release() override;
IFACEMETHODIMP QueryInterface(REFIID iid, void** ppv) override;
IFACEMETHODIMP AdviseSink(REFIID iid, IUnknown* unknown, DWORD mask) override;
IFACEMETHODIMP FindNextAttrTransition(LONG acp_start,
LONG acp_halt,
ULONG num_filter_attributes,
const TS_ATTRID* filter_attributes,
DWORD flags,
LONG* acp_next,
BOOL* found,
LONG* found_offset) override;
IFACEMETHODIMP GetACPFromPoint(TsViewCookie view_cookie,
const POINT* point,
DWORD flags,
LONG* acp) override;
IFACEMETHODIMP GetActiveView(TsViewCookie* view_cookie) override;
IFACEMETHODIMP GetEmbedded(LONG acp_pos,
REFGUID service,
REFIID iid,
IUnknown** unknown) override;
IFACEMETHODIMP GetEndACP(LONG* acp) override;
IFACEMETHODIMP GetFormattedText(LONG acp_start,
LONG acp_end,
IDataObject** data_object) override;
IFACEMETHODIMP GetScreenExt(TsViewCookie view_cookie, RECT* rect) override;
IFACEMETHODIMP GetSelection(ULONG selection_index,
ULONG selection_buffer_size,
TS_SELECTION_ACP* selection_buffer,
ULONG* fetched_count) override;
IFACEMETHODIMP GetStatus(TS_STATUS* pdcs) override;
IFACEMETHODIMP GetText(LONG acp_start,
LONG acp_end,
wchar_t* text_buffer,
ULONG text_buffer_size,
ULONG* text_buffer_copied,
TS_RUNINFO* run_info_buffer,
ULONG run_info_buffer_size,
ULONG* run_info_buffer_copied,
LONG* next_acp) override;
IFACEMETHODIMP GetTextExt(TsViewCookie view_cookie,
LONG acp_start,
LONG acp_end,
RECT* rect,
BOOL* clipped) override;
IFACEMETHODIMP GetWnd(TsViewCookie view_cookie, HWND* window_handle) override;
IFACEMETHODIMP InsertEmbedded(DWORD flags,
LONG acp_start,
LONG acp_end,
IDataObject* data_object,
TS_TEXTCHANGE* change) override;
IFACEMETHODIMP InsertEmbeddedAtSelection(DWORD flags,
IDataObject* data_object,
LONG* acp_start,
LONG* acp_end,
TS_TEXTCHANGE* change) override;
IFACEMETHODIMP InsertTextAtSelection(DWORD flags,
const wchar_t* text_buffer,
ULONG text_buffer_size,
LONG* acp_start,
LONG* acp_end,
TS_TEXTCHANGE* text_change) override;
IFACEMETHODIMP QueryInsert(LONG acp_test_start,
LONG acp_test_end,
ULONG text_size,
LONG* acp_result_start,
LONG* acp_result_end) override;
IFACEMETHODIMP QueryInsertEmbedded(const GUID* service,
const FORMATETC* format,
BOOL* insertable) override;
IFACEMETHODIMP RequestAttrsAtPosition(LONG acp_pos,
ULONG attribute_buffer_size,
const TS_ATTRID* attribute_buffer,
DWORD flags) override;
IFACEMETHODIMP RequestAttrsTransitioningAtPosition(
LONG acp_pos,
ULONG attribute_buffer_size,
const TS_ATTRID* attribute_buffer,
DWORD flags) override;
IFACEMETHODIMP RequestLock(DWORD lock_flags, HRESULT* result) override;
IFACEMETHODIMP RequestSupportedAttrs(
DWORD flags,
ULONG attribute_buffer_size,
const TS_ATTRID* attribute_buffer) override;
IFACEMETHODIMP RetrieveRequestedAttrs(
ULONG attribute_buffer_size,
TS_ATTRVAL* attribute_buffer,
ULONG* attribute_buffer_copied) override;
IFACEMETHODIMP SetSelection(
ULONG selection_buffer_size,
const TS_SELECTION_ACP* selection_buffer) override;
IFACEMETHODIMP SetText(DWORD flags,
LONG acp_start,
LONG acp_end,
const wchar_t* text_buffer,
ULONG text_buffer_size,
TS_TEXTCHANGE* text_change) override;
IFACEMETHODIMP UnadviseSink(IUnknown* unknown) override;
// ITfContextOwnerCompositionSink:
IFACEMETHODIMP OnStartComposition(ITfCompositionView* composition_view,
BOOL* ok) override;
IFACEMETHODIMP OnUpdateComposition(ITfCompositionView* composition_view,
ITfRange* range) override;
IFACEMETHODIMP OnEndComposition(
ITfCompositionView* composition_view) override;
// ITfLanguageProfileNotifySink:
IFACEMETHODIMP OnLanguageChange(LANGID langid, BOOL* pfAccept) override;
IFACEMETHODIMP OnLanguageChanged() override;
// ITfTextEditSink:
IFACEMETHODIMP OnEndEdit(ITfContext* context,
TfEditCookie read_only_edit_cookie,
ITfEditRecord* edit_record) override;
// ITfKeyTraceEventSink
IFACEMETHODIMP OnKeyTraceDown(WPARAM wParam, LPARAM lParam) override;
IFACEMETHODIMP OnKeyTraceUp(WPARAM wParam, LPARAM lParam) override;
// Called after |TSFBridgeImpl::CreateDocumentManager| to tell that the
// text-store is successfully associated with a Context.
void OnContextInitialized(ITfContext* context);
// Sets currently focused TextInputClient.
void SetFocusedTextInputClient(HWND focused_window,
TextInputClient* text_input_client);
// Removes currently focused TextInputClient.
void RemoveFocusedTextInputClient(TextInputClient* text_input_client);
// Sets ImeKeyEventDispatcher pointer.
void SetImeKeyEventDispatcher(
ImeKeyEventDispatcher* ime_key_event_dispatcher);
// Removes ImeKeyEventDispatcher pointer.
void RemoveImeKeyEventDispatcher(
ImeKeyEventDispatcher* ime_key_event_dispatcher);
// Cancels the ongoing composition if exists.
bool CancelComposition();
// Confirms the ongoing composition if exists.
bool ConfirmComposition();
// Sends OnLayoutChange() via |text_store_acp_sink_|.
void SendOnLayoutChange();
// Sends a Url change notification via |text_store_acp_sink_| if the current
// version of TSF supports empty text stores.
bool MaybeSendOnUrlChanged();
// Sets the flag to indicate TSF support for empty text stores.
void UseEmptyTextStore(bool is_enabled);
private:
friend class TSFTextStoreTest;
friend class TSFTextStoreTestCallback;
// Reset states tracking the composition in the text store.
void ResetCompositionState();
// Terminate an active composition for this text store.
bool TerminateComposition();
// Compare our cached text buffer and selection with the up-to-date
// text buffer and selection from TextInputClient. We also update
// cached text buffer and selection with the new version. Then notify
// input service about the change.
void CalculateTextandSelectionDiffAndNotifyIfNeeded();
// Synthesize keyevent and send to text input client to fire corresponding
// javascript keyevent during composition.
void DispatchKeyEvent(ui::EventType type, WPARAM wparam, LPARAM lparam);
// Start new composition on existing text.
void StartCompositionOnExistingText() const;
// Start new composition with new text.
void StartCompositionOnNewText(size_t start_offset,
const std::u16string& composition_string);
// Commit and insert text into TextInputClient. End any ongoing composition.
void CommitTextAndEndCompositionIfAny(size_t old_size, size_t new_size) const;
// Checks if the document has a read-only lock.
bool HasReadLock() const;
// Checks if the document has a read and write lock.
bool HasReadWriteLock() const;
// Gets the display attribute structure.
bool GetDisplayAttribute(TfGuidAtom guid_atom,
TF_DISPLAYATTRIBUTE* attribute);
// Gets the committed string size and underline information of the context.
bool GetCompositionStatus(ITfContext* context,
const TfEditCookie read_only_edit_cookie,
size_t* committed_size,
ImeTextSpans* spans);
// Reset all cached flags when |TSFTextStore::RequestLock| returns.
void ResetCacheAfterEditSession();
// Returns if current input method is an IME.
bool IsInputIME() const;
// Gets the style information from the display attribute for the actively
// composed text.
void GetStyle(const TF_DISPLAYATTRIBUTE& attribute, ImeTextSpan* span);
// Indicates if it is a dummy/empty text store without any text capability.
bool is_empty_text_store_ = false;
// The reference count of this instance.
volatile LONG ref_count_ = 0;
// A pointer of ITextStoreACPSink, this instance is given in AdviseSink.
Microsoft::WRL::ComPtr<ITextStoreACPSink> text_store_acp_sink_;
// The current mask of |text_store_acp_sink_|.
DWORD text_store_acp_sink_mask_ = 0;
// HWND of the current view window which is set in SetFocusedTextInputClient.
HWND window_handle_ = nullptr;
// Current TextInputClient which is set in SetFocusedTextInputClient.
raw_ptr<TextInputClient> text_input_client_ = nullptr;
// ImeKeyEventDispatcher instance which is used dispatch key events.
raw_ptr<ImeKeyEventDispatcher, AcrossTasksDanglingUntriaged>
ime_key_event_dispatcher_ = nullptr;
// |string_buffer_document_| contains all string in current active view.
// |string_pending_insertion_| contains only string in current edit session.
// |composition_start_| indicates the location for a composition to start at.
// |has_composition_range_| indicates the state of composition.
// |composition_range_| indicates the range of composition if any.
// Example: "aoi" is committed, and "umi" is under composition.
// In current edit session, user press "i" on keyboard.
// |string_buffer_document_|: "aoiumi"
// |string_pending_insertion_| : "i"
// |composition_start_|: 3
// |has_composition_range_| = true;
// |composition_range_start_| = 3;
// |composition_range_end_| = 6;
std::u16string string_buffer_document_;
std::u16string string_pending_insertion_;
size_t composition_start_ = 0;
bool has_composition_range_ = false;
gfx::Range composition_range_;
// |on_start_composition_called_| indicates that OnStartComposition() is
// called duriing current edit session.
bool on_start_composition_called_ = false;
// |previous_composition_string_| indicicates composition string in last
// edit session during same composition. |previous_composition_start_|
// indicates composition start in last session during same composition. If
// RequestLock() is called during two edit sessions, we don't want to set same
// composition string twice. |previous_composition_selection_range_| indicates
// the selection range during composition. We want to send the selection
// change to blink if IME only change the selection range but not the
// composition text. |previous_text_spans_| saves the IME spans in previous
// edit session during same composition.
std::u16string previous_composition_string_;
size_t previous_composition_start_ = 0;
gfx::Range previous_composition_selection_range_ = gfx::Range::InvalidRange();
ImeTextSpans previous_text_spans_;
// |new_text_inserted_| indicates there is text to be inserted
// into blink during ITextStoreACP::SetText().
// |replace_text_range_| indicates the start and end offsets of the text to be
// replaced by the new text to be inserted.
// |replace_text_size_| indicates the size of the text to be inserted.
// Example: "k" is going to replace "i"
// |string_buffer_document_|: "aeiou"
// |new_text_inserted_|: true
// |replace_text_range_start_|: 2
// |replace_text_range_end_|: 3
// |replace_text_size_|: 1
bool new_text_inserted_ = false;
gfx::Range replace_text_range_;
size_t replace_text_size_;
// |buffer_from_client_| contains all string returned from
// TextInputClient::GetTextFromRange();
std::u16string buffer_from_client_;
// |selection_from_client_| indicates the selection range returned from
// TextInputClient::GetEditableSelectionRange();
gfx::Range selection_from_client_;
// |composition_range_from_client_| indicates the composition range returned
// from TextInputClient::GetCompositionTextRange();
gfx::Range composition_from_client_;
// |wparam_keydown_cached_| and |lparam_keydown_cached_| contains key event
// info that is used to synthesize key event during composition.
// |wparam_keydown_fired_| indicates if a keydown event has been fired.
WPARAM wparam_keydown_cached_ = 0;
LPARAM lparam_keydown_cached_ = 0;
WPARAM wparam_keydown_fired_ = 0;
// |selection_start_| and |selection_end_| indicates the selection range.
// Example: "iue" is selected
// |string_buffer_document_|: "aiueo"
// |selection_.start()|: 1
// |selection_.end()|: 4
gfx::Range selection_;
// Indicates if the selection is an interim character. Please refer to
// https://docs.microsoft.com/en-us/windows/win32/api/textstor/ns-textstor-ts_selectionstyle
bool is_selection_interim_char_ = false;
// |start_offset| and |end_offset| of |text_spans_| indicates
// the offsets in |string_buffer_document_|.
// Example: "aoi" is committed. There are two underlines in "umi" and "no".
// |string_buffer_document_|: "aoiumino"
// |composition_start_|: 3
// text_spans_[0].start_offset: 3
// text_spans_[0].end_offset: 6
// text_spans_[1].start_offset: 6
// text_spans_[1].end_offset: 8
ImeTextSpans text_spans_;
// |edit_flag_| indicates that the status is edited during
// ITextStoreACPSink::OnLockGranted().
bool edit_flag_ = false;
// Checks for re-entrancy while notifying changes to TSF.
bool is_notification_in_progress_ = false;
// Checks for re-entrancy while writing to text input client.
bool is_tic_write_in_progress_ = false;
// The type of current lock.
// 0: No lock.
// TS_LF_READ: read-only lock.
// TS_LF_READWRITE: read/write lock.
DWORD current_lock_type_ = 0;
// Queue of the lock request used in RequestLock().
std::deque<DWORD> lock_queue_;
// Category manager and Display attribute manager are used to obtain the
// attributes of the composition string.
Microsoft::WRL::ComPtr<ITfCategoryMgr> category_manager_;
Microsoft::WRL::ComPtr<ITfDisplayAttributeMgr> display_attribute_manager_;
Microsoft::WRL::ComPtr<ITfContext> context_;
Microsoft::WRL::ComPtr<ITfInputProcessorProfileMgr>
input_processor_profile_mgr_;
// Current list of requested supported attribute values.
// Currently the supported attributes are URL and InputScope.
std::vector<TS_ATTRID> supported_attrs_;
};
} // namespace ui
#endif // UI_BASE_IME_WIN_TSF_TEXT_STORE_H_