! Copyright (C) 2006 Slava Pestov ! See http://factorcode.org/license.txt for BSD license. IN: objc-classes DEFER: FactorView IN: cocoa USING: arrays gadgets hashtables kernel math namespaces objc opengl sequences ; : ( class dim -- view ) >r -> alloc 0 0 r> first2 NSOpenGLView -> defaultPixelFormat -> initWithFrame:pixelFormat: dup 1 -> setPostsBoundsChangedNotifications: dup 1 -> setPostsFrameChangedNotifications: ; : view-dim -> bounds dup NSRect-w swap NSRect-h 2array ; : mouse-location ( view event -- loc ) over >r -> locationInWindow f -> convertPoint:fromView: dup NSPoint-x swap NSPoint-y r> -> frame NSRect-h swap - 2array ; : send-mouse-moved ( view event -- ) over >r mouse-location r> window move-hand fire-motion ; : button ( event -- n ) #! Cocoa -> Factor UI button mapping -> buttonNumber H{ { 0 1 } { 2 2 } { 1 3 } } hash ; : button&loc ( view event -- button# loc ) dup button -rot mouse-location ; : modifiers { { S+ HEX: 20000 } { C+ HEX: 40000 } { A+ HEX: 80000 } { M+ HEX: 100000 } } ; : key-codes H{ { 36 "RETURN" } { 48 "TAB" } { 51 "BACKSPACE" } { 115 "HOME" } { 117 "DELETE" } { 119 "END" } { 123 "LEFT" } { 124 "RIGHT" } { 125 "DOWN" } { 126 "UP" } } ; : key-code ( event -- string ) dup -> keyCode key-codes hash [ ] [ -> charactersIgnoringModifiers CF>string ] ?if ; : event>gesture ( event -- modifiers keycode ) dup -> modifierFlags modifiers modifier swap key-code ; : send-key-event ( view event quot -- ? ) >r event>gesture r> call swap window-focus handle-gesture ; inline : send-user-input ( view event -- ) -> characters CF>string swap window-focus user-input ; : send-key-down-event ( view event -- ) 2dup [ ] send-key-event [ send-user-input ] [ 2drop ] if ; : send-key-up-event ( view event -- ) [ ] send-key-event drop ; : send-button-down$ ( view event -- ) over >r button&loc r> window send-button-down ; : send-button-up$ ( view event -- ) over >r button&loc r> window send-button-up ; : send-wheel$ ( view event -- ) [ -> deltaY 0 > ] 2keep mouse-location rot window send-wheel ; : send-action$ ( view event gesture -- junk ) >r drop window r> send-action f ; : add-resize-observer ( observer object -- ) >r "updateFactorGadgetSize:" "NSViewFrameDidChangeNotification" r> add-observer ; : string-or-nil? ( NSString -- ? ) [ CF>string NSStringPboardType = ] [ t ] if* ; : valid-service? ( gadget send-type return-type -- ? ) over string-or-nil? over string-or-nil? and [ drop [ gadget-selection? ] [ drop t ] if ] [ 3drop f ] if ; "NSOpenGLView" "FactorView" { ! Events { "acceptsFirstMouse:" "bool" { "id" "SEL" "id" } [ 3drop 1 ] } { "mouseEntered:" "void" { "id" "SEL" "id" } [ [ nip send-mouse-moved ] ui-try ] } { "mouseExited:" "void" { "id" "SEL" "id" } [ [ 3drop forget-rollover ] ui-try ] } { "mouseMoved:" "void" { "id" "SEL" "id" } [ [ nip send-mouse-moved ] ui-try ] } { "mouseDragged:" "void" { "id" "SEL" "id" } [ [ nip send-mouse-moved ] ui-try ] } { "rightMouseDragged:" "void" { "id" "SEL" "id" } [ [ nip send-mouse-moved ] ui-try ] } { "otherMouseDragged:" "void" { "id" "SEL" "id" } [ [ nip send-mouse-moved ] ui-try ] } { "mouseDown:" "void" { "id" "SEL" "id" } [ [ nip send-button-down$ ] ui-try ] } { "mouseUp:" "void" { "id" "SEL" "id" } [ [ nip send-button-up$ ] ui-try ] } { "rightMouseDown:" "void" { "id" "SEL" "id" } [ [ nip send-button-down$ ] ui-try ] } { "rightMouseUp:" "void" { "id" "SEL" "id" } [ [ nip send-button-up$ ] ui-try ] } { "otherMouseDown:" "void" { "id" "SEL" "id" } [ [ nip send-button-down$ ] ui-try ] } { "otherMouseUp:" "void" { "id" "SEL" "id" } [ [ nip send-button-up$ ] ui-try ] } { "scrollWheel:" "void" { "id" "SEL" "id" } [ [ nip send-wheel$ ] ui-try ] } { "keyDown:" "void" { "id" "SEL" "id" } [ [ nip send-key-down-event ] ui-try ] } { "keyUp:" "void" { "id" "SEL" "id" } [ [ nip send-key-up-event ] ui-try ] } { "cut:" "id" { "id" "SEL" "id" } [ [ nip T{ cut-action } send-action$ ] ui-try ] } { "copy:" "id" { "id" "SEL" "id" } [ [ nip T{ copy-action } send-action$ ] ui-try ] } { "paste:" "id" { "id" "SEL" "id" } [ [ nip T{ paste-action } send-action$ ] ui-try ] } { "delete:" "id" { "id" "SEL" "id" } [ [ nip T{ delete-action } send-action$ ] ui-try ] } { "selectAll:" "id" { "id" "SEL" "id" } [ [ nip T{ select-all-action } send-action$ ] ui-try ] } ! Services { "validRequestorForSendType:returnType:" "id" { "id" "SEL" "id" "id" } [ ! We return either self or nil >r >r over window-focus r> r> valid-service? [ drop ] [ 2drop f ] if ] } { "writeSelectionToPasteboard:types:" "bool" { "id" "SEL" "id" "id" } [ CF>string-array NSStringPboardType swap member? [ >r drop window-focus gadget-selection dup [ r> set-pasteboard-string t ] [ r> 2drop f ] if ] [ 3drop f ] if ] } { "readSelectionFromPasteboard:" "bool" { "id" "SEL" "id" } [ pasteboard-string dup [ >r drop window-focus r> swap user-input t ] [ 3drop f ] if ] } { "updateFactorGadgetSize:" "void" { "id" "SEL" "id" } [ [ 2drop dup view-dim swap window set-gadget-dim ] ui-try ] } { "acceptsFirstResponder" "bool" { "id" "SEL" } [ 2drop 1 ] } { "initWithFrame:pixelFormat:" "id" { "id" "SEL" "NSRect" "id" } [ rot drop SUPER-> initWithFrame:pixelFormat: dup dup add-resize-observer ] } { "dealloc" "void" { "id" "SEL" } [ drop dup window close-world dup unregister-window dup remove-observer SUPER-> dealloc ] } } { } define-objc-class : ( world -- view ) FactorView over rect-dim [ register-window ] keep ;