2018-02-27 18:06:05 +01:00
|
|
|
/*
|
|
|
|
Copyright (C) 2011 by Mike McQuaid
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qsearchfield.h"
|
|
|
|
|
|
|
|
#include "qocoa_mac.h"
|
|
|
|
|
|
|
|
#import "Foundation/NSAutoreleasePool.h"
|
|
|
|
#import "Foundation/NSNotification.h"
|
|
|
|
#import "AppKit/NSSearchField.h"
|
|
|
|
|
|
|
|
#include <QApplication>
|
|
|
|
#include <QKeyEvent>
|
2018-07-01 22:26:46 +02:00
|
|
|
#include <QClipboard>
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
class QSearchFieldPrivate : public QObject {
|
2018-02-27 18:06:05 +01:00
|
|
|
public:
|
2018-09-05 18:15:28 +02:00
|
|
|
QSearchFieldPrivate(QSearchField *qSearchField, NSSearchField *nsSearchField)
|
|
|
|
: QObject(qSearchField), qSearchField(qSearchField), nsSearchField(nsSearchField) {}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void textDidChange(const QString &text) {
|
|
|
|
if (qSearchField) emit qSearchField->textChanged(text);
|
|
|
|
}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void textDidEndEditing() {
|
|
|
|
if (qSearchField)
|
|
|
|
emit qSearchField->editingFinished();
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void returnPressed() {
|
|
|
|
if (qSearchField) {
|
|
|
|
emit qSearchField->returnPressed();
|
|
|
|
QKeyEvent* event = new QKeyEvent(QEvent::KeyPress, Qt::Key_Return, Qt::NoModifier);
|
|
|
|
QApplication::postEvent(qSearchField, event);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
2018-09-05 18:15:28 +02:00
|
|
|
}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void keyDownPressed() {
|
|
|
|
if (qSearchField) {
|
|
|
|
QKeyEvent* event = new QKeyEvent(QEvent::KeyPress, Qt::Key_Down, Qt::NoModifier);
|
|
|
|
QApplication::postEvent(qSearchField, event);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
2018-09-05 18:15:28 +02:00
|
|
|
}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void keyUpPressed() {
|
|
|
|
if (qSearchField) {
|
|
|
|
QKeyEvent* event = new QKeyEvent(QEvent::KeyPress, Qt::Key_Up, Qt::NoModifier);
|
|
|
|
QApplication::postEvent(qSearchField, event);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
2018-09-05 18:15:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QPointer<QSearchField> qSearchField;
|
|
|
|
NSSearchField *nsSearchField;
|
2018-02-27 18:06:05 +01:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
@interface QSearchFieldDelegate : NSObject<NSTextFieldDelegate> {
|
2018-02-27 18:06:05 +01:00
|
|
|
@public
|
2018-09-05 18:15:28 +02:00
|
|
|
QPointer<QSearchFieldPrivate> pimpl;
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
-(void)controlTextDidChange:(NSNotification*)notification;
|
|
|
|
-(void)controlTextDidEndEditing:(NSNotification*)notification;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation QSearchFieldDelegate
|
|
|
|
-(void)controlTextDidChange:(NSNotification*)notification {
|
2018-09-05 18:15:28 +02:00
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (pimpl) pimpl->textDidChange(toQString([[notification object] stringValue]));
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-07-01 22:26:46 +02:00
|
|
|
-(void)controlTextDidEndEditing:(NSNotification*)notification {
|
2018-09-05 18:15:28 +02:00
|
|
|
Q_UNUSED(notification);
|
|
|
|
// No Q_ASSERT here as it is called on destruction.
|
|
|
|
if (!pimpl) return;
|
|
|
|
pimpl->textDidEndEditing();
|
|
|
|
if ([[[notification userInfo] objectForKey:@"NSTextMovement"] intValue] == NSReturnTextMovement)
|
|
|
|
pimpl->returnPressed();
|
2018-07-01 22:26:46 +02:00
|
|
|
}
|
|
|
|
|
2018-02-27 18:06:05 +01:00
|
|
|
-(BOOL)control: (NSControl *)control textView:
|
2018-09-05 18:15:28 +02:00
|
|
|
(NSTextView *)textView doCommandBySelector:
|
|
|
|
(SEL)commandSelector {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return NO;
|
|
|
|
if (commandSelector == @selector(moveDown:)) {
|
|
|
|
pimpl->keyDownPressed();
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
else if (commandSelector == @selector(moveUp:)) {
|
|
|
|
pimpl->keyUpPressed();
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
return NO;
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface QocoaSearchField : NSSearchField
|
|
|
|
-(BOOL)performKeyEquivalent:(NSEvent*)event;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation QocoaSearchField
|
|
|
|
-(BOOL)performKeyEquivalent:(NSEvent*)event {
|
2018-09-05 18:15:28 +02:00
|
|
|
// First, check if we have the focus.
|
|
|
|
// If no, it probably means this event isn't for us.
|
|
|
|
NSResponder* firstResponder = [[NSApp keyWindow] firstResponder];
|
|
|
|
if ([firstResponder isKindOfClass:[NSText class]] && [(NSText*)firstResponder delegate] == self) {
|
|
|
|
|
|
|
|
if ([event type] == NSEventTypeKeyDown && [event modifierFlags] & NSEventModifierFlagCommand) {
|
|
|
|
QString keyString = toQString([event characters]);
|
|
|
|
if (keyString == "a") // Cmd+a
|
|
|
|
{
|
|
|
|
[self performSelector:@selector(selectText:)];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
else if (keyString == "c") // Cmd+c
|
|
|
|
{
|
|
|
|
[[self currentEditor] copy: nil];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
else if (keyString == "v") // Cmd+v
|
|
|
|
{
|
|
|
|
[[self currentEditor] paste: nil];
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
else if (keyString == "x") // Cmd+x
|
|
|
|
{
|
|
|
|
[[self currentEditor] cut: nil];
|
|
|
|
return YES;
|
|
|
|
}
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
2018-09-05 18:15:28 +02:00
|
|
|
}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
return NO;
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
QSearchField::QSearchField(QWidget *parent) : QWidget(parent) {
|
|
|
|
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
|
|
|
|
NSSearchField *search = [[QocoaSearchField alloc] init];
|
|
|
|
QSearchFieldDelegate *delegate = [[QSearchFieldDelegate alloc] init];
|
|
|
|
pimpl = delegate->pimpl = new QSearchFieldPrivate(this, search);
|
|
|
|
[search setDelegate:delegate];
|
|
|
|
setupLayout(search, this);
|
|
|
|
setFixedHeight(24);
|
|
|
|
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
|
|
|
|
[search release];
|
|
|
|
[pool drain];
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::setText(const QString &text) {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return;
|
2018-07-01 22:26:46 +02:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
|
|
|
|
[pimpl->nsSearchField setStringValue:fromQString(text)];
|
|
|
|
if (!text.isEmpty()) {
|
|
|
|
[pimpl->nsSearchField selectText:pimpl->nsSearchField];
|
|
|
|
[[pimpl->nsSearchField currentEditor] setSelectedRange:NSMakeRange([[pimpl->nsSearchField stringValue] length], 0)];
|
|
|
|
}
|
|
|
|
[pool drain];
|
2018-07-01 22:26:46 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::setPlaceholderText(const QString &text) {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return;
|
|
|
|
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
|
|
|
|
[[pimpl->nsSearchField cell] setPlaceholderString:fromQString(text)];
|
|
|
|
[pool drain];
|
2018-07-01 22:26:46 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::clear() {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return;
|
|
|
|
[pimpl->nsSearchField setStringValue:@""];
|
|
|
|
emit textChanged(QString());
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::selectAll() {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return;
|
|
|
|
[pimpl->nsSearchField performSelector:@selector(selectText:)];
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
QString QSearchField::text() const {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
if (!pimpl) return QString();
|
|
|
|
return toQString([pimpl->nsSearchField stringValue]);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
QString QSearchField::placeholderText() const {
|
|
|
|
Q_ASSERT(pimpl);
|
|
|
|
return toQString([[pimpl->nsSearchField cell] placeholderString]);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::setFocus(Qt::FocusReason) {}
|
2018-02-27 18:06:05 +01:00
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::setFocus() {
|
|
|
|
setFocus(Qt::OtherFocusReason);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
void QSearchField::resizeEvent(QResizeEvent *resizeEvent) {
|
|
|
|
QWidget::resizeEvent(resizeEvent);
|
2018-02-27 18:06:05 +01:00
|
|
|
}
|
|
|
|
|
2018-09-05 18:15:28 +02:00
|
|
|
bool QSearchField::eventFilter(QObject *o, QEvent *e) {
|
|
|
|
return QWidget::eventFilter(o, e);
|
2018-07-01 22:26:46 +02:00
|
|
|
}
|
|
|
|
|