From 1c97b0d76e2902a5f64360cd14c866d22e7e724b Mon Sep 17 00:00:00 2001 From: Sankaranarayanan Viswanathan Date: Fri, 21 Oct 2016 18:44:22 -0400 Subject: [PATCH] menus: allow keyboard control --- basis/ui/gadgets/menus/menus.factor | 113 ++++++++++++++++++++++++---- 1 file changed, 98 insertions(+), 15 deletions(-) diff --git a/basis/ui/gadgets/menus/menus.factor b/basis/ui/gadgets/menus/menus.factor index 71e40105df..429630e573 100644 --- a/basis/ui/gadgets/menus/menus.factor +++ b/basis/ui/gadgets/menus/menus.factor @@ -1,29 +1,56 @@ ! Copyright (C) 2005, 2009 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. -USING: accessors kernel locals math.rectangles math.vectors -namespaces opengl sequences sorting ui.commands ui.gadgets -ui.gadgets.buttons ui.gadgets.glass ui.gadgets.packs -ui.gadgets.worlds ui.gadgets.wrappers ui.gestures ui.operations -ui.pens ui.pens.solid ui.theme ui.tools.common ; +USING: accessors combinators kernel locals math math.rectangles +math.vectors memoize models namespaces opengl sequences sorting +ui.commands ui.gadgets ui.gadgets.buttons ui.gadgets.glass +ui.gadgets.packs ui.gadgets.worlds ui.gadgets.wrappers ui.gestures +ui.operations ui.pens ui.pens.solid ui.theme ui.tools.common ; FROM: ui.gadgets.wrappers => wrapper ; IN: ui.gadgets.menus +rect show-glass ; +PRIVATE> + : show-menu ( owner menu -- ) [ (show-menu) ] keep request-focus ; -GENERIC: ( target hook command -- button ) +TUPLE: menu-button < button ; -M:: object ( target hook command -- button ) +>align ; inline + +MEMO: menu-button-pen-boundary ( -- pen ) + f f roll-button-rollover-border dup dup ; + +MEMO: menu-button-pen-interior ( -- pen ) + f f roll-button-selected-background f over ; + +: menu-button-theme ( menu-button -- menu-button ) + menu-button-pen-boundary >>boundary + menu-button-pen-interior >>interior + align-left ; inline + +: ( label quot -- menu-button ) + menu-button new-button menu-button-theme ; inline + +PRIVATE> + +GENERIC: ( target hook command -- menu-item ) + +M:: object ( target hook command -- menu-item ) command command-name [ hook call target command command-button-quot call hide-glass - ] ; + ] ; { 0 5 } >>dim menu-border-color >>interior ; -TUPLE: menu < wrapper ; +TUPLE: menu < wrapper + items ; + +> [ menu-button? ] filter ; + +:: prepare-menu ( menu items -- ) + f :> model + items menu-buttons :> buttons + buttons [ model add-connection ] each + menu model >>model buttons >>items drop ; + +PRIVATE> + +M: menu-button model-changed + swap value>> over = >>selected? relayout-1 ; + +M: menu-button handle-gesture + [ + { + { [ over mouse-enter? ] [ nip activate-item ] } + { [ over mouse-leave? ] [ nip inactivate-item ] } + [ 2drop ] + } cond + ] 2keep call-next-method ; + +> ] [ control-value ] bi :> ( items curr ) + curr [ + items length :> max + curr items index :> indx + indx dir + max rem items nth + ] [ items first ] if menu set-control-value ; + +: activate-menu-item ( menu -- ) + control-value [ + dup quot>> ( button -- ) call-effect + ] when* ; + +PRIVATE> menu H{ { T{ key-down f f "ESC" } [ hide-glass ] } + { T{ key-down f f "DOWN" } [ 1 next-item ] } + { T{ key-down f f "UP" } [ -1 next-item ] } + { T{ key-down f f "RET" } [ activate-menu-item ] } } set-gestures : ( gadgets -- menu ) - - { 0 3 } >>gap - margins - menu-border-color >>boundary - menu-background >>interior - menu new-wrapper ; + [ + { 0 3 } >>gap + margins + menu-border-color >>boundary + menu-background >>interior + menu new-wrapper + ] [ dupd prepare-menu ] bi ; : ( target hook commands -- menu ) [ ] 2with map ;