Skip to content

Commit 24d542c

Browse files
committed
add: duplicate action
resolve: #205
1 parent cde633a commit 24d542c

File tree

5 files changed

+78
-38
lines changed

5 files changed

+78
-38
lines changed

addon/chrome/content/preferences.xhtml

+21-7
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,21 @@
66
<html:link rel="localization" href="__addonRef__-preferences.ftl" />
77
</linkset>
88
<html:style>
9-
.action-button { min-width: auto; }
9+
.action-button { min-width:auto; width:30px; height:24px;
10+
-moz-context-properties: fill, fill-opacity; fill: currentColor;}
11+
#__addonRef__-action-add {
12+
list-style-image:url("");
13+
} #__addonRef__-action-remove {
14+
list-style-image:url("");
15+
} #__addonRef__-action-edit {
16+
list-style-image:url("");
17+
} #__addonRef__-action-duplicate {
18+
list-style-image:url("");
19+
} #__addonRef__-action-export {
20+
list-style-image:url("");
21+
} #__addonRef__-action-import {
22+
list-style-image:url("");
23+
}
1024
</html:style>
1125
<groupbox style="width: -moz-available">
1226
<label><html:h2 data-l10n-id="action"></html:h2></label>
@@ -16,31 +30,31 @@
1630
<hbox>
1731
<button
1832
id="__addonRef__-action-add"
19-
label="+"
2033
class="action-button"
2134
data-l10n-id="action-add"
2235
></button>
2336
<button
2437
id="__addonRef__-action-remove"
25-
label="-"
2638
class="action-button action-selection"
2739
data-l10n-id="action-remove"
2840
></button>
2941
<button
3042
id="__addonRef__-action-edit"
31-
label=""
32-
class="action-button action-selection"
43+
class="action-button action-selection action-selection-single"
3344
data-l10n-id="action-edit"
3445
></button>
46+
<button
47+
id="__addonRef__-action-duplicate"
48+
class="action-button action-selection action-selection-single"
49+
data-l10n-id="action-duplicate"
50+
></button>
3551
<button
3652
id="__addonRef__-action-export"
37-
label=""
3853
class="action-button action-selection"
3954
data-l10n-id="action-export"
4055
></button>
4156
<button
4257
id="__addonRef__-action-import"
43-
label=""
4458
class="action-button"
4559
data-l10n-id="action-import"
4660
></button>

addon/locale/en-US/preferences.ftl

+2-5
Original file line numberDiff line numberDiff line change
@@ -22,19 +22,16 @@ action-operation-script = Script
2222
action-operation-triggerAction = Trigger Another Action
2323
2424
action-add =
25-
.label = +
2625
.tooltiptext = Add a new action
2726
action-remove =
28-
.label = -
2927
.tooltiptext = Delete selected action
3028
action-edit =
31-
.label =
3229
.tooltiptext = Edit selected action
30+
action-duplicate =
31+
.tooltiptext = Duplicate selected action
3332
action-export =
34-
.label =
3533
.tooltiptext = Export selected actions to file
3634
action-import =
37-
.label =
3835
.tooltiptext = Import actions from file
3936
4037
menu = Menu

addon/locale/it-IT/preferences.ftl

+2-5
Original file line numberDiff line numberDiff line change
@@ -22,19 +22,16 @@ action-operation-script = Script
2222
action-operation-triggerAction = Innesca altra azione
2323
2424
action-add =
25-
.label = +
2625
.tooltiptext = Add a new action
2726
action-remove =
28-
.label = -
2927
.tooltiptext = Delete selected action
3028
action-edit =
31-
.label =
3229
.tooltiptext = Edit selected action
30+
action-duplicate =
31+
.tooltiptext = Duplicate selected action
3332
action-export =
34-
.label =
3533
.tooltiptext = Export selected actions to file
3634
action-import =
37-
.label =
3835
.tooltiptext = Import actions from file
3936
4037
menu = Menu

addon/locale/zh-CN/preferences.ftl

+2-5
Original file line numberDiff line numberDiff line change
@@ -22,19 +22,16 @@ action-operation-script = 自定义脚本
2222
action-operation-triggerAction = 触发另一个动作
2323
2424
action-add =
25-
.label = +
2625
.tooltiptext = 创建新动作
2726
action-remove =
28-
.label = -
2927
.tooltiptext = 移除选中的动作
3028
action-edit =
31-
.label =
3229
.tooltiptext = 编辑选中的动作
30+
action-duplicate =
31+
.tooltiptext = 复制选中的动作
3332
action-export =
34-
.label =
3533
.tooltiptext = 导出选中的动作到文件
3634
action-import =
37-
.label =
3835
.tooltiptext = 从文件批量导入动作
3936
4037
menu = 菜单

src/modules/preferenceWindow.ts

+51-16
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ async function initUI() {
1919
const renderLock = Zotero.Promise.defer();
2020
if (!isWindowAlive(addon.data.prefs.window)) return;
2121
addon.data.prefs.tableHelper = new ztoolkit.VirtualizedTable(
22-
addon.data.prefs.window!,
22+
addon.data.prefs.window!
2323
)
2424
.setContainerId(`${config.addonRef}-table-container`)
2525
.setProp({
@@ -34,19 +34,20 @@ async function initUI() {
3434
.setProp("getRowData", getRowData as any)
3535
// Update selected key when selection changes
3636
.setProp("onSelectionChange", (selection) => {
37-
const switchButtons =
38-
addon.data.prefs.window?.document.querySelectorAll(".action-selection");
39-
for (let i = 0; i < addon.data.actions.cachedKeys.length; i++) {
40-
if (selection.isSelected(i)) {
41-
addon.data.actions.selectedKey = addon.data.actions.cachedKeys[i];
42-
switchButtons &&
43-
switchButtons.forEach((e) => e.removeAttribute("disabled"));
44-
return;
45-
}
46-
}
47-
addon.data.actions.selectedKey = undefined;
48-
switchButtons &&
49-
switchButtons.forEach((e) => e.setAttribute("disabled", "true"));
37+
const selectedKeys = getSelection();
38+
addon.data.actions.selectedKey = selectedKeys[0];
39+
40+
addon.data.prefs.window?.document
41+
.querySelectorAll(".action-selection")
42+
?.forEach((e) =>
43+
setButtonDisabled(e as XUL.Button, selectedKeys.length === 0)
44+
);
45+
46+
addon.data.prefs.window?.document
47+
.querySelectorAll(".action-selection-single")
48+
?.forEach((e) =>
49+
setButtonDisabled(e as XUL.Button, selectedKeys.length !== 1)
50+
);
5051
})
5152
// When pressing delete, delete selected line and refresh table.
5253
// Returning false to prevent default event.
@@ -96,7 +97,7 @@ function initEvents() {
9697
.querySelector(`#${config.addonRef}-action-add`)
9798
?.addEventListener("command", (e) => {
9899
const key = addon.api.actionManager.updateAction(
99-
Object.assign({}, emptyAction),
100+
Object.assign({}, emptyAction)
100101
);
101102
updateUI();
102103
editAndUpdate(key);
@@ -117,6 +118,32 @@ function initEvents() {
117118
await editAndUpdate();
118119
});
119120

121+
doc
122+
.querySelector(`#${config.addonRef}-action-duplicate`)
123+
?.addEventListener("command", async (e) => {
124+
const newAction = Object.assign(
125+
{},
126+
addon.data.actions.map.get(addon.data.actions.selectedKey!)
127+
);
128+
if (newAction.name) {
129+
// Add (1) (2) ... to the end of the name
130+
if (newAction.name.match(/\(\d+\)$/)) {
131+
newAction.name = newAction.name.replace(
132+
/\((\d+)\)$/,
133+
(_, num) => `(${parseInt(num) + 1})`
134+
);
135+
} else {
136+
newAction.name += " (1)";
137+
}
138+
} else {
139+
// Use time as name
140+
newAction.name = new Date().toLocaleString();
141+
}
142+
const key = addon.api.actionManager.updateAction(newAction);
143+
updateUI();
144+
await editAndUpdate(key);
145+
});
146+
120147
doc
121148
.querySelector(`#${config.addonRef}-action-export`)
122149
?.addEventListener("command", async (e) => {
@@ -154,7 +181,7 @@ function getRowData(index: number) {
154181
return {
155182
event: getString(`prefs-action-event-${ActionEventTypes[action.event]}`),
156183
operation: getString(
157-
`prefs-action-operation-${ActionOperationTypes[action.operation]}`,
184+
`prefs-action-operation-${ActionOperationTypes[action.operation]}`
158185
),
159186
data: action.data,
160187
shortcut: action.shortcut,
@@ -172,3 +199,11 @@ function getSelection() {
172199
const keys = addon.data.actions.cachedKeys;
173200
return Array.from(indices).map((i) => keys[i]);
174201
}
202+
203+
function setButtonDisabled(button: XUL.Button, disabled: boolean = true) {
204+
if (disabled) {
205+
button.setAttribute("disabled", "true");
206+
} else {
207+
button.removeAttribute("disabled");
208+
}
209+
}

0 commit comments

Comments
 (0)