/*n*********************************************************************** * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_vcl.hxx" #include "salinst.h" #include "salgdi.h" #include "salframe.h" #include "salframeview.h" #include "aqua11yfactory.h" #include #include "vcl/window.hxx" #include "vcl/svapp.hxx" #define WHEEL_EVENT_FACTOR 1.5 static USHORT ImplGetModifierMask( unsigned int nMask ) { USHORT nRet = 0; if( (nMask & NSShiftKeyMask) != 0 ) nRet |= KEY_SHIFT; if( (nMask & NSControlKeyMask) != 0 ) nRet |= KEY_MOD3; if( (nMask & NSAlternateKeyMask) != 0 ) nRet |= KEY_MOD2; if( (nMask & NSCommandKeyMask) != 0 ) nRet |= KEY_MOD1; return nRet; } static USHORT ImplMapCharCode( sal_Unicode aCode ) { static USHORT aKeyCodeMap[ 128 ] = { 0, 0, 0, 0, 0, 0, 0, 0, KEY_BACKSPACE, KEY_TAB, KEY_RETURN, 0, 0, KEY_RETURN, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, KEY_TAB, 0, KEY_ESCAPE, 0, 0, 0, 0, KEY_SPACE, 0, 0, 0, 0, 0, 0, 0, 0, 0, KEY_MULTIPLY, KEY_ADD, KEY_COMMA, KEY_SUBTRACT, KEY_POINT, KEY_DIVIDE, KEY_0, KEY_1, KEY_2, KEY_3, KEY_4, KEY_5, KEY_6, KEY_7, KEY_8, KEY_9, 0, 0, KEY_LESS, KEY_EQUAL, KEY_GREATER, 0, 0, KEY_A, KEY_B, KEY_C, KEY_D, KEY_E, KEY_F, KEY_G, KEY_H, KEY_I, KEY_J, KEY_K, KEY_L, KEY_M, KEY_N, KEY_O, KEY_P, KEY_Q, KEY_R, KEY_S, KEY_T, KEY_U, KEY_V, KEY_W, KEY_X, KEY_Y, KEY_Z, 0, 0, 0, 0, 0, KEY_QUOTELEFT, KEY_A, KEY_B, KEY_C, KEY_D, KEY_E, KEY_F, KEY_G, KEY_H, KEY_I, KEY_J, KEY_K, KEY_L, KEY_M, KEY_N, KEY_O, KEY_P, KEY_Q, KEY_R, KEY_S, KEY_T, KEY_U, KEY_V, KEY_W, KEY_X, KEY_Y, KEY_Z, 0, 0, 0, KEY_TILDE, KEY_BACKSPACE }; // Note: the mapping 0x7f should by rights be KEY_DELETE // however if you press "backspace" 0x7f is reported // whereas for "delete" 0xf728 gets reported // Note: the mapping of 0x19 to KEY_TAB is because for unknown reasons // tab alone is reported as 0x09 (as expected) but shift-tab is // reported as 0x19 (end of medium) static USHORT aFunctionKeyCodeMap[ 128 ] = { KEY_UP, KEY_DOWN, KEY_LEFT, KEY_RIGHT, KEY_F1, KEY_F2, KEY_F3, KEY_F4, KEY_F5, KEY_F6, KEY_F7, KEY_F8, KEY_F9, KEY_F10, KEY_F11, KEY_F12, KEY_F13, KEY_F14, KEY_F15, KEY_F16, KEY_F17, KEY_F18, KEY_F19, KEY_F20, KEY_F21, KEY_F22, KEY_F23, KEY_F24, KEY_F25, KEY_F26, 0, 0, 0, 0, 0, 0, 0, 0, 0, KEY_INSERT, KEY_DELETE, KEY_HOME, 0, KEY_END, KEY_PAGEUP, KEY_PAGEDOWN, 0, 0, 0, 0, 0, 0, 0, KEY_MENU, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, KEY_UNDO, KEY_REPEAT, KEY_FIND, KEY_HELP, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; USHORT nKeyCode = 0; if( aCode < sizeof( aKeyCodeMap) / sizeof( aKeyCodeMap[0] ) ) nKeyCode = aKeyCodeMap[ aCode ]; else if( aCode >= 0xf700 && aCode < 0xf780 ) nKeyCode = aFunctionKeyCodeMap[ aCode - 0xf700 ]; return nKeyCode; } // store the frame the mouse last entered static AquaSalFrame* s_pMouseFrame = NULL; // store the last pressed button for enter/exit events // which lack that information static USHORT s_nLastButton = 0; // combinations of keys we need to handle ourselves static const struct ExceptionalKey { const USHORT nKeyCode; const unsigned int nModifierMask; } aExceptionalKeys[] = { { KEY_D, NSControlKeyMask | NSShiftKeyMask | NSAlternateKeyMask }, { KEY_D, NSCommandKeyMask | NSShiftKeyMask | NSAlternateKeyMask } }; static AquaSalFrame* getMouseContainerFrame() { int nWindows = 0; NSCountWindows( &nWindows ); int* pWindows = (int*)alloca( nWindows * sizeof(int) ); // note: NSWindowList is supposed to be in z-order front to back NSWindowList( nWindows, pWindows ); AquaSalFrame* pDispatchFrame = NULL; for(int i = 0; i < nWindows && ! pDispatchFrame; i++ ) { NSWindow* pWin = [NSApp windowWithWindowNumber: pWindows[i]]; if( pWin && [pWin isMemberOfClass: [SalFrameWindow class]] && [(SalFrameWindow*)pWin containsMouse] ) pDispatchFrame = [(SalFrameWindow*)pWin getSalFrame]; } return pDispatchFrame; } @implementation SalFrameWindow -(id)initWithSalFrame: (AquaSalFrame*)pFrame { mDraggingDestinationHandler = nil; mpFrame = pFrame; NSRect aRect = { { pFrame->maGeometry.nX, pFrame->maGeometry.nY }, { pFrame->maGeometry.nWidth, pFrame->maGeometry.nHeight } }; pFrame->VCLToCocoa( aRect ); return [super initWithContentRect: aRect styleMask: mpFrame->getStyleMask() backing: NSBackingStoreBuffered defer: NO ]; } -(AquaSalFrame*)getSalFrame { return mpFrame; } -(MacOSBOOL)containsMouse { // is this event actually inside that NSWindow ? NSPoint aPt = [NSEvent mouseLocation]; NSRect aFrameRect = [self frame]; MacOSBOOL bInRect = NSPointInRect( aPt, aFrameRect ); return bInRect; } -(MacOSBOOL)canBecomeKeyWindow { if( (mpFrame->mnStyle & ( SAL_FRAME_STYLE_FLOAT | SAL_FRAME_STYLE_TOOLTIP | SAL_FRAME_STYLE_INTRO )) == 0 ) return YES; if( (mpFrame->mnStyle & SAL_FRAME_STYLE_OWNERDRAWDECORATION) != 0 ) return YES; if( mpFrame->mbFullScreen ) return YES; if( (mpFrame->mnStyle & SAL_FRAME_STYLE_FLOAT_FOCUSABLE) ) return YES; return [super canBecomeKeyWindow]; } -(void)windowDidBecomeKey: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { static const ULONG nGuessDocument = SAL_FRAME_STYLE_MOVEABLE| SAL_FRAME_STYLE_SIZEABLE| SAL_FRAME_STYLE_CLOSEABLE; if( mpFrame->mpMenu ) mpFrame->mpMenu->setMainMenu(); else if( ! mpFrame->mpParent && ( (mpFrame->mnStyle & nGuessDocument) == nGuessDocument || // set default menu for e.g. help mpFrame->mbFullScreen ) ) // ser default menu for e.g. presentation { AquaSalMenu::setDefaultMenu(); } #if 0 // FIXME: we should disable menus while in modal mode // however from down here there is currently no reliable way to // find out when to do this if( (mpFrame->mpParent && mpFrame->mpParent->GetWindow()->IsInModalMode()) ) AquaSalMenu::enableMainMenu( false ); #endif mpFrame->CallCallback( SALEVENT_GETFOCUS, 0 ); mpFrame->SendPaintEvent(); // repaint controls as active } } -(void)windowDidResignKey: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->CallCallback(SALEVENT_LOSEFOCUS, 0); mpFrame->SendPaintEvent(); // repaint controls as inactive } } -(void)windowDidChangeScreen: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) mpFrame->screenParametersChanged(); } -(void)windowDidMove: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->UpdateFrameGeometry(); mpFrame->CallCallback( SALEVENT_MOVE, 0 ); } } -(void)windowDidResize: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->UpdateFrameGeometry(); mpFrame->CallCallback( SALEVENT_RESIZE, 0 ); mpFrame->SendPaintEvent(); } } -(void)windowDidMiniaturize: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mbShown = false; mpFrame->UpdateFrameGeometry(); mpFrame->CallCallback( SALEVENT_RESIZE, 0 ); } } -(void)windowDidDeminiaturize: (NSNotification*)pNotification { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mbShown = true; mpFrame->UpdateFrameGeometry(); mpFrame->CallCallback( SALEVENT_RESIZE, 0 ); } } -(MacOSBOOL)windowShouldClose: (NSNotification*)pNotification { YIELD_GUARD; MacOSBOOL bRet = YES; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { // #i84461# end possible input mpFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, 0 ); if( AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->CallCallback( SALEVENT_CLOSE, 0 ); bRet = NO; // application will close the window or not, AppKit shouldn't } } return bRet; } -(void)dockMenuItemTriggered: (id)sender { YIELD_GUARD; if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) mpFrame->ToTop( SAL_FRAME_TOTOP_RESTOREWHENMIN | SAL_FRAME_TOTOP_GRABFOCUS ); } -(::com::sun::star::uno::Reference < ::com::sun::star::accessibility::XAccessibleContext >)accessibleContext { return mpFrame -> GetWindow() -> GetAccessible() -> getAccessibleContext(); } -(NSDragOperation)draggingEntered:(id )sender { return [mDraggingDestinationHandler draggingEntered: sender]; } -(NSDragOperation)draggingUpdated:(id )sender { return [mDraggingDestinationHandler draggingUpdated: sender]; } -(void)draggingExited:(id )sender { [mDraggingDestinationHandler draggingExited: sender]; } -(MacOSBOOL)prepareForDragOperation:(id )sender { return [mDraggingDestinationHandler prepareForDragOperation: sender]; } -(MacOSBOOL)performDragOperation:(id )sender { return [mDraggingDestinationHandler performDragOperation: sender]; } -(void)concludeDragOperation:(id )sender { [mDraggingDestinationHandler concludeDragOperation: sender]; } -(void)registerDraggingDestinationHandler:(id)theHandler { mDraggingDestinationHandler = theHandler; } -(void)unregisterDraggingDestinationHandler:(id)theHandler { mDraggingDestinationHandler = nil; } @end @implementation SalFrameView +(void)unsetMouseFrame: (AquaSalFrame*)pFrame { if( pFrame == s_pMouseFrame ) s_pMouseFrame = NULL; } -(id)initWithSalFrame: (AquaSalFrame*)pFrame { if ((self = [super initWithFrame: [NSWindow contentRectForFrameRect: [pFrame->getWindow() frame] styleMask: pFrame->mnStyleMask]]) != nil) { mDraggingDestinationHandler = nil; mpFrame = pFrame; mMarkedRange = NSMakeRange(NSNotFound, 0); mSelectedRange = NSMakeRange(NSNotFound, 0); mpReferenceWrapper = nil; mpMouseEventListener = nil; mpLastSuperEvent = nil; } return self; } -(void)resetCursorRects { if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { // FIXME: does this leak the returned NSCursor of getCurrentCursor ? NSRect aRect = { { 0, 0 }, { mpFrame->maGeometry.nWidth, mpFrame->maGeometry.nHeight } }; [self addCursorRect: aRect cursor: mpFrame->getCurrentCursor()]; } } -(MacOSBOOL)acceptsFirstResponder { return YES; } -(MacOSBOOL)acceptsFirstMouse: (NSEvent*)pEvent { return YES; } -(MacOSBOOL)isOpaque { return mpFrame ? (mpFrame->getClipPath() != 0 ? NO : YES) : YES; } // helper class similar to a vos::OGuard for the SalYieldMutex // the difference is that it only does tryToAcquire instead of aquire // so dreaded deadlocks like #i93512# are prevented class TryGuard { public: TryGuard() { mbGuarded = ImplSalYieldMutexTryToAcquire(); } ~TryGuard() { if( mbGuarded ) ImplSalYieldMutexRelease(); } bool IsGuarded() { return mbGuarded; } private: bool mbGuarded; }; -(void)drawRect: (NSRect)aRect { // HOTFIX: #i93512# prevent deadlocks if any other thread already has the SalYieldMutex TryGuard aTryGuard; if( !aTryGuard.IsGuarded() ) { // NOTE: the mpFrame access below is not guarded yet! // TODO: mpFrame et al need to be guarded by an independent mutex AquaSalGraphics* pGraphics = (mpFrame && AquaSalFrame::isAlive(mpFrame)) ? mpFrame->mpGraphics : NULL; if( pGraphics ) { // we did not get the mutex so we cannot draw now => request to redraw later // convert the NSRect to a CGRect for Refreshrect() const CGRect aCGRect = {{aRect.origin.x,aRect.origin.y},{aRect.size.width,aRect.size.height}}; pGraphics->RefreshRect( aCGRect ); } return; } if( mpFrame && AquaSalFrame::isAlive( mpFrame ) ) { if( mpFrame->mpGraphics ) { mpFrame->mpGraphics->UpdateWindow( aRect ); if( mpFrame->getClipPath() ) [mpFrame->getWindow() invalidateShadow]; } } } -(void)sendMouseEventToFrame: (NSEvent*)pEvent button:(USHORT)nButton eventtype:(USHORT)nEvent { YIELD_GUARD; AquaSalFrame* pDispatchFrame = AquaSalFrame::GetCaptureFrame(); bool bIsCaptured = false; if( pDispatchFrame ) { bIsCaptured = true; if( nEvent == SALEVENT_MOUSELEAVE ) // no leave events if mouse is captured nEvent = SALEVENT_MOUSEMOVE; } else if( s_pMouseFrame ) pDispatchFrame = s_pMouseFrame; else pDispatchFrame = mpFrame; /* #i81645# Cocoa reports mouse events while a button is pressed to the window in which it was first pressed. This is reasonable and fine and gets one around most cases where on other platforms one uses CaptureMouse or XGrabPointer, however vcl expects mouse events to occur in the window the mouse is over, unless the mouse is explicitly captured. So we need to find the window the mouse is actually over for conformance with other platforms. */ if( ! bIsCaptured && nButton && pDispatchFrame && AquaSalFrame::isAlive( pDispatchFrame ) ) { // is this event actually inside that NSWindow ? NSPoint aPt = [NSEvent mouseLocation]; NSRect aFrameRect = [pDispatchFrame->getWindow() frame]; if ( ! NSPointInRect( aPt, aFrameRect ) ) { // no, it is not // now we need to find the one it may be in /* #i93756# we ant to get enumerate the application windows in z-order to check if any contains the mouse. This could be elegantly done with this code: // use NSApp to check windows in ZOrder whether they contain the mouse pointer NSWindow* pWindow = [NSApp makeWindowsPerform: @selector(containsMouse) inOrder: YES]; if( pWindow && [pWindow isMemberOfClass: [SalFrameWindow class]] ) pDispatchFrame = [(SalFrameWindow*)pWindow getSalFrame]; However if a non SalFrameWindow is on screen (like e.g. the file dialog) it can be hit with the containsMouse selector, which it doesn't support. Sadly NSApplication:makeWindowsPerform does not check (for performance reasons I assume) whether a window supports a selector before sending it. */ AquaSalFrame* pMouseFrame = getMouseContainerFrame(); if( pMouseFrame ) pDispatchFrame = pMouseFrame; } } if( pDispatchFrame && AquaSalFrame::isAlive( pDispatchFrame ) ) { pDispatchFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); pDispatchFrame->mnLastModifierFlags = [pEvent modifierFlags]; NSPoint aPt = [NSEvent mouseLocation]; pDispatchFrame->CocoaToVCL( aPt ); USHORT nModMask = ImplGetModifierMask( [pEvent modifierFlags] ); // #i82284# emulate ctrl left if( nModMask == KEY_MOD3 && nButton == MOUSE_LEFT ) { nModMask = 0; nButton = MOUSE_RIGHT; } SalMouseEvent aEvent; aEvent.mnTime = pDispatchFrame->mnLastEventTime; aEvent.mnX = static_cast(aPt.x) - pDispatchFrame->maGeometry.nX; aEvent.mnY = static_cast(aPt.y) - pDispatchFrame->maGeometry.nY; aEvent.mnButton = nButton; aEvent.mnCode = aEvent.mnButton | nModMask; // --- RTL --- (mirror mouse pos) if( Application::GetSettings().GetLayoutRTL() ) aEvent.mnX = pDispatchFrame->maGeometry.nWidth-1-aEvent.mnX; pDispatchFrame->CallCallback( nEvent, &aEvent ); } } -(void)mouseDown: (NSEvent*)pEvent { if ( mpMouseEventListener != nil && [mpMouseEventListener respondsToSelector: @selector(mouseDown:)]) { [mpMouseEventListener mouseDown: [pEvent copyWithZone: NULL]]; } s_nLastButton = MOUSE_LEFT; [self sendMouseEventToFrame:pEvent button:MOUSE_LEFT eventtype:SALEVENT_MOUSEBUTTONDOWN]; } -(void)mouseDragged: (NSEvent*)pEvent { if ( mpMouseEventListener != nil && [mpMouseEventListener respondsToSelector: @selector(mouseDragged:)]) { [mpMouseEventListener mouseDragged: [pEvent copyWithZone: NULL]]; } s_nLastButton = MOUSE_LEFT; [self sendMouseEventToFrame:pEvent button:MOUSE_LEFT eventtype:SALEVENT_MOUSEMOVE]; } -(void)mouseUp: (NSEvent*)pEvent { s_nLastButton = 0; [self sendMouseEventToFrame:pEvent button:MOUSE_LEFT eventtype:SALEVENT_MOUSEBUTTONUP]; } -(void)mouseMoved: (NSEvent*)pEvent { s_nLastButton = 0; [self sendMouseEventToFrame:pEvent button:0 eventtype:SALEVENT_MOUSEMOVE]; } -(void)mouseEntered: (NSEvent*)pEvent { s_pMouseFrame = mpFrame; [self sendMouseEventToFrame:pEvent button:s_nLastButton eventtype:SALEVENT_MOUSEMOVE]; } -(void)mouseExited: (NSEvent*)pEvent { if( s_pMouseFrame == mpFrame ) s_pMouseFrame = NULL; [self sendMouseEventToFrame:pEvent button:s_nLastButton eventtype:SALEVENT_MOUSELEAVE]; } -(void)rightMouseDown: (NSEvent*)pEvent { s_nLastButton = MOUSE_RIGHT; [self sendMouseEventToFrame:pEvent button:MOUSE_RIGHT eventtype:SALEVENT_MOUSEBUTTONDOWN]; } -(void)rightMouseDragged: (NSEvent*)pEvent { s_nLastButton = MOUSE_RIGHT; [self sendMouseEventToFrame:pEvent button:MOUSE_RIGHT eventtype:SALEVENT_MOUSEMOVE]; } -(void)rightMouseUp: (NSEvent*)pEvent { s_nLastButton = 0; [self sendMouseEventToFrame:pEvent button:MOUSE_RIGHT eventtype:SALEVENT_MOUSEBUTTONUP]; } -(void)otherMouseDown: (NSEvent*)pEvent { if( [pEvent buttonNumber] == 2 ) { s_nLastButton = MOUSE_MIDDLE; [self sendMouseEventToFrame:pEvent button:MOUSE_MIDDLE eventtype:SALEVENT_MOUSEBUTTONDOWN]; } else s_nLastButton = 0; } -(void)otherMouseDragged: (NSEvent*)pEvent { if( [pEvent buttonNumber] == 2 ) { s_nLastButton = MOUSE_MIDDLE; [self sendMouseEventToFrame:pEvent button:MOUSE_MIDDLE eventtype:SALEVENT_MOUSEMOVE]; } else s_nLastButton = 0; } -(void)otherMouseUp: (NSEvent*)pEvent { s_nLastButton = 0; if( [pEvent buttonNumber] == 2 ) [self sendMouseEventToFrame:pEvent button:MOUSE_MIDDLE eventtype:SALEVENT_MOUSEBUTTONUP]; } - (void)magnifyWithEvent: (NSEvent*)pEvent { YIELD_GUARD; // TODO: ?? -(float)magnification; if( AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); mpFrame->mnLastModifierFlags = [pEvent modifierFlags]; float dZ = 0.0; for(;;) { dZ += [pEvent deltaZ]; NSEvent* pNextEvent = [NSApp nextEventMatchingMask: NSScrollWheelMask untilDate: nil inMode: NSDefaultRunLoopMode dequeue: YES ]; if( !pNextEvent ) break; pEvent = pNextEvent; } NSPoint aPt = [NSEvent mouseLocation]; mpFrame->CocoaToVCL( aPt ); SalWheelMouseEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.mnX = static_cast(aPt.x) - mpFrame->maGeometry.nX; aEvent.mnY = static_cast(aPt.y) - mpFrame->maGeometry.nY; aEvent.mnCode = ImplGetModifierMask( mpFrame->mnLastModifierFlags ); aEvent.mnCode |= KEY_MOD1; // we want zooming, no scrolling aEvent.mbDeltaIsPixel = TRUE; // --- RTL --- (mirror mouse pos) if( Application::GetSettings().GetLayoutRTL() ) aEvent.mnX = mpFrame->maGeometry.nWidth-1-aEvent.mnX; if( dZ != 0.0 ) { aEvent.mnDelta = static_cast(floor(dZ)); aEvent.mnNotchDelta = dZ < 0 ? -1 : 1; if( aEvent.mnDelta == 0 ) aEvent.mnDelta = aEvent.mnNotchDelta; aEvent.mbHorz = FALSE; aEvent.mnScrollLines = dZ > 0 ? dZ/WHEEL_EVENT_FACTOR : -dZ/WHEEL_EVENT_FACTOR; if( aEvent.mnScrollLines == 0 ) aEvent.mnScrollLines = 1; mpFrame->CallCallback( SALEVENT_WHEELMOUSE, &aEvent ); } } } - (void)rotateWithEvent: (NSEvent*)pEvent { //Rotation : -(float)rotation; // TODO: create new CommandType so rotation is available to the applications } - (void)swipeWithEvent: (NSEvent*)pEvent { YIELD_GUARD; if( AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); mpFrame->mnLastModifierFlags = [pEvent modifierFlags]; // merge pending scroll wheel events float dX = 0.0; float dY = 0.0; for(;;) { dX += [pEvent deltaX]; dY += [pEvent deltaY]; NSEvent* pNextEvent = [NSApp nextEventMatchingMask: NSScrollWheelMask untilDate: nil inMode: NSDefaultRunLoopMode dequeue: YES ]; if( !pNextEvent ) break; pEvent = pNextEvent; } NSPoint aPt = [NSEvent mouseLocation]; mpFrame->CocoaToVCL( aPt ); SalWheelMouseEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.mnX = static_cast(aPt.x) - mpFrame->maGeometry.nX; aEvent.mnY = static_cast(aPt.y) - mpFrame->maGeometry.nY; aEvent.mnCode = ImplGetModifierMask( mpFrame->mnLastModifierFlags ); aEvent.mbDeltaIsPixel = TRUE; // --- RTL --- (mirror mouse pos) if( Application::GetSettings().GetLayoutRTL() ) aEvent.mnX = mpFrame->maGeometry.nWidth-1-aEvent.mnX; if( dX != 0.0 ) { aEvent.mnDelta = static_cast(floor(dX)); aEvent.mnNotchDelta = dX < 0 ? -1 : 1; if( aEvent.mnDelta == 0 ) aEvent.mnDelta = aEvent.mnNotchDelta; aEvent.mbHorz = TRUE; aEvent.mnScrollLines = SAL_WHEELMOUSE_EVENT_PAGESCROLL; mpFrame->CallCallback( SALEVENT_WHEELMOUSE, &aEvent ); } if( dY != 0.0 && AquaSalFrame::isAlive( mpFrame )) { aEvent.mnDelta = static_cast(floor(dY)); aEvent.mnNotchDelta = dY < 0 ? -1 : 1; if( aEvent.mnDelta == 0 ) aEvent.mnDelta = aEvent.mnNotchDelta; aEvent.mbHorz = FALSE; aEvent.mnScrollLines = SAL_WHEELMOUSE_EVENT_PAGESCROLL; mpFrame->CallCallback( SALEVENT_WHEELMOUSE, &aEvent ); } } } -(void)scrollWheel: (NSEvent*)pEvent { YIELD_GUARD; if( AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); mpFrame->mnLastModifierFlags = [pEvent modifierFlags]; // merge pending scroll wheel events float dX = 0.0; float dY = 0.0; for(;;) { dX += [pEvent deltaX]; dY += [pEvent deltaY]; NSEvent* pNextEvent = [NSApp nextEventMatchingMask: NSScrollWheelMask untilDate: nil inMode: NSDefaultRunLoopMode dequeue: YES ]; if( !pNextEvent ) break; pEvent = pNextEvent; } NSPoint aPt = [NSEvent mouseLocation]; mpFrame->CocoaToVCL( aPt ); SalWheelMouseEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.mnX = static_cast(aPt.x) - mpFrame->maGeometry.nX; aEvent.mnY = static_cast(aPt.y) - mpFrame->maGeometry.nY; aEvent.mnCode = ImplGetModifierMask( mpFrame->mnLastModifierFlags ); aEvent.mbDeltaIsPixel = TRUE; // --- RTL --- (mirror mouse pos) if( Application::GetSettings().GetLayoutRTL() ) aEvent.mnX = mpFrame->maGeometry.nWidth-1-aEvent.mnX; if( dX != 0.0 ) { aEvent.mnDelta = static_cast(floor(dX)); aEvent.mnNotchDelta = dX < 0 ? -1 : 1; if( aEvent.mnDelta == 0 ) aEvent.mnDelta = aEvent.mnNotchDelta; aEvent.mbHorz = TRUE; aEvent.mnScrollLines = dY > 0 ? dX/WHEEL_EVENT_FACTOR : -dX/WHEEL_EVENT_FACTOR; if( aEvent.mnScrollLines == 0 ) aEvent.mnScrollLines = 1; mpFrame->CallCallback( SALEVENT_WHEELMOUSE, &aEvent ); } if( dY != 0.0 && AquaSalFrame::isAlive( mpFrame ) ) { aEvent.mnDelta = static_cast(floor(dY)); aEvent.mnNotchDelta = dY < 0 ? -1 : 1; if( aEvent.mnDelta == 0 ) aEvent.mnDelta = aEvent.mnNotchDelta; aEvent.mbHorz = FALSE; aEvent.mnScrollLines = dY > 0 ? dY/WHEEL_EVENT_FACTOR : -dY/WHEEL_EVENT_FACTOR; if( aEvent.mnScrollLines < 1 ) aEvent.mnScrollLines = 1; mpFrame->CallCallback( SALEVENT_WHEELMOUSE, &aEvent ); } } } -(void)keyDown: (NSEvent*)pEvent { YIELD_GUARD; if( AquaSalFrame::isAlive( mpFrame ) ) { mpLastEvent = pEvent; mbInKeyInput = true; mbNeedSpecialKeyHandle = false; mbKeyHandled = false; mpFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); mpFrame->mnLastModifierFlags = [pEvent modifierFlags]; if( ! [self handleKeyDownException: pEvent] ) { NSArray* pArray = [NSArray arrayWithObject: pEvent]; [self interpretKeyEvents: pArray]; } mbInKeyInput = false; } } -(MacOSBOOL)handleKeyDownException:(NSEvent*)pEvent { // check for a very special set of modified characters NSString* pUnmodifiedString = [pEvent charactersIgnoringModifiers]; if( pUnmodifiedString && [pUnmodifiedString length] == 1 ) { /* #i103102# key events with command and alternate don't make it through interpretKeyEvents (why ?). Try to dispatch them here first, if not successful continue normally */ if( (mpFrame->mnLastModifierFlags & (NSAlternateKeyMask | NSCommandKeyMask)) == (NSAlternateKeyMask | NSCommandKeyMask) ) { if( [self sendSingleCharacter: mpLastEvent] ) return YES; } unichar keyChar = [pUnmodifiedString characterAtIndex: 0]; USHORT nKeyCode = ImplMapCharCode( keyChar ); // Caution: should the table grow to more than 5 or 6 entries, // we must consider moving it to a kind of hash map const unsigned int nExceptions = sizeof( aExceptionalKeys ) / sizeof( aExceptionalKeys[0] ); for( unsigned int i = 0; i < nExceptions; i++ ) { if( nKeyCode == aExceptionalKeys[i].nKeyCode && (mpFrame->mnLastModifierFlags & aExceptionalKeys[i].nModifierMask) == aExceptionalKeys[i].nModifierMask ) { [self sendKeyInputAndReleaseToFrame: nKeyCode character: 0]; return YES; } } } return NO; } -(void)flagsChanged: (NSEvent*)pEvent { YIELD_GUARD; if( AquaSalFrame::isAlive( mpFrame ) ) { mpFrame->mnLastEventTime = static_cast( [pEvent timestamp] * 1000.0 ); mpFrame->mnLastModifierFlags = [pEvent modifierFlags]; } } -(void)insertText:(id)aString { YIELD_GUARD; if( AquaSalFrame::isAlive( mpFrame ) ) { NSString* pInsert = nil; if( [aString isMemberOfClass: [NSAttributedString class]] ) pInsert = [aString string]; else pInsert = aString; int nLen = 0; if( pInsert && ( nLen = [pInsert length] ) > 0 ) { OUString aInsertString( GetOUString( pInsert ) ); // aCharCode initializer is safe since aInsertString will at least contain '\0' sal_Unicode aCharCode = *aInsertString.getStr(); if( nLen == 1 && aCharCode < 0x80 && aCharCode > 0x1f && ! [self hasMarkedText ] ) { USHORT nKeyCode = ImplMapCharCode( aCharCode ); unsigned int nLastModifiers = mpFrame->mnLastModifierFlags; // #i99567# // find out the unmodified key code // sanity check if( mpLastEvent && ( [mpLastEvent type] == NSKeyDown || [mpLastEvent type] == NSKeyUp ) ) { // get unmodified string NSString* pUnmodifiedString = [mpLastEvent charactersIgnoringModifiers]; if( pUnmodifiedString && [pUnmodifiedString length] == 1 ) { // map the unmodified key code unichar keyChar = [pUnmodifiedString characterAtIndex: 0]; nKeyCode = ImplMapCharCode( keyChar ); } nLastModifiers = [mpLastEvent modifierFlags]; } // #i99567# // applications and vcl's edit fields ignore key events with ALT // however we're at a place where we know text should be inserted // so it seems we need to strip the Alt modifier here if( (nLastModifiers & (NSControlKeyMask | NSAlternateKeyMask | NSCommandKeyMask)) == NSAlternateKeyMask ) { nLastModifiers = 0; } [self sendKeyInputAndReleaseToFrame: nKeyCode character: aCharCode modifiers: nLastModifiers]; } else { SalExtTextInputEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.maText = aInsertString; aEvent.mpTextAttr = NULL; aEvent.mnCursorPos = aInsertString.getLength(); aEvent.mnDeltaStart = 0; aEvent.mnCursorFlags = 0; aEvent.mbOnlyCursor = FALSE; mpFrame->CallCallback( SALEVENT_EXTTEXTINPUT, &aEvent ); if( AquaSalFrame::isAlive( mpFrame ) ) mpFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, 0 ); } } else { SalExtTextInputEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.maText = String(); aEvent.mpTextAttr = NULL; aEvent.mnCursorPos = 0; aEvent.mnDeltaStart = 0; aEvent.mnCursorFlags = 0; aEvent.mbOnlyCursor = FALSE; mpFrame->CallCallback( SALEVENT_EXTTEXTINPUT, &aEvent ); if( AquaSalFrame::isAlive( mpFrame ) ) mpFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, 0 ); } mbKeyHandled = true; [self unmarkText]; } } -(void)insertTab: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_TAB character: '\t' modifiers: 0]; } -(void)insertBacktab: (id)aSender { [self sendKeyInputAndReleaseToFrame: (KEY_TAB | KEY_SHIFT) character: '\t' modifiers: 0]; } -(void)moveLeft: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_LEFT character: 0 modifiers: 0]; } -(void)moveLeftAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_LEFT character: 0 modifiers: NSShiftKeyMask]; } -(void)moveBackwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_BACKWARD character: 0 modifiers: 0]; } -(void)moveRight: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_RIGHT character: 0 modifiers: 0]; } -(void)moveRightAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_RIGHT character: 0 modifiers: NSShiftKeyMask]; } -(void)moveForwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_FORWARD character: 0 modifiers: 0]; } -(void)moveWordLeft: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_WORD_BACKWARD character: 0 modifiers: 0]; } -(void)moveWordBackward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_WORD_BACKWARD character: 0 modifiers: 0]; } -(void)moveWordBackwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_WORD_BACKWARD character: 0 modifiers: 0]; } -(void)moveWordLeftAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_WORD_BACKWARD character: 0 modifiers: 0]; } -(void)moveWordRight: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_WORD_FORWARD character: 0 modifiers: 0]; } -(void)moveWordForward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_WORD_FORWARD character: 0 modifiers: 0]; } -(void)moveWordForwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_WORD_FORWARD character: 0 modifiers: 0]; } -(void)moveWordRightAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_WORD_FORWARD character: 0 modifiers: 0]; } -(void)moveToEndOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_LINE character: 0 modifiers: 0]; } -(void)moveToRightEndOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_LINE character: 0 modifiers: 0]; } -(void)moveToEndOfLineAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_END_OF_LINE character: 0 modifiers: 0]; } -(void)moveToRightEndOfLineAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_END_OF_LINE character: 0 modifiers: 0]; } -(void)moveToBeginningOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_LINE character: 0 modifiers: 0]; } -(void)moveToLeftEndOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_LINE character: 0 modifiers: 0]; } -(void)moveToBeginningOfLineAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_LINE character: 0 modifiers: 0]; } -(void)moveToLeftEndOfLineAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_LINE character: 0 modifiers: 0]; } -(void)moveToEndOfParagraph: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveToEndOfParagraphAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_END_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveParagraphForward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveParagraphForwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_END_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveToBeginningOfParagraph: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveParagraphBackward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveToBeginningOfParagraphAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveParagraphBackwardAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)moveToEndOfDocument: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)scrollToEndOfDocument: (id)aSender { // this is not exactly what we should do, but it makes "End" and "Shift-End" behave consistent [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_END_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)moveToEndOfDocumentAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_END_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)moveToBeginningOfDocument: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)scrollToBeginningOfDocument: (id)aSender { // this is not exactly what we should do, but it makes "Home" and "Shift-Home" behave consistent [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::MOVE_TO_BEGIN_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)moveToBeginningOfDocumentAndModifySelection: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_TO_BEGIN_OF_DOCUMENT character: 0 modifiers: 0]; } -(void)moveUp: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_UP character: 0 modifiers: 0]; } -(void)moveDown: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_DOWN character: 0 modifiers: 0]; } -(void)insertNewline: (id)aSender { // #i91267# make enter and shift-enter work by evaluating the modifiers [self sendKeyInputAndReleaseToFrame: KEY_RETURN character: '\n' modifiers: mpFrame->mnLastModifierFlags]; } -(void)deleteBackward: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_BACKSPACE character: '\b' modifiers: 0]; } -(void)deleteForward: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_DELETE character: 0x7f modifiers: 0]; } -(void)deleteBackwardByDecomposingPreviousCharacter: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_BACKSPACE character: '\b' modifiers: 0]; } -(void)deleteWordBackward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_WORD_BACKWARD character: 0 modifiers: 0]; } -(void)deleteWordForward: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_WORD_FORWARD character: 0 modifiers: 0]; } -(void)deleteToBeginningOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_TO_BEGIN_OF_LINE character: 0 modifiers: 0]; } -(void)deleteToEndOfLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_TO_END_OF_LINE character: 0 modifiers: 0]; } -(void)deleteToBeginningOfParagraph: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_TO_BEGIN_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)deleteToEndOfParagraph: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::DELETE_TO_END_OF_PARAGRAPH character: 0 modifiers: 0]; } -(void)insertLineBreak: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::INSERT_LINEBREAK character: 0 modifiers: 0]; } -(void)insertParagraphSeparator: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::INSERT_PARAGRAPH character: 0 modifiers: 0]; } -(void)selectWord: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_WORD character: 0 modifiers: 0]; } -(void)selectLine: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_LINE character: 0 modifiers: 0]; } -(void)selectParagraph: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_PARAGRAPH character: 0 modifiers: 0]; } -(void)selectAll: (id)aSender { [self sendKeyInputAndReleaseToFrame: com::sun::star::awt::Key::SELECT_ALL character: 0 modifiers: 0]; } -(void)cancelOperation: (id)aSender { [self sendKeyInputAndReleaseToFrame: KEY_ESCAPE character: 0x1b modifiers: 0]; } -(void)noop: (id)aSender { if( ! mbKeyHandled ) { if( ! [self sendSingleCharacter:mpLastEvent] ) { /* prevent recursion */ if( mpLastEvent != mpLastSuperEvent && [NSApp respondsToSelector: @selector(sendSuperEvent:)] ) { id pLastSuperEvent = mpLastSuperEvent; mpLastSuperEvent = mpLastEvent; [NSApp performSelector:@selector(sendSuperEvent:) withObject: mpLastEvent]; mpLastSuperEvent = pLastSuperEvent; std::map< NSEvent*, bool >::iterator it = GetSalData()->maKeyEventAnswer.find( mpLastEvent ); if( it != GetSalData()->maKeyEventAnswer.end() ) it->second = true; } } } } -(MacOSBOOL)sendKeyInputAndReleaseToFrame: (USHORT)nKeyCode character: (sal_Unicode)aChar { return [self sendKeyInputAndReleaseToFrame: nKeyCode character: aChar modifiers: mpFrame->mnLastModifierFlags]; } -(MacOSBOOL)sendKeyInputAndReleaseToFrame: (USHORT)nKeyCode character: (sal_Unicode)aChar modifiers: (unsigned int)nMod { return [self sendKeyToFrameDirect: nKeyCode character: aChar modifiers: nMod] || [self sendSingleCharacter: mpLastEvent]; } -(MacOSBOOL)sendKeyToFrameDirect: (USHORT)nKeyCode character: (sal_Unicode)aChar modifiers: (unsigned int)nMod { YIELD_GUARD; long nRet = 0; if( AquaSalFrame::isAlive( mpFrame ) ) { SalKeyEvent aEvent; aEvent.mnTime = mpFrame->mnLastEventTime; aEvent.mnCode = nKeyCode | ImplGetModifierMask( nMod ); aEvent.mnCharCode = aChar; aEvent.mnRepeat = FALSE; nRet = mpFrame->CallCallback( SALEVENT_KEYINPUT, &aEvent ); std::map< NSEvent*, bool >::iterator it = GetSalData()->maKeyEventAnswer.find( mpLastEvent ); if( it != GetSalData()->maKeyEventAnswer.end() ) it->second = nRet ? true : false; if( AquaSalFrame::isAlive( mpFrame ) ) mpFrame->CallCallback( SALEVENT_KEYUP, &aEvent ); } return nRet ? YES : NO; } -(MacOSBOOL)sendSingleCharacter: (NSEvent *)pEvent { NSString* pUnmodifiedString = [pEvent charactersIgnoringModifiers]; if( pUnmodifiedString && [pUnmodifiedString length] == 1 ) { unichar keyChar = [pUnmodifiedString characterAtIndex: 0]; USHORT nKeyCode = ImplMapCharCode( keyChar ); if( nKeyCode != 0 ) { // don't send unicodes in the private use area if( keyChar >= 0xf700 && keyChar < 0xf780 ) keyChar = 0; MacOSBOOL bRet = [self sendKeyToFrameDirect: nKeyCode character: keyChar modifiers: mpFrame->mnLastModifierFlags]; mbInKeyInput = false; return bRet; } } return NO; } // NSTextInput protocol - (NSArray *)validAttributesForMarkedText { return [NSArray arrayWithObjects:NSUnderlineStyleAttributeName, nil]; } - (MacOSBOOL)hasMarkedText { MacOSBOOL bHasMarkedText; bHasMarkedText = ( mMarkedRange.location != NSNotFound ) && ( mMarkedRange.length != 0 ); // hack to check keys like "Control-j" if( mbInKeyInput ) { mbNeedSpecialKeyHandle = true; } // FIXME: // #i106901# // if we come here outside of mbInKeyInput, this is likely to be because // of the keyboard viewer. For unknown reasons having no marked range // in this case causes a crash. So we say we have a marked range anyway // This is a hack, since it is not understood what a) causes that crash // and b) why we should have a marked range at this point. if( ! mbInKeyInput ) bHasMarkedText = YES; return bHasMarkedText; } - (NSRange)markedRange { // FIXME: // #i106901# // if we come here outside of mbInKeyInput, this is likely to be because // of the keyboard viewer. For unknown reasons having no marked range // in this case causes a crash. So we say we have a marked range anyway // This is a hack, since it is not understood what a) causes that crash // and b) why we should have a marked range at this point. if( ! mbInKeyInput ) return NSMakeRange( 0, 0 ); return [self hasMarkedText] ? mMarkedRange : NSMakeRange( NSNotFound, 0 ); } - (NSRange)selectedRange { return mSelectedRange; } - (void)setMarkedText:(id)aString selectedRange:(NSRange)selRange { if( ![aString isKindOfClass:[NSAttributedString class]] ) aString = [[[NSAttributedString alloc] initWithString:aString] autorelease]; NSRange rangeToReplace = [self hasMarkedText] ? [self markedRange] : [self selectedRange]; if( rangeToReplace.location == NSNotFound ) { mMarkedRange = NSMakeRange( selRange.location, [aString length] ); mSelectedRange = NSMakeRange( selRange.location, selRange.length ); } else { mMarkedRange = NSMakeRange( rangeToReplace.location, [aString length] ); mSelectedRange = NSMakeRange( rangeToReplace.location + selRange.location, selRange.length ); } int len = [aString length]; SalExtTextInputEvent aInputEvent; aInputEvent.mnTime = mpFrame->mnLastEventTime; aInputEvent.mnDeltaStart = 0; aInputEvent.mbOnlyCursor = FALSE; if( len > 0 ) { NSString *pString = [aString string]; OUString aInsertString( GetOUString( pString ) ); std::vector aInputFlags = std::vector( std::max( 1, len ), 0 ); for ( int i = 0; i < len; i++ ) { unsigned int nUnderlineValue; NSRange effectiveRange; effectiveRange = NSMakeRange(i, 1); nUnderlineValue = [[aString attribute:NSUnderlineStyleAttributeName atIndex:i effectiveRange:&effectiveRange] unsignedIntValue]; switch (nUnderlineValue & 0xff) { case NSUnderlineStyleSingle: aInputFlags[i] = SAL_EXTTEXTINPUT_ATTR_UNDERLINE; break; case NSUnderlineStyleThick: aInputFlags[i] = SAL_EXTTEXTINPUT_ATTR_UNDERLINE | SAL_EXTTEXTINPUT_ATTR_HIGHLIGHT; break; case NSUnderlineStyleDouble: aInputFlags[i] = SAL_EXTTEXTINPUT_ATTR_BOLDUNDERLINE; break; default: aInputFlags[i] = SAL_EXTTEXTINPUT_ATTR_HIGHLIGHT; break; } } aInputEvent.maText = aInsertString; aInputEvent.mnCursorPos = selRange.location; aInputEvent.mpTextAttr = &aInputFlags[0]; mpFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void *)&aInputEvent ); } else { aInputEvent.maText = String(); aInputEvent.mnCursorPos = 0; aInputEvent.mnCursorFlags = 0; aInputEvent.mpTextAttr = 0; mpFrame->CallCallback( SALEVENT_EXTTEXTINPUT, (void *)&aInputEvent ); mpFrame->CallCallback( SALEVENT_ENDEXTTEXTINPUT, 0 ); } mbKeyHandled= true; } - (void)unmarkText { mSelectedRange = mMarkedRange = NSMakeRange(NSNotFound, 0); } - (NSAttributedString *)attributedSubstringFromRange:(NSRange)theRange { // FIXME return nil; } - (unsigned int)characterIndexForPoint:(NSPoint)thePoint { // FIXME return 0; } #if defined(MAC_OS_X_VERSION_10_5) && (MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_5) /* build target 10.5 or greater */ - (NSInteger)conversationIdentifier #else /* build target 10.4 */ - (long)conversationIdentifier #endif { return (long)self; } - (void)doCommandBySelector:(SEL)aSelector { if( AquaSalFrame::isAlive( mpFrame ) ) { #if OSL_DEBUG_LEVEL > 1 // fprintf( stderr, "SalFrameView: doCommandBySelector %s\n", (char*)aSelector ); #endif if( (mpFrame->mnICOptions & SAL_INPUTCONTEXT_TEXT) != 0 && aSelector != NULL && [self respondsToSelector: aSelector] ) { [self performSelector: aSelector]; } else { [self sendSingleCharacter:mpLastEvent]; } } mbKeyHandled = true; } -(void)clearLastEvent { mpLastEvent = nil; } - (NSRect)firstRectForCharacterRange:(NSRange)theRange { SalExtTextInputPosEvent aPosEvent; mpFrame->CallCallback( SALEVENT_EXTTEXTINPUTPOS, (void *)&aPosEvent ); NSRect rect; rect.origin.x = aPosEvent.mnX + mpFrame->maGeometry.nX; rect.origin.y = aPosEvent.mnY + mpFrame->maGeometry.nY + 4; // add some space for underlines rect.size.width = aPosEvent.mnWidth; rect.size.height = aPosEvent.mnHeight; mpFrame->VCLToCocoa( rect ); return rect; } -(id)parentAttribute { return (NSView *) mpFrame -> mpWindow; } -(::com::sun::star::accessibility::XAccessibleContext *)accessibleContext { if ( mpReferenceWrapper == nil ) { // some frames never become visible .. Window *pWindow = mpFrame -> GetWindow(); if ( ! pWindow ) return nil; mpReferenceWrapper = new ReferenceWrapper; mpReferenceWrapper -> rAccessibleContext = pWindow -> /*GetAccessibleChildWindow( 0 ) ->*/ GetAccessible() -> getAccessibleContext(); [ AquaA11yFactory insertIntoWrapperRepository: self forAccessibleContext: mpReferenceWrapper -> rAccessibleContext ]; } return [ super accessibleContext ]; } -(NSView *)viewElementForParent { return (NSView *) mpFrame -> mpWindow; } -(void)registerMouseEventListener: (id)theListener { mpMouseEventListener = theListener; } -(void)unregisterMouseEventListener: (id)theListener { mpMouseEventListener = nil; } -(NSDragOperation)draggingEntered:(id )sender { return [mDraggingDestinationHandler draggingEntered: sender]; } -(NSDragOperation)draggingUpdated:(id )sender { return [mDraggingDestinationHandler draggingUpdated: sender]; } -(void)draggingExited:(id )sender { [mDraggingDestinationHandler draggingExited: sender]; } -(MacOSBOOL)prepareForDragOperation:(id )sender { return [mDraggingDestinationHandler prepareForDragOperation: sender]; } -(MacOSBOOL)performDragOperation:(id )sender { return [mDraggingDestinationHandler performDragOperation: sender]; } -(void)concludeDragOperation:(id )sender { [mDraggingDestinationHandler concludeDragOperation: sender]; } -(void)registerDraggingDestinationHandler:(id)theHandler { mDraggingDestinationHandler = theHandler; } -(void)unregisterDraggingDestinationHandler:(id)theHandler { mDraggingDestinationHandler = nil; } @end