-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathprefsWidgets.js
183 lines (151 loc) · 5.2 KB
/
prefsWidgets.js
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
'use strict';
import GObject from 'gi://GObject';
import Gtk from 'gi://Gtk';
import GLib from 'gi://GLib';
export const boxProperties = {
spacing: 0,
margin_start: 6,
margin_end: 6,
hexpand: true,
halign: Gtk.Align.START,
margin_top: 0,
margin_bottom: 0,
};
export const newColumnViewColumn = function(title, factorySetupFunc, factoryBindFunc) {
const factory = new Gtk.SignalListItemFactory();
const columnViewColumn = new Gtk.ColumnViewColumn({
title,
factory
});
if (factorySetupFunc) {
factory.connect('setup', (factory, listItem) => {
factorySetupFunc(factory, listItem);
});
}
if (factoryBindFunc) {
factory.connect('bind', (factory, listItem) => {
factoryBindFunc(factory, listItem);
});
}
return columnViewColumn;
}
export const newRemoveButton = function() {
return new BoxRemoveButton();
}
export const newLabelSwitch = function(text, tooltipText, active) {
return new LabelSwitch(text, tooltipText, active);
}
export const updateStyle = function(widget, css) {
const cssProvider = new Gtk.CssProvider();
cssProvider.load_from_data(css, -1);
widget.get_style_context().add_provider(cssProvider, Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION);
}
export const _newBox = function(properties) {
const box = new Gtk.Box({
spacing: 6,
margin_top: 6,
margin_bottom: 6,
margin_start: 6,
margin_end: 6,
})
Object.assign(box, properties);
return box;
}
export const _newDropDown = function(values, activeValue) {
const dropDownValues = values.map(cv => cv[1]);
const dropDown = Gtk.DropDown.new_from_strings(dropDownValues);
dropDown.set_valign(Gtk.Align.BASELINE);
for (let i = 0; i < dropDownValues.length; i++) {
if (dropDownValues[i] === activeValue)
dropDown.set_selected(i);
}
const factory = dropDown.get_factory();
factory.connect('bind', (factory, listItem) => {
const box = listItem.get_child();
const label = box.get_first_child();
const widthChars = Math.max(...dropDownValues.map(
// GLib.utf8_strlen(v, -1) causes right margin between the label and box is too large, so -2 to reduce this margin
v => GLib.utf8_strlen(v, -1) - 2));
label.set_width_chars(widthChars);
});
return dropDown;
}
export const LabelSwitch = GObject.registerClass({
Signals: {
'active': {
param_types: [GObject.TYPE_BOOLEAN, Gtk.Switch]
}
}
}, class LabelSwitch extends Gtk.Box {
_init(text, tooltipText, active) {
super._init(boxProperties);
this.tooltip_text = tooltipText;
const [button, switcherBox, switcher] = this._initSwitch(text, tooltipText);
switcher.active = active ? active : false;
this.append(button);
this.append(switcherBox);
switcher.connect('notify::active', (switcher) => {
this.emit('active', switcher.get_active(), switcher);
});
}
_initSwitch(text, tooltipText) {
const button = new Gtk.Button({
label: text,
can_target: false
});
// Imitate a button
// Here we don't use Gtk.Button with a Gtk.Switch. Because I don't want to get into the trouble that
// the click event can't be propagated down to the Gtk.Switch.
const switcherBox = new Gtk.Box({css_name: 'button'});
const switcher = new Gtk.Switch({valign: Gtk.Align.CENTER});
updateStyle(button,
// Use .text-button if the button displays a label; Use .image-button if it displays an image
`.text-button {
padding-right: 0px;
padding-left: 6px;
border-top-right-radius: 0px;
border-bottom-right-radius: 0px;
}`);
updateStyle(switcherBox,
`button {
padding-right: 6px;
padding-left: 6px;
border-top-left-radius: 0px;
border-bottom-left-radius: 0px;
}`);
switcherBox.append(switcher);
return [button, switcherBox, switcher];
}
});
export const BoxRemoveButton = GObject.registerClass({
Signals: {'clicked': {}}
}, class BoxRemoveButton extends Gtk.Box {
_init() {
super._init(boxProperties);
Object.assign(this, {
hexpand: true,
halign: Gtk.Align.START
});
const boxRemoveButton = new Gtk.Button({
icon_name: 'edit-delete-symbolic',
});
this.append(boxRemoveButton);
boxRemoveButton.connect('clicked', () => {
this.emit('clicked');
});
}
});
// TODO This function does not work
export const addScrolledWindow = function(widget) {
const scroll = new Gtk.ScrolledWindow({
vexpand: true,
hexpand: true,
hscrollbar_policy: Gtk.PolicyType.NEVER,
vscrollbar_policy: Gtk.PolicyType.AUTOMATIC
});
const parent = widget.get_parent();
widget.unparent();
scroll.set_child(widget);
// How to add widget to Adw.PreferencesPage. parent is a Adw.PreferencesPage?
parent.add_child(scroll);
}