mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
419 lines
14 KiB
C++
419 lines
14 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=2 sw=2 et tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "ContentEventHandler.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsIContent.h"
|
|
#include "nsIEditor.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsPresContext.h"
|
|
#include "mozilla/AutoRestore.h"
|
|
#include "mozilla/EventDispatcher.h"
|
|
#include "mozilla/IMEStateManager.h"
|
|
#include "mozilla/MiscEvents.h"
|
|
#include "mozilla/TextComposition.h"
|
|
#include "mozilla/TextEvents.h"
|
|
|
|
using namespace mozilla::widget;
|
|
|
|
namespace mozilla {
|
|
|
|
#define IDEOGRAPHIC_SPACE (NS_LITERAL_STRING("\x3000"))
|
|
|
|
/******************************************************************************
|
|
* TextComposition
|
|
******************************************************************************/
|
|
|
|
TextComposition::TextComposition(nsPresContext* aPresContext,
|
|
nsINode* aNode,
|
|
WidgetGUIEvent* aEvent)
|
|
: mPresContext(aPresContext)
|
|
, mNode(aNode)
|
|
, mNativeContext(aEvent->widget->GetInputContext().mNativeIMEContext)
|
|
, mCompositionStartOffset(0)
|
|
, mCompositionTargetOffset(0)
|
|
, mIsSynthesizedForTests(aEvent->mFlags.mIsSynthesizedForTests)
|
|
, mIsComposing(false)
|
|
, mIsEditorHandlingEvent(false)
|
|
, mIsRequestingCommit(false)
|
|
, mIsRequestingCancel(false)
|
|
, mRequestedToCommitOrCancel(false)
|
|
{
|
|
}
|
|
|
|
void
|
|
TextComposition::Destroy()
|
|
{
|
|
mPresContext = nullptr;
|
|
mNode = nullptr;
|
|
// TODO: If the editor is still alive and this is held by it, we should tell
|
|
// this being destroyed for cleaning up the stuff.
|
|
}
|
|
|
|
bool
|
|
TextComposition::MatchesNativeContext(nsIWidget* aWidget) const
|
|
{
|
|
return mNativeContext == aWidget->GetInputContext().mNativeIMEContext;
|
|
}
|
|
|
|
void
|
|
TextComposition::DispatchEvent(WidgetGUIEvent* aEvent,
|
|
nsEventStatus* aStatus,
|
|
EventDispatchingCallback* aCallBack,
|
|
bool aIsSynthesized)
|
|
{
|
|
if (Destroyed()) {
|
|
*aStatus = nsEventStatus_eConsumeNoDefault;
|
|
return;
|
|
}
|
|
|
|
// If this instance has requested to commit or cancel composition but
|
|
// is not synthesizing commit event, that means that the IME commits or
|
|
// cancels the composition asynchronously. Typically, iBus behaves so.
|
|
// Then, synthesized events which were dispatched immediately after
|
|
// the request has already committed our editor's composition string and
|
|
// told it to web apps. Therefore, we should ignore the delayed events.
|
|
if (mRequestedToCommitOrCancel && !aIsSynthesized) {
|
|
*aStatus = nsEventStatus_eConsumeNoDefault;
|
|
return;
|
|
}
|
|
|
|
if (aEvent->message == NS_COMPOSITION_UPDATE) {
|
|
mLastData = aEvent->AsCompositionEvent()->data;
|
|
}
|
|
|
|
EventDispatcher::Dispatch(mNode, mPresContext,
|
|
aEvent, nullptr, aStatus, aCallBack);
|
|
|
|
if (NS_WARN_IF(Destroyed())) {
|
|
return;
|
|
}
|
|
|
|
// Emulate editor behavior of text event handler if no editor handles
|
|
// composition/text events.
|
|
if (aEvent->message == NS_TEXT_TEXT && !HasEditor()) {
|
|
EditorWillHandleTextEvent(aEvent->AsTextEvent());
|
|
EditorDidHandleTextEvent();
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
else if (aEvent->message == NS_COMPOSITION_END) {
|
|
MOZ_ASSERT(!mIsComposing, "Why is the editor still composing?");
|
|
MOZ_ASSERT(!HasEditor(), "Why does the editor still keep to hold this?");
|
|
}
|
|
#endif // #ifdef DEBUG
|
|
|
|
// Notify composition update to widget if possible
|
|
NotityUpdateComposition(aEvent);
|
|
}
|
|
|
|
void
|
|
TextComposition::NotityUpdateComposition(WidgetGUIEvent* aEvent)
|
|
{
|
|
nsEventStatus status;
|
|
|
|
// When compositon start, notify the rect of first offset character.
|
|
// When not compositon start, notify the rect of selected composition
|
|
// string if text event.
|
|
if (aEvent->message == NS_COMPOSITION_START) {
|
|
nsCOMPtr<nsIWidget> widget = mPresContext->GetRootWidget();
|
|
// Update composition start offset
|
|
WidgetQueryContentEvent selectedTextEvent(true,
|
|
NS_QUERY_SELECTED_TEXT,
|
|
widget);
|
|
widget->DispatchEvent(&selectedTextEvent, status);
|
|
if (selectedTextEvent.mSucceeded) {
|
|
mCompositionStartOffset = selectedTextEvent.mReply.mOffset;
|
|
} else {
|
|
// Unknown offset
|
|
NS_WARNING("Cannot get start offset of IME composition");
|
|
mCompositionStartOffset = 0;
|
|
}
|
|
mCompositionTargetOffset = mCompositionStartOffset;
|
|
} else if (aEvent->mClass != eTextEventClass) {
|
|
return;
|
|
} else {
|
|
mCompositionTargetOffset =
|
|
mCompositionStartOffset + aEvent->AsTextEvent()->TargetClauseOffset();
|
|
}
|
|
|
|
NotifyIME(NOTIFY_IME_OF_COMPOSITION_UPDATE);
|
|
}
|
|
|
|
void
|
|
TextComposition::DispatchCompositionEventRunnable(uint32_t aEventMessage,
|
|
const nsAString& aData,
|
|
bool aIsSynthesizingCommit)
|
|
{
|
|
nsContentUtils::AddScriptRunner(
|
|
new CompositionEventDispatcher(this, mNode, aEventMessage, aData,
|
|
aIsSynthesizingCommit));
|
|
}
|
|
|
|
nsresult
|
|
TextComposition::RequestToCommit(nsIWidget* aWidget, bool aDiscard)
|
|
{
|
|
// If this composition is already requested to be committed or canceled,
|
|
// we don't need to request it again because even if the first request
|
|
// failed, new request won't success, probably. And we shouldn't synthesize
|
|
// events for committing or canceling composition twice or more times.
|
|
if (mRequestedToCommitOrCancel) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<TextComposition> kungFuDeathGrip(this);
|
|
|
|
{
|
|
AutoRestore<bool> saveRequestingCancel(mIsRequestingCancel);
|
|
AutoRestore<bool> saveRequestingCommit(mIsRequestingCommit);
|
|
if (aDiscard) {
|
|
mIsRequestingCancel = true;
|
|
mIsRequestingCommit = false;
|
|
} else {
|
|
mIsRequestingCancel = false;
|
|
mIsRequestingCommit = true;
|
|
}
|
|
// FYI: CompositionEvent and TextEvent caused by a call of NotifyIME()
|
|
// may be discarded by PresShell if it's not safe to dispatch the
|
|
// event.
|
|
nsresult rv =
|
|
aWidget->NotifyIME(IMENotification(aDiscard ?
|
|
REQUEST_TO_CANCEL_COMPOSITION :
|
|
REQUEST_TO_COMMIT_COMPOSITION));
|
|
if (rv == NS_ERROR_NOT_IMPLEMENTED) {
|
|
return rv;
|
|
}
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
mRequestedToCommitOrCancel = true;
|
|
|
|
// If the request is performed synchronously, this must be already destroyed.
|
|
if (Destroyed()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Otherwise, synthesize the commit in content.
|
|
nsAutoString data(aDiscard ? EmptyString() : mLastData);
|
|
bool changingData = mLastData != data;
|
|
if (changingData) {
|
|
DispatchCompositionEventRunnable(NS_COMPOSITION_UPDATE, data, true);
|
|
}
|
|
// If the last composition string and new data are different, we need to
|
|
// dispatch text event for removing IME selection. However, if the commit
|
|
// string is empty string and it's not changed from the last data, we don't
|
|
// need to dispatch text event.
|
|
if (changingData || !data.IsEmpty()) {
|
|
DispatchCompositionEventRunnable(NS_TEXT_TEXT, data, true);
|
|
}
|
|
DispatchCompositionEventRunnable(NS_COMPOSITION_END, data, true);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
TextComposition::NotifyIME(IMEMessage aMessage)
|
|
{
|
|
NS_ENSURE_TRUE(mPresContext, NS_ERROR_NOT_AVAILABLE);
|
|
return IMEStateManager::NotifyIME(aMessage, mPresContext);
|
|
}
|
|
|
|
void
|
|
TextComposition::EditorWillHandleTextEvent(const WidgetTextEvent* aTextEvent)
|
|
{
|
|
mIsComposing = aTextEvent->IsComposing();
|
|
mRanges = aTextEvent->mRanges;
|
|
mIsEditorHandlingEvent = true;
|
|
|
|
MOZ_ASSERT(mLastData == aTextEvent->theText,
|
|
"The text of a text event must be same as previous data attribute value "
|
|
"of the latest compositionupdate event");
|
|
}
|
|
|
|
void
|
|
TextComposition::EditorDidHandleTextEvent()
|
|
{
|
|
mString = mLastData;
|
|
mIsEditorHandlingEvent = false;
|
|
}
|
|
|
|
void
|
|
TextComposition::StartHandlingComposition(nsIEditor* aEditor)
|
|
{
|
|
MOZ_ASSERT(!HasEditor(), "There is a handling editor already");
|
|
mEditorWeak = do_GetWeakReference(aEditor);
|
|
}
|
|
|
|
void
|
|
TextComposition::EndHandlingComposition(nsIEditor* aEditor)
|
|
{
|
|
#ifdef DEBUG
|
|
nsCOMPtr<nsIEditor> editor = GetEditor();
|
|
MOZ_ASSERT(editor == aEditor, "Another editor handled the composition?");
|
|
#endif // #ifdef DEBUG
|
|
mEditorWeak = nullptr;
|
|
}
|
|
|
|
already_AddRefed<nsIEditor>
|
|
TextComposition::GetEditor() const
|
|
{
|
|
nsCOMPtr<nsIEditor> editor = do_QueryReferent(mEditorWeak);
|
|
return editor.forget();
|
|
}
|
|
|
|
bool
|
|
TextComposition::HasEditor() const
|
|
{
|
|
nsCOMPtr<nsIEditor> editor = GetEditor();
|
|
return !!editor;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* TextComposition::CompositionEventDispatcher
|
|
******************************************************************************/
|
|
|
|
TextComposition::CompositionEventDispatcher::CompositionEventDispatcher(
|
|
TextComposition* aComposition,
|
|
nsINode* aEventTarget,
|
|
uint32_t aEventMessage,
|
|
const nsAString& aData,
|
|
bool aIsSynthesizedEvent)
|
|
: mTextComposition(aComposition)
|
|
, mEventTarget(aEventTarget)
|
|
, mEventMessage(aEventMessage)
|
|
, mData(aData)
|
|
, mIsSynthesizedEvent(aIsSynthesizedEvent)
|
|
{
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TextComposition::CompositionEventDispatcher::Run()
|
|
{
|
|
nsRefPtr<nsPresContext> presContext = mTextComposition->mPresContext;
|
|
if (!presContext || !presContext->GetPresShell() ||
|
|
presContext->GetPresShell()->IsDestroying()) {
|
|
return NS_OK; // cannot dispatch any events anymore
|
|
}
|
|
|
|
// The widget can be different from the widget which has dispatched
|
|
// composition events because GetWidget() returns a widget which is proper
|
|
// for calling NotifyIME(). However, this must no be problem since both
|
|
// widget should share native IME context. Therefore, even if an event
|
|
// handler uses the widget for requesting IME to commit or cancel, it works.
|
|
nsCOMPtr<nsIWidget> widget(mTextComposition->GetWidget());
|
|
if (NS_WARN_IF(!widget)) {
|
|
return NS_OK; // cannot dispatch any events anymore
|
|
}
|
|
|
|
nsEventStatus status = nsEventStatus_eIgnore;
|
|
switch (mEventMessage) {
|
|
case NS_COMPOSITION_START: {
|
|
WidgetCompositionEvent compStart(true, NS_COMPOSITION_START, widget);
|
|
WidgetQueryContentEvent selectedText(true, NS_QUERY_SELECTED_TEXT,
|
|
widget);
|
|
ContentEventHandler handler(presContext);
|
|
handler.OnQuerySelectedText(&selectedText);
|
|
NS_ASSERTION(selectedText.mSucceeded, "Failed to get selected text");
|
|
compStart.data = selectedText.mReply.mString;
|
|
IMEStateManager::DispatchCompositionEvent(mEventTarget, presContext,
|
|
&compStart, &status, nullptr,
|
|
mIsSynthesizedEvent);
|
|
break;
|
|
}
|
|
case NS_COMPOSITION_UPDATE:
|
|
case NS_COMPOSITION_END: {
|
|
WidgetCompositionEvent compEvent(true, mEventMessage, widget);
|
|
compEvent.data = mData;
|
|
IMEStateManager::DispatchCompositionEvent(mEventTarget, presContext,
|
|
&compEvent, &status, nullptr,
|
|
mIsSynthesizedEvent);
|
|
break;
|
|
}
|
|
case NS_TEXT_TEXT: {
|
|
WidgetTextEvent textEvent(true, NS_TEXT_TEXT, widget);
|
|
textEvent.theText = mData;
|
|
IMEStateManager::DispatchCompositionEvent(mEventTarget, presContext,
|
|
&textEvent, &status, nullptr,
|
|
mIsSynthesizedEvent);
|
|
break;
|
|
}
|
|
default:
|
|
MOZ_CRASH("Unsupported event");
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* TextCompositionArray
|
|
******************************************************************************/
|
|
|
|
TextCompositionArray::index_type
|
|
TextCompositionArray::IndexOf(nsIWidget* aWidget)
|
|
{
|
|
for (index_type i = Length(); i > 0; --i) {
|
|
if (ElementAt(i - 1)->MatchesNativeContext(aWidget)) {
|
|
return i - 1;
|
|
}
|
|
}
|
|
return NoIndex;
|
|
}
|
|
|
|
TextCompositionArray::index_type
|
|
TextCompositionArray::IndexOf(nsPresContext* aPresContext)
|
|
{
|
|
for (index_type i = Length(); i > 0; --i) {
|
|
if (ElementAt(i - 1)->GetPresContext() == aPresContext) {
|
|
return i - 1;
|
|
}
|
|
}
|
|
return NoIndex;
|
|
}
|
|
|
|
TextCompositionArray::index_type
|
|
TextCompositionArray::IndexOf(nsPresContext* aPresContext,
|
|
nsINode* aNode)
|
|
{
|
|
index_type index = IndexOf(aPresContext);
|
|
if (index == NoIndex) {
|
|
return NoIndex;
|
|
}
|
|
nsINode* node = ElementAt(index)->GetEventTargetNode();
|
|
return node == aNode ? index : NoIndex;
|
|
}
|
|
|
|
TextComposition*
|
|
TextCompositionArray::GetCompositionFor(nsIWidget* aWidget)
|
|
{
|
|
index_type i = IndexOf(aWidget);
|
|
return i != NoIndex ? ElementAt(i) : nullptr;
|
|
}
|
|
|
|
TextComposition*
|
|
TextCompositionArray::GetCompositionFor(nsPresContext* aPresContext,
|
|
nsINode* aNode)
|
|
{
|
|
index_type i = IndexOf(aPresContext, aNode);
|
|
return i != NoIndex ? ElementAt(i) : nullptr;
|
|
}
|
|
|
|
TextComposition*
|
|
TextCompositionArray::GetCompositionInContent(nsPresContext* aPresContext,
|
|
nsIContent* aContent)
|
|
{
|
|
// There should be only one composition per content object.
|
|
for (index_type i = Length(); i > 0; --i) {
|
|
nsINode* node = ElementAt(i - 1)->GetEventTargetNode();
|
|
if (node && nsContentUtils::ContentIsDescendantOf(node, aContent)) {
|
|
return ElementAt(i - 1);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
} // namespace mozilla
|