X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/dbeddfb93d3479d03d8ec4c0121dfbe3bbcc422b..ee032c59df7ac4facdfdffce016768ab858e18c0:/src/osx/cocoa/textctrl.mm?ds=inline diff --git a/src/osx/cocoa/textctrl.mm b/src/osx/cocoa/textctrl.mm index 163f8a1fbc..ca728cf9db 100644 --- a/src/osx/cocoa/textctrl.mm +++ b/src/osx/cocoa/textctrl.mm @@ -47,90 +47,482 @@ #include "wx/thread.h" #include "wx/osx/private.h" -#include "wx/osx/carbon/private/mactext.h" +#include "wx/osx/cocoa/private/textimpl.h" +@interface wxNSSecureTextField : NSSecureTextField +{ +} +@end -@implementation wxNSTextField +@implementation wxNSSecureTextField + ++ (void)initialize +{ + static BOOL initialized = NO; + if (!initialized) + { + initialized = YES; + wxOSXCocoaClassAddWXMethods( self ); + } +} + +- (void)controlTextDidChange:(NSNotification *)aNotification +{ + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer(); + if ( wxpeer ) { + wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } + } +} + +- (void)controlTextDidEndEditing:(NSNotification *)aNotification +{ + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + impl->DoNotifyFocusEvent( false, NULL ); + } +} + +@end + +@interface wxNSTextScrollView : NSScrollView +{ +} +@end + +@interface wxNSTextView : NSTextView +{ + wxNSTextScrollView* scrollView; +} + +- (void)setScrollView: (wxNSTextScrollView *) sv; +- (wxNSTextScrollView*) scrollView; + +@end + +@implementation wxNSTextScrollView + ++ (void)initialize +{ + static BOOL initialized = NO; + if (!initialized) + { + initialized = YES; + wxOSXCocoaClassAddWXMethods( self ); + } +} + +- (void)textDidChange:(NSNotification *)aNotification +{ + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer(); + if ( wxpeer ) { + wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } + } +} + +- (BOOL)textView:(NSTextView *)aTextView doCommandBySelector:(SEL)commandSelector +{ + wxUnusedVar(aTextView); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer(); + if (commandSelector == @selector(insertNewline:)) + { + if ( wxpeer && wxpeer->GetWindowStyle() & wxTE_PROCESS_ENTER ) + { + wxCommandEvent event(wxEVT_COMMAND_TEXT_ENTER, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } + } + } + + return NO; +} -- (void)setImplementation: (wxWidgetImpl *) theImplementation +- (void)textDidEndEditing:(NSNotification *)aNotification { - impl = theImplementation; + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + impl->DoNotifyFocusEvent( false, NULL ); + } } +@end + +@implementation wxNSTextView -- (wxWidgetImpl*) implementation +- (BOOL) becomeFirstResponder { - return impl; + BOOL val = [super becomeFirstResponder]; + + if ( val ) + { + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( scrollView ); + if (impl ) + impl->DoNotifyFocusEvent( true, NULL ); + + } + return val; } -- (BOOL) isFlipped +- (void)setScrollView: (wxNSTextScrollView *) sv { - return YES; + scrollView = sv; } -// use our common calls -- (void) setTitle:(NSString *) title +- (wxNSTextScrollView*) scrollView { - [self setStringValue: title]; + return scrollView; } @end -class wxNSTextFieldControl : public wxMacTextControl +@implementation wxNSTextField + ++ (void)initialize { -public : - wxNSTextFieldControl( wxTextCtrl *wxPeer, WXWidget w ) : wxMacTextControl(wxPeer, w) + static BOOL initialized = NO; + if (!initialized) { + initialized = YES; + wxOSXCocoaClassAddWXMethods( self ); } - virtual ~wxNSTextFieldControl() +} + +- (void) setEnabled:(BOOL) flag +{ + [super setEnabled: flag]; + + if (![self drawsBackground]) { + // Static text is drawn incorrectly when disabled. + // For an explanation, see + // http://www.cocoabuilder.com/archive/message/cocoa/2006/7/21/168028 + if (flag) { + [self setTextColor: [NSColor controlTextColor]]; + } else { + [self setTextColor: [NSColor secondarySelectedControlColor]]; + } + } +} + +- (void)controlTextDidChange:(NSNotification *)aNotification +{ + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) { + wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer(); + if ( wxpeer ) { + wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } } +} + +typedef BOOL (*wxOSX_insertNewlineHandlerPtr)(NSView* self, SEL _cmd, NSControl *control, NSTextView* textView, SEL commandSelector); + +- (BOOL)control:(NSControl*)control textView:(NSTextView*)textView doCommandBySelector:(SEL)commandSelector +{ + wxUnusedVar(textView); + wxUnusedVar(control); + if (commandSelector == @selector(insertNewline:)) + { + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer(); + if ( wxpeer && wxpeer->GetWindowStyle() & wxTE_PROCESS_ENTER ) + { + wxCommandEvent event(wxEVT_COMMAND_TEXT_ENTER, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } + } + } + + return NO; +} + +- (void)controlTextDidEndEditing:(NSNotification *)aNotification +{ + wxUnusedVar(aNotification); + wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self ); + if ( impl ) + { + impl->DoNotifyFocusEvent( false, NULL ); + } +} +@end + +// wxNSTextViewControl + +wxNSTextViewControl::wxNSTextViewControl( wxTextCtrl *wxPeer, WXWidget w ) : wxWidgetCocoaImpl(wxPeer, w) +{ + wxNSTextScrollView* sv = (wxNSTextScrollView*) w; + m_scrollView = sv; + + [m_scrollView setHasVerticalScroller:YES]; + [m_scrollView setHasHorizontalScroller:NO]; + [m_scrollView setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable]; + NSSize contentSize = [m_scrollView contentSize]; + + wxNSTextView* tv = [[wxNSTextView alloc] initWithFrame: NSMakeRect(0, 0, + contentSize.width, contentSize.height)]; + m_textView = tv; + [tv setVerticallyResizable:YES]; + [tv setHorizontallyResizable:NO]; + [tv setAutoresizingMask:NSViewWidthSizable]; + + [m_scrollView setDocumentView: tv]; + + [tv setDelegate: w]; + [tv setScrollView:sv]; +} + +wxNSTextViewControl::~wxNSTextViewControl() +{ + if (m_textView) + [m_textView setDelegate: nil]; +} - virtual void VisibilityChanged(bool shown){} - virtual wxString GetStringValue() const +wxString wxNSTextViewControl::GetStringValue() const +{ + if (m_textView) { - wxCFStringRef cf( (CFStringRef) [[(wxNSTextField*) m_osxView stringValue] retain] ); + wxCFStringRef cf( (CFStringRef) [[m_textView string] retain] ); return cf.AsString(m_wxPeer->GetFont().GetEncoding()); } - virtual void SetStringValue( const wxString &str) + return wxEmptyString; +} +void wxNSTextViewControl::SetStringValue( const wxString &str) +{ + if (m_textView) + [m_textView setString: wxCFStringRef( str , m_wxPeer->GetFont().GetEncoding() ).AsNSString()]; +} +void wxNSTextViewControl::Copy() +{ + if (m_textView) + [m_textView copy:nil]; + +} + +void wxNSTextViewControl::Cut() +{ + if (m_textView) + [m_textView cut:nil]; +} + +void wxNSTextViewControl::Paste() +{ + if (m_textView) + [m_textView paste:nil]; +} + +bool wxNSTextViewControl::CanPaste() const +{ + return true; +} + +void wxNSTextViewControl::SetEditable(bool editable) +{ + if (m_textView) + [m_textView setEditable: editable]; +} + +void wxNSTextViewControl::GetSelection( long* from, long* to) const +{ + if (m_textView) + { + NSRange range = [m_textView selectedRange]; + *from = range.location; + *to = range.location + range.length; + } +} + +void wxNSTextViewControl::SetSelection( long from , long to ) +{ + NSRange selrange = NSMakeRange(from, to-from); + [m_textView setSelectedRange:selrange]; + [m_textView scrollRangeToVisible:selrange]; +} + +void wxNSTextViewControl::WriteText(const wxString& str) +{ + // temp hack to get logging working early + wxString former = GetStringValue(); + SetStringValue( former + str ); + SetSelection(GetStringValue().length(), GetStringValue().length()); +} + +// wxNSTextFieldControl + +wxNSTextFieldControl::wxNSTextFieldControl( wxTextCtrl *wxPeer, WXWidget w ) : wxWidgetCocoaImpl(wxPeer, w) +{ + m_textField = (NSTextField*) w; + [m_textField setDelegate: w]; +} + +wxNSTextFieldControl::~wxNSTextFieldControl() +{ + if (m_textField) + [m_textField setDelegate: nil]; +} + +wxString wxNSTextFieldControl::GetStringValue() const +{ + wxCFStringRef cf( (CFStringRef) [[m_textField stringValue] retain] ); + return cf.AsString(m_wxPeer->GetFont().GetEncoding()); +} +void wxNSTextFieldControl::SetStringValue( const wxString &str) +{ + [m_textField setStringValue: wxCFStringRef( str , m_wxPeer->GetFont().GetEncoding() ).AsNSString()]; +} +void wxNSTextFieldControl::Copy() +{ + NSText* editor = [m_textField currentEditor]; + if ( editor ) + { + [editor copy:nil]; + } +} + +void wxNSTextFieldControl::Cut() +{ + NSText* editor = [m_textField currentEditor]; + if ( editor ) + { + [editor cut:nil]; + } +} + +void wxNSTextFieldControl::Paste() +{ + NSText* editor = [m_textField currentEditor]; + if ( editor ) { - [(wxNSTextField*) m_osxView setStringValue: wxCFStringRef( str , m_wxPeer->GetFont().GetEncoding() ).AsNSString()]; + [editor paste:nil]; } - virtual void Copy() {} - virtual void Cut() {} - virtual void Paste() {} - virtual bool CanPaste() const { return false;} - virtual void SetEditable(bool editable) {} - virtual void GetSelection( long* from, long* to) const {} - virtual void SetSelection( long from , long to ){} - virtual void WriteText(const wxString& str) +} + +bool wxNSTextFieldControl::CanPaste() const +{ + return true; +} + +void wxNSTextFieldControl::SetEditable(bool editable) +{ + [m_textField setEditable:editable]; +} + +void wxNSTextFieldControl::GetSelection( long* from, long* to) const +{ + NSText* editor = [m_textField currentEditor]; + if ( editor ) { - // temp hack to get logging working early - wxString former = GetStringValue(); - SetStringValue( former + str ); + NSRange range = [editor selectedRange]; + *from = range.location; + *to = range.location + range.length; } -}; +} + +void wxNSTextFieldControl::SetSelection( long from , long to ) +{ + NSText* editor = [m_textField currentEditor]; + if ( editor ) + { + [editor setSelectedRange:NSMakeRange(from, to-from)]; + } +} + +void wxNSTextFieldControl::WriteText(const wxString& str) +{ + // temp hack to get logging working early + wxString former = GetStringValue(); + SetStringValue( former + str ); + SetSelection(GetStringValue().length(), GetStringValue().length()); +} + +void wxNSTextFieldControl::controlAction(WXWidget WXUNUSED(slf), + void* WXUNUSED(_cmd), void *WXUNUSED(sender)) +{ + wxWindow* wxpeer = (wxWindow*) GetWXPeer(); + if ( wxpeer && (wxpeer->GetWindowStyle() & wxTE_PROCESS_ENTER) ) + { + wxCommandEvent event(wxEVT_COMMAND_TEXT_ENTER, wxpeer->GetId()); + event.SetEventObject( wxpeer ); + event.SetString( static_cast(wxpeer)->GetValue() ); + wxpeer->HandleWindowEvent( event ); + } +} + +// +// +// wxWidgetImplType* wxWidgetImpl::CreateTextControl( wxTextCtrl* wxpeer, - wxWindowMac* parent, - wxWindowID id, + wxWindowMac* WXUNUSED(parent), + wxWindowID WXUNUSED(id), const wxString& str, const wxPoint& pos, const wxSize& size, long style, - long extraStyle) + long WXUNUSED(extraStyle)) { - NSView* sv = (wxpeer->GetParent()->GetHandle() ); - NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ; - wxNSTextField* v = [[wxNSTextField alloc] initWithFrame:r]; - [sv addSubview:v]; - - //[v setBezeled:NO]; - //[v setEditable:NO]; - //[v setDrawsBackground:NO]; + wxWidgetCocoaImpl* c = NULL; + + if ( style & wxTE_MULTILINE || style & wxTE_RICH || style & wxTE_RICH2 ) + { + wxNSTextScrollView* v = nil; + v = [[wxNSTextScrollView alloc] initWithFrame:r]; + c = new wxNSTextViewControl( wxpeer, v ); + static_cast(c)->SetStringValue(str); + } + else + { + NSTextField* v = nil; + if ( style & wxTE_PASSWORD ) + v = [[wxNSSecureTextField alloc] initWithFrame:r]; + else + v = [[wxNSTextField alloc] initWithFrame:r]; + + if ( style & wxNO_BORDER ) + { + // FIXME: How can we remove the native control's border? + // setBordered is separate from the text ctrl's border. + } + + [v setBezeled:NO]; + [v setBordered:NO]; + + c = new wxNSTextFieldControl( wxpeer, v ); + static_cast(c)->SetStringValue(str); + } - wxWidgetCocoaImpl* c = new wxNSTextFieldControl( wxpeer, v ); - [v setImplementation:c]; return c; }