-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcombobox._hs
159 lines (122 loc) · 2.97 KB
/
combobox._hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
behavior combobox
init
add .combobox to me
set :input to the first <input/> in me
set :menu to the first <[role=menu]/> in me
set :menu's *top to my height
for menuItem in (:menu).children
add [@role=menuitem] to menuItem
end
-- -------------------------
-- INPUT EVENTS
-- -------------------------
on focus from <input/> in me
exit
on blur from <input/> in me
if menuIsVisible() then
wait 100ms
call hideMenu()
end
-- -------------------------
-- MOUSE EVENTS
-- -------------------------
on click from <.open-button /> in me
focus() the :input
call toggleMenu()
on mouseover
if target's [@role] == "menuitem" then
take .selected for target
end
on mousedown
if target's [@role] == "menuitem" then
set item to target
else
set item to the closest <[role="menuitem"]/> to the target
end
if the item is not null then
call selectMenuItem(item)
call hideMenu()
call :input.focus()
end
-- -------------------------
-- KEYBOARD EVENTS
-- -------------------------
on keydown[code=="ArrowDown"]
halt the event
if menuIsHidden() then
call showMenu()
exit
end
set selected to getSelectedItem()
if selected is null then
take .selected for the first <[role="menuitem"]/> in me
exit
end
set nextItem to the next <[role="menuitem"]/> from selected
if nextItem is not null then
take .selected for nextItem
call nextItem.scrollIntoView({block:"nearest"})
end
on keydown[code=="ArrowUp"]
halt the event
if menuIsHidden() then
call showMenu()
exit
end
set selected to getSelectedItem()
if selected is null then
take .selected for the last <[role="menuitem"]/> in :menu
exit
end
set prevItem to the previous <[role="menuitem"]/> from selected
if prevItem is not null then
take .selected for prevItem
call prevItem.scrollIntoView({block:"nearest"})
end
on keypress[code=="Enter"]
if menuIsVisible then
halt the event
set current to the first <[role="menuitem"].selected /> in me
call selectMenuItem(current)
call hideMenu()
end
on keypress[code=="Escape"]
call hideMenu()
-- -------------------------
-- PRIVATE FUNCTIONS OMG!!
-- -------------------------
def menuIsVisible
return (:menu's *opacity is 1)
def menuIsHidden
return (:menu's *opacity is 0)
def showMenu
if menuIsHidden() then
remove [@hidden] from :menu
set the :menu's *opacity to 1
end
def hideMenu
if menuIsVisible() then
async do
wait a tick
transition the :menu's opacity to 0 over 100ms
add [@hidden=true] to :menu
end
end
def toggleMenu
if menuIsVisible() then
call hideMenu()
else
call showMenu()
end
def selectMenuItem(node)
if node is not null then
set :input.value to getValue(node)
end
def getSelectedItem
return the first <[role="menuitem"].selected /> in :menu
def getValue(node)
set value to the node's [@value]
if value is not null then
return value
end
return the node's innerText