/*
* Copyright (C) 2007 Kevin Ollivier <kevino@theolliviers.com>
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "EditorClientWx.h"
#include "EditCommand.h"
#include "Editor.h"
#include "FocusController.h"
#include "Frame.h"
#include "FrameView.h"
#include "HostWindow.h"
#include "KeyboardEvent.h"
#include "KeyboardCodes.h"
#include "NotImplemented.h"
#include "Page.h"
#include "PlatformKeyboardEvent.h"
#include "PlatformString.h"
#include "SelectionController.h"
#include "WebFrame.h"
#include "WebFramePrivate.h"
#include "WebView.h"
#include "WebViewPrivate.h"
#include <stdio.h>
namespace WebCore {
static const unsigned CtrlKey = 1 << 0;
static const unsigned AltKey = 1 << 1;
static const unsigned ShiftKey = 1 << 2;
struct KeyDownEntry {
unsigned virtualKey;
unsigned modifiers;
const char* name;
};
struct KeyPressEntry {
unsigned charCode;
unsigned modifiers;
const char* name;
};
static const KeyDownEntry keyDownEntries[] = {
{ VK_LEFT, 0, "MoveLeft" },
{ VK_LEFT, ShiftKey, "MoveLeftAndModifySelection" },
{ VK_LEFT, CtrlKey, "MoveWordLeft" },
{ VK_LEFT, CtrlKey | ShiftKey, "MoveWordLeftAndModifySelection" },
{ VK_RIGHT, 0, "MoveRight" },
{ VK_RIGHT, ShiftKey, "MoveRightAndModifySelection" },
{ VK_RIGHT, CtrlKey, "MoveWordRight" },
{ VK_RIGHT, CtrlKey | ShiftKey, "MoveWordRightAndModifySelection" },
{ VK_UP, 0, "MoveUp" },
{ VK_UP, ShiftKey, "MoveUpAndModifySelection" },
{ VK_PRIOR, ShiftKey, "MovePageUpAndModifySelection" },
{ VK_DOWN, 0, "MoveDown" },
{ VK_DOWN, ShiftKey, "MoveDownAndModifySelection" },
{ VK_NEXT, ShiftKey, "MovePageDownAndModifySelection" },
{ VK_PRIOR, 0, "MovePageUp" },
{ VK_NEXT, 0, "MovePageDown" },
{ VK_HOME, 0, "MoveToBeginningOfLine" },
{ VK_HOME, ShiftKey, "MoveToBeginningOfLineAndModifySelection" },
{ VK_HOME, CtrlKey, "MoveToBeginningOfDocument" },
{ VK_HOME, CtrlKey | ShiftKey, "MoveToBeginningOfDocumentAndModifySelection" },
{ VK_END, 0, "MoveToEndOfLine" },
{ VK_END, ShiftKey, "MoveToEndOfLineAndModifySelection" },
{ VK_END, CtrlKey, "MoveToEndOfDocument" },
{ VK_END, CtrlKey | ShiftKey, "MoveToEndOfDocumentAndModifySelection" },
{ VK_BACK, 0, "DeleteBackward" },
{ VK_BACK, ShiftKey, "DeleteBackward" },
{ VK_DELETE, 0, "DeleteForward" },
{ VK_BACK, CtrlKey, "DeleteWordBackward" },
{ VK_DELETE, CtrlKey, "DeleteWordForward" },
{ 'B', CtrlKey, "ToggleBold" },
{ 'I', CtrlKey, "ToggleItalic" },
{ VK_ESCAPE, 0, "Cancel" },
//FIXME: this'll never happen. We can trash it or make it a normal period
{ VK_OEM_PERIOD, CtrlKey, "Cancel" },
{ VK_TAB, 0, "InsertTab" },
{ VK_TAB, ShiftKey, "InsertBacktab" },
{ VK_RETURN, 0, "InsertNewline" },
{ VK_RETURN, CtrlKey, "InsertNewline" },
{ VK_RETURN, AltKey, "InsertNewline" },
{ VK_RETURN, AltKey | ShiftKey, "InsertNewline" },
{ 'A', CtrlKey, "SelectAll" },
{ 'Z', CtrlKey, "Undo" },
{ 'Z', CtrlKey | ShiftKey, "Redo" },
};
static const KeyPressEntry keyPressEntries[] = {
{ '\t', 0, "InsertTab" },
{ '\t', ShiftKey, "InsertBacktab" },
{ '\r', 0, "InsertNewline" },
{ '\r', CtrlKey, "InsertNewline" },
{ '\r', AltKey, "InsertNewline" },
{ '\r', AltKey | ShiftKey, "InsertNewline" },
};
EditorClientWx::~EditorClientWx()
{
m_page = NULL;
}
void EditorClientWx::setPage(Page* page)
{
m_page = page;
}
void EditorClientWx::pageDestroyed()
{
delete this;
}
bool EditorClientWx::shouldDeleteRange(Range*)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldShowDeleteInterface(HTMLElement*)
{
notImplemented();
return false;
}
bool EditorClientWx::smartInsertDeleteEnabled()
{
notImplemented();
return false;
}
bool EditorClientWx::isSelectTrailingWhitespaceEnabled()
{
notImplemented();
return false;
}
bool EditorClientWx::isContinuousSpellCheckingEnabled()
{
notImplemented();
return false;
}
void EditorClientWx::toggleContinuousSpellChecking()
{
notImplemented();
}
bool EditorClientWx::isGrammarCheckingEnabled()
{
notImplemented();
return false;
}
void EditorClientWx::toggleGrammarChecking()
{
notImplemented();
}
int EditorClientWx::spellCheckerDocumentTag()
{
notImplemented();
return 0;
}
bool EditorClientWx::selectWordBeforeMenuEvent()
{
notImplemented();
return false;
}
bool EditorClientWx::isEditable()
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin)
return webKitWin->IsEditable();
}
return false;
}
bool EditorClientWx::shouldBeginEditing(Range*)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldEndEditing(Range*)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldInsertNode(Node*, Range*,
EditorInsertAction)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldInsertText(const String&, Range*,
EditorInsertAction)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldApplyStyle(CSSStyleDeclaration*,
Range*)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldMoveRangeAfterDelete(Range*, Range*)
{
notImplemented();
return true;
}
bool EditorClientWx::shouldChangeSelectedRange(Range* fromRange, Range* toRange,
EAffinity, bool stillSelecting)
{
notImplemented();
return true;
}
void EditorClientWx::didBeginEditing()
{
notImplemented();
}
void EditorClientWx::respondToChangedContents()
{
notImplemented();
}
void EditorClientWx::didEndEditing()
{
notImplemented();
}
void EditorClientWx::didWriteSelectionToPasteboard()
{
notImplemented();
}
void EditorClientWx::didSetSelectionTypesForPasteboard()
{
notImplemented();
}
void EditorClientWx::registerCommandForUndo(PassRefPtr<EditCommand> command)
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
webKitWin->m_impl->undoStack.append(EditCommandWx(command));
}
}
}
void EditorClientWx::registerCommandForRedo(PassRefPtr<EditCommand> command)
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
webKitWin->m_impl->redoStack.insert(0, EditCommandWx(command));
}
}
}
void EditorClientWx::clearUndoRedoOperations()
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
webKitWin->m_impl->redoStack.clear();
webKitWin->m_impl->undoStack.clear();
}
}
}
bool EditorClientWx::canUndo() const
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
return webKitWin->m_impl->undoStack.size() != 0;
}
}
return false;
}
bool EditorClientWx::canRedo() const
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin && webKitWin) {
return webKitWin->m_impl->redoStack.size() != 0;
}
}
return false;
}
void EditorClientWx::undo()
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
webKitWin->m_impl->undoStack.last().editCommand()->unapply();
webKitWin->m_impl->undoStack.removeLast();
}
}
}
void EditorClientWx::redo()
{
Frame* frame = m_page->focusController()->focusedOrMainFrame();
if (frame) {
wxWebView* webKitWin = dynamic_cast<wxWebView*>(frame->view()->hostWindow()->platformPageClient());
if (webKitWin) {
webKitWin->m_impl->redoStack.first().editCommand()->reapply();
webKitWin->m_impl->redoStack.remove(0);
}
}
}
bool EditorClientWx::handleEditingKeyboardEvent(KeyboardEvent* event)
{
Node* node = event->target()->toNode();
ASSERT(node);
Frame* frame = node->document()->frame();
ASSERT(frame);
const PlatformKeyboardEvent* keyEvent = event->keyEvent();
//NB: this is what windows does, but they also have a keypress event for Alt+Enter which clearly won't get hit with this
if (!keyEvent || keyEvent->altKey()) // do not treat this as text input if Alt is down
return false;
Editor::Command command = frame->editor()->command(interpretKeyEvent(event));
if (keyEvent->type() == PlatformKeyboardEvent::RawKeyDown) {
// WebKit doesn't have enough information about mode to decide how commands that just insert text if executed via Editor should be treated,
// so we leave it upon WebCore to either handle them immediately (e.g. Tab that changes focus) or if not to let a CHAR event be generated
// (e.g. Tab that inserts a Tab character, or Enter).
return !command.isTextInsertion() && command.execute(event);
}
if (command.execute(event))
return true;
// Don't insert null or control characters as they can result in unexpected behaviour
if (event->charCode() < ' ')
return false;
return frame->editor()->insertText(event->keyEvent()->text(), event);
}
const char* EditorClientWx::interpretKeyEvent(const KeyboardEvent* evt)
{
ASSERT(evt->keyEvent()->type() == PlatformKeyboardEvent::RawKeyDown || evt->keyEvent()->type() == PlatformKeyboardEvent::Char);
static HashMap<int, const char*>* keyDownCommandsMap = 0;
static HashMap<int, const char*>* keyPressCommandsMap = 0;
if (!keyDownCommandsMap) {
keyDownCommandsMap = new HashMap<int, const char*>;
keyPressCommandsMap = new HashMap<int, const char*>;
for (unsigned i = 0; i < WXSIZEOF(keyDownEntries); i++)
keyDownCommandsMap->set(keyDownEntries[i].modifiers << 16 | keyDownEntries[i].virtualKey, keyDownEntries[i].name);
for (unsigned i = 0; i < WXSIZEOF(keyPressEntries); i++)
keyPressCommandsMap->set(keyPressEntries[i].modifiers << 16 | keyPressEntries[i].charCode, keyPressEntries[i].name);
}
unsigned modifiers = 0;
if (evt->shiftKey())
modifiers |= ShiftKey;
if (evt->altKey())
modifiers |= AltKey;
if (evt->ctrlKey())
modifiers |= CtrlKey;
if (evt->keyEvent()->type() == PlatformKeyboardEvent::RawKeyDown) {
int mapKey = modifiers << 16 | evt->keyCode();
return mapKey ? keyDownCommandsMap->get(mapKey) : 0;
}
int mapKey = modifiers << 16 | evt->charCode();
return mapKey ? keyPressCommandsMap->get(mapKey) : 0;
}
void EditorClientWx::handleInputMethodKeydown(KeyboardEvent* event)
{
// NOTE: we don't currently need to handle this. When key events occur,
// both this method and handleKeyboardEvent get a chance at handling them.
// We might use this method later on for IME-specific handling.
}
void EditorClientWx::handleKeyboardEvent(KeyboardEvent* event)
{
if (handleEditingKeyboardEvent(event))
event->setDefaultHandled();
}
void EditorClientWx::textFieldDidBeginEditing(Element*)
{
notImplemented();
}
void EditorClientWx::textFieldDidEndEditing(Element*)
{
notImplemented();
}
void EditorClientWx::textDidChangeInTextField(Element*)
{
notImplemented();
}
bool EditorClientWx::doTextFieldCommandFromEvent(Element*, KeyboardEvent*)
{
notImplemented();
return false;
}
void EditorClientWx::textWillBeDeletedInTextField(Element*)
{
notImplemented();
}
void EditorClientWx::textDidChangeInTextArea(Element*)
{
notImplemented();
}
void EditorClientWx::respondToChangedSelection()
{
notImplemented();
}
void EditorClientWx::ignoreWordInSpellDocument(const String&)
{
notImplemented();
}
void EditorClientWx::learnWord(const String&)
{
notImplemented();
}
void EditorClientWx::checkSpellingOfString(const UChar*, int length, int* misspellingLocation, int* misspellingLength)
{
notImplemented();
}
void EditorClientWx::checkGrammarOfString(const UChar*, int length, Vector<GrammarDetail>&, int* badGrammarLocation, int* badGrammarLength)
{
notImplemented();
}
void EditorClientWx::updateSpellingUIWithGrammarString(const String&, const GrammarDetail& detail)
{
notImplemented();
}
void EditorClientWx::updateSpellingUIWithMisspelledWord(const String&)
{
notImplemented();
}
void EditorClientWx::showSpellingUI(bool show)
{
notImplemented();
}
bool EditorClientWx::spellingUIIsShowing()
{
notImplemented();
return false;
}
void EditorClientWx::getGuessesForWord(const String&, Vector<String>& guesses)
{
notImplemented();
}
String EditorClientWx::getAutoCorrectSuggestionForMisspelledWord(const WebCore::String&)
{
notImplemented();
return String();
}
void EditorClientWx::setInputMethodState(bool enabled)
{
notImplemented();
}
}