-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbutton.go
257 lines (231 loc) · 9.67 KB
/
button.go
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
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
package consolizer
import (
"github.com/supercom32/consolizer/constants"
"github.com/supercom32/consolizer/internal/memory"
"github.com/supercom32/consolizer/internal/stringformat"
"github.com/supercom32/consolizer/types"
)
type buttonHistoryType struct {
buttonAlias string
layerAlias string
}
var buttonHistory buttonHistoryType
type ButtonInstanceType struct {
layerAlias string
buttonAlias string
}
type buttonType struct{}
var Button buttonType
func (shared *ButtonInstanceType) Delete() *ButtonInstanceType {
if memory.IsButtonExists(shared.layerAlias, shared.buttonAlias) {
memory.DeleteButton(shared.layerAlias, shared.buttonAlias)
}
return nil
}
/*
IsButtonPressed allows you to detect if any text button was pressed or not. In
order to obtain the button pressed and to clear this state, you must call the
GetButtonPressed method.
*/
func (shared *ButtonInstanceType) IsButtonPressed() bool {
if buttonHistory.layerAlias != "" && buttonHistory.buttonAlias != "" {
if buttonHistory.layerAlias == shared.layerAlias && buttonHistory.buttonAlias == shared.buttonAlias {
for shared.IsButtonStatePressed() {
}
buttonHistory.layerAlias = ""
buttonHistory.buttonAlias = ""
return true
}
}
return false
}
/*
GetButtonPressed allows you to detect which text button was pressed or not. In
the event no button was pressed, empty values for the layer and button
alias are returned instead. In addition, the following information should be
noted:
- If any button is successfully returned, the pressed state is automatically
cleared.
*/
func (shared *ButtonInstanceType) GetButtonPressed() (string, string) {
if buttonHistory.layerAlias != "" && buttonHistory.buttonAlias != "" {
layerAlias := buttonHistory.layerAlias
buttonAlias := buttonHistory.buttonAlias
buttonHistory.layerAlias = ""
buttonHistory.buttonAlias = ""
return layerAlias, buttonAlias
}
return "", ""
}
func (shared *ButtonInstanceType) IsButtonStatePressed() bool {
buttonEntry := memory.GetButton(shared.layerAlias, shared.buttonAlias)
if buttonEntry.IsPressed == true {
return true
}
return false
}
/*
Add allows you to add a button to a text layer. Once called, an instance of your control is
returned which will allow you to read or manipulate the properties for it. The Style of the button
will be determined by the style entry passed in. If you wish to remove a button from a text
layer, simply call 'DeleteButton'. In addition, the following information should be noted:
- Buttons are not drawn physically to the text layer provided. Instead
they are rendered to the terminal at the same time when the text layer is
rendered. This allows you to create buttons without actually overwriting
the text layer data under it.
- If the button to be drawn falls outside the range of the provided layer,
then only the visible portion of the button will be drawn.
- If the width of your button is less than the length of your button label,
then the width will automatically default to the width of your button label.
- If the height of your button is less than 3 characters high, then the height
will automatically default to the minimum of 3 characters.
*/
func (shared *buttonType) Add(layerAlias string, buttonAlias string, buttonLabel string, styleEntry types.TuiStyleEntryType, xLocation int, yLocation int, width int, height int, isEnabled bool) ButtonInstanceType {
memory.AddButton(layerAlias, buttonAlias, buttonLabel, styleEntry, xLocation, yLocation, width, height)
buttonEntry := memory.GetButton(layerAlias, buttonAlias)
buttonEntry.IsEnabled = isEnabled
var buttonInstance ButtonInstanceType
buttonInstance.layerAlias = layerAlias
buttonInstance.buttonAlias = buttonAlias
return buttonInstance
}
/*
DeleteButton allows you to remove a button from a text layer. In addition,
the following information should be noted:
- If you attempt to delete a button which does not exist, then the request
will simply be ignored.
*/
func (shared *buttonType) DeleteButton(layerAlias string, buttonAlias string) {
memory.DeleteButton(layerAlias, buttonAlias)
}
/*
DeleteAllButtons allows you to delete all buttons on a given text layer.
*/
func (shared *buttonType) DeleteAllButtons(layerAlias string) {
memory.DeleteAllButtonsFromLayer(layerAlias)
}
/*
drawButtonsOnLayer allows you to draw all buttons on a given text layer.
*/
func (shared *buttonType) drawButtonsOnLayer(layerEntry types.LayerEntryType) {
layerAlias := layerEntry.LayerAlias
for currentKey := range memory.Button.Entries[layerAlias] {
buttonEntry := memory.Button.Entries[layerAlias][currentKey]
drawButton(&layerEntry, currentKey, buttonEntry.ButtonLabel, buttonEntry.StyleEntry, buttonEntry.IsPressed, buttonEntry.IsSelected, buttonEntry.IsEnabled, buttonEntry.XLocation, buttonEntry.YLocation, buttonEntry.Width, buttonEntry.Height)
}
}
/*
drawButton allows you to draw A button on a given text layer. The
Style of the button will be determined by the style entry passed in. In
addition, the following information should be noted:
- Buttons are not drawn physically to the text layer provided. Instead,
they are rendered to the terminal at the same time when the text layer is
rendered. This allows you to create buttons without actually overwriting
the text layer data under it.
- If the button to be drawn falls outside the range of the provided layer,
then only the visible portion of the button will be drawn.
*/
func drawButton(layerEntry *types.LayerEntryType, buttonAlias string, buttonLabel string, styleEntry types.TuiStyleEntryType, isPressed bool, isSelected bool, isEnabled bool, xLocation int, yLocation int, width int, height int) {
localStyleEntry := types.NewTuiStyleEntry(&styleEntry)
attributeEntry := types.NewAttributeEntry()
attributeEntry.ForegroundColor = styleEntry.ButtonForegroundColor
attributeEntry.BackgroundColor = styleEntry.ButtonBackgroundColor
attributeEntry.CellType = constants.CellTypeButton
attributeEntry.CellControlAlias = buttonAlias
if height < 3 {
height = 3
}
if width-2 <= len(buttonLabel) {
width = len(buttonLabel) + 2
}
localStyleEntry.LineDrawingTextForegroundColor = localStyleEntry.ButtonRaisedColor
localStyleEntry.LineDrawingTextBackgroundColor = localStyleEntry.ButtonBackgroundColor
fillArea(layerEntry, attributeEntry, " ", xLocation, yLocation, width, height, constants.NullCellControlLocation)
if isPressed {
drawFrame(layerEntry, localStyleEntry, attributeEntry, constants.FrameStyleSunken, xLocation, yLocation, width, height, false)
} else {
drawFrame(layerEntry, localStyleEntry, attributeEntry, constants.FrameStyleRaised, xLocation, yLocation, width, height, false)
}
centerXLocation := (width - len(buttonLabel)) / 2
centerYLocation := height / 2
arrayOfRunes := stringformat.GetRunesFromString(buttonLabel)
if isSelected {
attributeEntry.IsUnderlined = true
}
if !isEnabled {
attributeEntry.ForegroundColor = styleEntry.ButtonLabelDisabledColor
}
printLayer(layerEntry, attributeEntry, xLocation+centerXLocation, yLocation+centerYLocation, arrayOfRunes)
}
/*
updateButtonStates allows you to update the state of all buttons. This needs
to be called when input occurs so that changes in button state are reflected
to the user as quickly as possible. In the event that a screen update is
required this method returns true.
*/
func (shared *buttonType) updateButtonStates(isMouseTriggered bool) bool {
if isMouseTriggered {
// Update the button state if a mouse caused a change.
return shared.updateButtonStateMouse()
} else {
// Add code to update when keyboard caused a change.
}
return false
}
/*
updateButtonStateMouse allows you to update button states that are triggered
by mouse events. If a screen update is required, then this method returns
true.
*/
func (shared *buttonType) updateButtonStateMouse() bool {
isUpdateRequired := false
mouseXLocation, mouseYLocation, buttonPressed, _ := memory.GetMouseStatus()
characterEntry := getCellInformationUnderMouseCursor(mouseXLocation, mouseYLocation)
layerAlias := characterEntry.LayerAlias
buttonAlias := characterEntry.AttributeEntry.CellControlAlias
// If not a button, reset all buttons if needed.
if characterEntry.AttributeEntry.CellType != constants.CellTypeButton {
for currentLayer, _ := range memory.Button.Entries {
for currentButton, _ := range memory.Button.Entries[currentLayer] {
// In case of delete race condition, we check this first.
if !memory.IsButtonExists(currentLayer, currentButton) {
continue
}
// If button not found, reset all buttons if necessary...
buttonEntry := memory.GetButton(currentLayer, currentButton)
if buttonEntry.IsPressed == true {
buttonHistory.layerAlias = layerAlias
buttonHistory.buttonAlias = buttonAlias
buttonEntry.Mutex.Lock()
buttonEntry.IsPressed = false
buttonEntry.Mutex.Unlock()
isUpdateRequired = true
}
}
}
return isUpdateRequired
}
if buttonAlias != "" && buttonPressed == 0 && memory.IsButtonExists(layerAlias, buttonAlias) {
buttonEntry := memory.GetButton(layerAlias, buttonAlias)
if buttonEntry.IsPressed == true {
buttonEntry.Mutex.Lock()
buttonEntry.IsPressed = false
buttonEntry.Mutex.Unlock()
isUpdateRequired = true
}
} else if buttonAlias != "" && buttonPressed != 0 && memory.IsButtonExists(layerAlias, buttonAlias) {
// If button was found and mouse is being pressed, update button only
// if required.
buttonEntry := memory.GetButton(layerAlias, buttonAlias)
if buttonEntry.IsEnabled && buttonEntry.IsPressed == false {
buttonEntry.Mutex.Lock()
buttonHistory.layerAlias = layerAlias
buttonHistory.buttonAlias = buttonAlias
buttonEntry.IsPressed = true
buttonEntry.Mutex.Unlock()
setFocusedControl(layerAlias, buttonAlias, constants.CellTypeButton)
isUpdateRequired = true
}
}
return isUpdateRequired
}