-
-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathbutton.ts
112 lines (94 loc) · 2.48 KB
/
button.ts
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
// Copyright 2023 Im-Beast. MIT license.
import { ComponentOptions } from "../component.ts";
import { Box } from "./box.ts";
import type { BoxObject } from "../canvas/box.ts";
import { Label, LabelAlign, LabelRectangle } from "./label.ts";
import { Signal, SignalOfObject } from "../signals/mod.ts";
import { signalify } from "../utils/signals.ts";
const centerAlign: LabelAlign = {
horizontal: "center",
vertical: "center",
};
export interface ButtonOptions extends ComponentOptions {
label?: {
text: string | Signal<string>;
align?: LabelAlign | SignalOfObject<LabelAlign>;
};
}
/**
* Component for creating interactive button
*
* @example
* ```ts
* new Button({
* parent: tui,
* label: { text: "click\nme" },
* theme: {
* base: crayon.bgGreen,
* focused: crayon.bgLightGreen,
* active: crayon.bgYellow,
* },
* rectangle: {
* column: 1,
* row: 1,
* height: 5,
* width: 10,
* },
* zIndex: 0,
* });
* ```
*/
export class Button extends Box {
declare drawnObjects: { box: BoxObject };
declare subComponents: { label?: Label };
label: {
text: Signal<string>;
align: Signal<LabelAlign>;
};
constructor(options: ButtonOptions) {
super(options);
let { label } = options;
if (!label) {
label = { text: "", align: centerAlign };
}
label.text = signalify(label.text);
label.align = signalify(label.align ?? centerAlign);
this.label = label as this["label"];
this.label.text.subscribe(() => {
this.#updateLabelSubcomponent();
});
}
draw(): void {
super.draw();
this.#updateLabelSubcomponent();
}
interact(method: "mouse" | "keyboard"): void {
const interactionInterval = Date.now() - this.lastInteraction.time;
this.state.value = this.state.peek() === "focused" && (interactionInterval < 500 || method === "keyboard")
? "active"
: "focused";
super.interact(method);
}
#updateLabelSubcomponent(): void {
if (!this.label.text.value) {
this.subComponents.label?.destroy();
return;
}
if (this.subComponents.label) {
return;
}
const label = new Label({
parent: this,
theme: this.theme,
zIndex: this.zIndex,
rectangle: this.rectangle as Signal<LabelRectangle>,
overwriteRectangle: true,
text: this.label.text,
align: this.label.align,
});
label.state = this.state;
label.style = this.style;
label.subComponentOf = this;
this.subComponents.label = label;
}
}