2010-06-10 04:56:17 +04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2009 Apple Inc. 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 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 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.
|
|
|
|
*
|
|
|
|
* Modified by Josh Aas of Mozilla Corporation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#import "ComplexTextInputPanel.h"
|
2011-05-27 12:15:20 +04:00
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
|
2013-12-20 03:19:17 +04:00
|
|
|
#include <algorithm>
|
2011-05-27 12:15:20 +04:00
|
|
|
#include "mozilla/Preferences.h"
|
2013-12-20 03:19:17 +04:00
|
|
|
#include "nsChildView.h"
|
2011-05-27 12:15:20 +04:00
|
|
|
|
|
|
|
using namespace mozilla;
|
2010-06-10 04:56:17 +04:00
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
extern "C" OSStatus TSMProcessRawKeyEvent(EventRef anEvent);
|
|
|
|
|
2010-06-10 04:56:17 +04:00
|
|
|
#define kInputWindowHeight 20
|
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
@interface ComplexTextInputPanelImpl : NSPanel {
|
|
|
|
NSTextView *mInputTextView;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (ComplexTextInputPanelImpl*)sharedComplexTextInputPanelImpl;
|
|
|
|
|
|
|
|
- (NSTextInputContext*)inputContext;
|
|
|
|
- (void)interpretKeyEvent:(NSEvent*)event string:(NSString**)string;
|
|
|
|
- (void)cancelComposition;
|
|
|
|
- (BOOL)inComposition;
|
|
|
|
|
|
|
|
// This places the text input panel fully onscreen and below the lower left
|
|
|
|
// corner of the focused plugin.
|
|
|
|
- (void)adjustTo:(NSPoint)point;
|
2010-06-10 04:56:17 +04:00
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation ComplexTextInputPanelImpl
|
|
|
|
|
|
|
|
+ (ComplexTextInputPanelImpl*)sharedComplexTextInputPanelImpl
|
2010-06-10 04:56:17 +04:00
|
|
|
{
|
2014-12-11 17:44:07 +03:00
|
|
|
static ComplexTextInputPanelImpl *sComplexTextInputPanelImpl;
|
|
|
|
if (!sComplexTextInputPanelImpl)
|
|
|
|
sComplexTextInputPanelImpl = [[ComplexTextInputPanelImpl alloc] init];
|
|
|
|
return sComplexTextInputPanelImpl;
|
2010-06-10 04:56:17 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
- (id)init
|
|
|
|
{
|
|
|
|
// In the original Apple code the style mask is given by a function which is not open source.
|
|
|
|
// What could possibly be worth hiding in that function, I do not know.
|
|
|
|
// Courtesy of gdb: stylemask: 011000011111, 0x61f
|
|
|
|
self = [super initWithContentRect:NSZeroRect styleMask:0x61f backing:NSBackingStoreBuffered defer:YES];
|
|
|
|
if (!self)
|
|
|
|
return nil;
|
|
|
|
|
|
|
|
// Set the frame size.
|
|
|
|
NSRect visibleFrame = [[NSScreen mainScreen] visibleFrame];
|
|
|
|
NSRect frame = NSMakeRect(visibleFrame.origin.x, visibleFrame.origin.y, visibleFrame.size.width, kInputWindowHeight);
|
|
|
|
|
|
|
|
[self setFrame:frame display:NO];
|
|
|
|
|
|
|
|
mInputTextView = [[NSTextView alloc] initWithFrame:[self.contentView frame]];
|
|
|
|
mInputTextView.autoresizingMask = NSViewWidthSizable | NSViewHeightSizable | NSViewMaxXMargin | NSViewMinXMargin | NSViewMaxYMargin | NSViewMinYMargin;
|
|
|
|
|
|
|
|
NSScrollView* scrollView = [[NSScrollView alloc] initWithFrame:[self.contentView frame]];
|
|
|
|
scrollView.documentView = mInputTextView;
|
|
|
|
self.contentView = scrollView;
|
|
|
|
[scrollView release];
|
|
|
|
|
|
|
|
[self setFloatingPanel:YES];
|
|
|
|
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
|
|
selector:@selector(keyboardInputSourceChanged:)
|
|
|
|
name:NSTextInputContextKeyboardSelectionDidChangeNotification
|
|
|
|
object:nil];
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
|
|
|
|
[mInputTextView release];
|
|
|
|
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)keyboardInputSourceChanged:(NSNotification *)notification
|
|
|
|
{
|
2012-08-22 19:56:38 +04:00
|
|
|
static int8_t sDoCancel = -1;
|
2011-05-11 08:52:45 +04:00
|
|
|
if (!sDoCancel || ![self inComposition]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (sDoCancel < 0) {
|
2011-09-29 10:19:26 +04:00
|
|
|
bool cancelComposition = false;
|
2011-05-27 12:15:20 +04:00
|
|
|
static const char* kPrefName =
|
|
|
|
"ui.plugin.cancel_composition_at_input_source_changed";
|
|
|
|
nsresult rv = Preferences::GetBool(kPrefName, &cancelComposition);
|
2011-05-11 08:52:45 +04:00
|
|
|
NS_ENSURE_SUCCESS(rv, );
|
|
|
|
sDoCancel = cancelComposition ? 1 : 0;
|
|
|
|
}
|
|
|
|
if (sDoCancel) {
|
|
|
|
[self cancelComposition];
|
|
|
|
}
|
2010-06-10 04:56:17 +04:00
|
|
|
}
|
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
- (void)interpretKeyEvent:(NSEvent*)event string:(NSString**)string
|
2010-06-10 04:56:17 +04:00
|
|
|
{
|
|
|
|
*string = nil;
|
|
|
|
|
2015-02-20 19:37:02 +03:00
|
|
|
if (![[mInputTextView inputContext] handleEvent:event]) {
|
|
|
|
return;
|
2014-12-11 17:44:07 +03:00
|
|
|
}
|
2010-06-10 04:56:17 +04:00
|
|
|
|
|
|
|
if ([mInputTextView hasMarkedText]) {
|
|
|
|
// Don't show the input method window for dead keys
|
2014-12-11 17:44:07 +03:00
|
|
|
if ([[event characters] length] > 0) {
|
2010-06-10 04:56:17 +04:00
|
|
|
[self orderFront:nil];
|
2014-12-11 17:44:07 +03:00
|
|
|
}
|
|
|
|
return;
|
2010-06-10 04:56:17 +04:00
|
|
|
} else {
|
|
|
|
[self orderOut:nil];
|
|
|
|
|
|
|
|
NSString *text = [[mInputTextView textStorage] string];
|
2014-12-11 17:44:07 +03:00
|
|
|
if ([text length] > 0) {
|
2010-06-10 04:56:17 +04:00
|
|
|
*string = [[text copy] autorelease];
|
2014-12-11 17:44:07 +03:00
|
|
|
}
|
2010-06-10 04:56:17 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
[mInputTextView setString:@""];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSTextInputContext*)inputContext
|
|
|
|
{
|
|
|
|
return [mInputTextView inputContext];
|
|
|
|
}
|
|
|
|
|
2010-12-03 08:22:20 +03:00
|
|
|
- (void)cancelComposition
|
|
|
|
{
|
|
|
|
[mInputTextView setString:@""];
|
|
|
|
[self orderOut:nil];
|
|
|
|
}
|
|
|
|
|
2011-01-21 04:08:11 +03:00
|
|
|
- (BOOL)inComposition
|
|
|
|
{
|
|
|
|
return [mInputTextView hasMarkedText];
|
|
|
|
}
|
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
- (void)adjustTo:(NSPoint)point
|
2013-12-20 03:19:17 +04:00
|
|
|
{
|
|
|
|
NSRect selfRect = [self frame];
|
2014-12-11 17:44:07 +03:00
|
|
|
NSRect rect = NSMakeRect(point.x,
|
|
|
|
point.y - selfRect.size.height,
|
|
|
|
500,
|
2013-12-20 03:19:17 +04:00
|
|
|
selfRect.size.height);
|
|
|
|
|
|
|
|
// Adjust to screen.
|
|
|
|
NSRect screenRect = [[NSScreen mainScreen] visibleFrame];
|
|
|
|
if (rect.origin.x < screenRect.origin.x) {
|
|
|
|
rect.origin.x = screenRect.origin.x;
|
|
|
|
}
|
|
|
|
if (rect.origin.y < screenRect.origin.y) {
|
|
|
|
rect.origin.y = screenRect.origin.y;
|
|
|
|
}
|
|
|
|
CGFloat xMostOfScreen = screenRect.origin.x + screenRect.size.width;
|
|
|
|
CGFloat yMostOfScreen = screenRect.origin.y + screenRect.size.height;
|
|
|
|
CGFloat xMost = rect.origin.x + rect.size.width;
|
|
|
|
CGFloat yMost = rect.origin.y + rect.size.height;
|
|
|
|
if (xMostOfScreen < xMost) {
|
|
|
|
rect.origin.x -= xMost - xMostOfScreen;
|
|
|
|
}
|
|
|
|
if (yMostOfScreen < yMost) {
|
|
|
|
rect.origin.y -= yMost - yMostOfScreen;
|
|
|
|
}
|
|
|
|
|
|
|
|
[self setFrame:rect display:[self isVisible]];
|
|
|
|
}
|
|
|
|
|
2010-06-10 04:56:17 +04:00
|
|
|
@end
|
2014-12-11 17:44:07 +03:00
|
|
|
|
|
|
|
class ComplexTextInputPanelPrivate : public ComplexTextInputPanel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ComplexTextInputPanelPrivate();
|
|
|
|
|
2015-02-20 19:37:02 +03:00
|
|
|
virtual void InterpretKeyEvent(void* aEvent, nsAString& aOutText);
|
2014-12-11 17:44:07 +03:00
|
|
|
virtual bool IsInComposition();
|
|
|
|
virtual void PlacePanel(int32_t x, int32_t y);
|
2015-02-20 19:37:02 +03:00
|
|
|
virtual void* GetInputContext() { return [mPanel inputContext]; }
|
|
|
|
virtual void CancelComposition() { [mPanel cancelComposition]; }
|
|
|
|
|
2014-12-11 17:44:07 +03:00
|
|
|
private:
|
|
|
|
~ComplexTextInputPanelPrivate();
|
|
|
|
ComplexTextInputPanelImpl* mPanel;
|
|
|
|
};
|
|
|
|
|
|
|
|
ComplexTextInputPanelPrivate::ComplexTextInputPanelPrivate()
|
|
|
|
{
|
|
|
|
mPanel = [[ComplexTextInputPanelImpl alloc] init];
|
|
|
|
}
|
|
|
|
|
|
|
|
ComplexTextInputPanelPrivate::~ComplexTextInputPanelPrivate()
|
|
|
|
{
|
|
|
|
[mPanel release];
|
|
|
|
}
|
|
|
|
|
|
|
|
ComplexTextInputPanel*
|
|
|
|
ComplexTextInputPanel::GetSharedComplexTextInputPanel()
|
|
|
|
{
|
|
|
|
static ComplexTextInputPanelPrivate *sComplexTextInputPanelPrivate;
|
|
|
|
if (!sComplexTextInputPanelPrivate) {
|
|
|
|
sComplexTextInputPanelPrivate = new ComplexTextInputPanelPrivate();
|
|
|
|
}
|
|
|
|
return sComplexTextInputPanelPrivate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-02-20 19:37:02 +03:00
|
|
|
ComplexTextInputPanelPrivate::InterpretKeyEvent(void* aEvent, nsAString& aOutText)
|
2014-12-11 17:44:07 +03:00
|
|
|
{
|
|
|
|
NSString* textString = nil;
|
2015-02-20 19:37:02 +03:00
|
|
|
[mPanel interpretKeyEvent:(NSEvent*)aEvent string:&textString];
|
2014-12-11 17:44:07 +03:00
|
|
|
|
|
|
|
if (textString) {
|
|
|
|
nsCocoaUtils::GetStringForNSString(textString, aOutText);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ComplexTextInputPanelPrivate::IsInComposition()
|
|
|
|
{
|
|
|
|
return !![mPanel inComposition];
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ComplexTextInputPanelPrivate::PlacePanel(int32_t x, int32_t y)
|
|
|
|
{
|
|
|
|
[mPanel adjustTo:NSMakePoint(x, y)];
|
|
|
|
}
|