diff --git a/waterfox/browser/components/sidebar/_locales/en/messages.json b/waterfox/browser/components/sidebar/_locales/en/messages.json
index c902600bec85d..c9071c5bc71fe 100644
--- a/waterfox/browser/components/sidebar/_locales/en/messages.json
+++ b/waterfox/browser/components/sidebar/_locales/en/messages.json
@@ -614,6 +614,7 @@
"config_groupTabTemporaryStateForNewTabsFromBookmarks_label": { "message": "For tabs opened from bookmarks:" },
"config_groupTabTemporaryStateForNewTabsFromOthers_label": { "message": "For tabs opened at a time from non-bookmark triggers:" },
"config_groupTabTemporaryStateForOrphanedTabs_label": { "message": "Opened to replace a closed parent tab:" },
+ "config_groupTabTemporaryStateForAPI_label": { "message": "Opened to group tabs from other addons via API:" },
"config_inheritContextualIdentityToChildTabMode_label": { "message": "Container:" },
"config_inheritContextualIdentityToChildTabMode_default": { "message": "(no control)" },
@@ -730,6 +731,7 @@
"config_insertDroppedTabsAt_first": { "message": "The top of the tree (next to the parent)" },
"config_insertDroppedTabsAt_end": { "message": "The end of the tree" },
+ "config_autoCreateFolderForBookmarksFromTree_label": { "message": "Auto-group bookmark items under a folder when they are guessed to be created from multiple tree item tabs (ex. drag and drop of multiselected tabs)" },
"config_autoExpandOnLongHoverDelay_before": { "message": "Expand collapsed tree with hovering over " },
"config_autoExpandOnLongHoverDelay_after": { "message": "msec while dragging" },
"config_autoExpandOnLongHoverRestoreIniitalState_label": { "message": "Collapse such trees automatically after a drag action is finished" },
@@ -892,6 +894,7 @@
"tabContextMenu_shareTabURL_more_label": { "message": "More…" },
+ "tabContextMenu_closeDuplicatedTabs_label": { "message": "Close D&uplicate Tabs" },
"tabContextMenu_closeMultipleTabs_label": { "message": "Close &Multiple Tabs" },
"tabContextMenu_closeTabsToTop_label": { "message": "Close Tabs to the &Top" },
"tabContextMenu_closeTabsToBottom_label": { "message": "Close Tabs to the &Bottom" },
diff --git a/waterfox/browser/components/sidebar/_locales/ja/messages.json b/waterfox/browser/components/sidebar/_locales/ja/messages.json
index af98063c97a27..b923dca2c73aa 100644
--- a/waterfox/browser/components/sidebar/_locales/ja/messages.json
+++ b/waterfox/browser/components/sidebar/_locales/ja/messages.json
@@ -611,6 +611,7 @@
"config_groupTabTemporaryStateForNewTabsFromBookmarks_label": { "message": "ブックマークから開かれたタブをグループ化した場合:" },
"config_groupTabTemporaryStateForNewTabsFromOthers_label": { "message": "ブックマーク以外から連続して開かれたタブをグループ化した場合:" },
"config_groupTabTemporaryStateForOrphanedTabs_label": { "message": "閉じられた親タブを置き換える形で開かれた場合:" },
+ "config_groupTabTemporaryStateForAPI_label": { "message": "他の拡張機能からAPI経由でタブをグループ化した場合:" },
"config_inheritContextualIdentityToChildTabMode_label": { "message": "コンテナー:" },
"config_inheritContextualIdentityToChildTabMode_default": { "message": "(制御しない)" },
@@ -727,6 +728,7 @@
"config_insertDroppedTabsAt_first": { "message": "ツリーの先頭(親タブの隣)" },
"config_insertDroppedTabsAt_end": { "message": "ツリーの末尾" },
+ "config_autoCreateFolderForBookmarksFromTree_label": { "message": "ツリー構造を持ったタブを複数選択してのドラッグ&ドロップなどの操作で複数のブックマークが作成されたと推測できる場合に、作成されたブックマークを自動的に1つのフォルダーにまとめる" },
"config_autoExpandOnLongHoverDelay_before": { "message": "ドラッグ操作中、折りたたまれたツリーの上で" },
"config_autoExpandOnLongHoverDelay_after": { "message": "ミリ秒以上経過したらツリーを展開する" },
"config_autoExpandOnLongHoverRestoreIniitalState_label": { "message": "ドラッグ操作の終了後、ツリーを折りたたんだ状態に自動的に戻す" },
@@ -888,6 +890,7 @@
"tabContextMenu_shareTabURL_label": { "message": "共有(&H)" },
"tabContextMenu_shareTabURL_more_label": { "message": "その他..." },
+ "tabContextMenu_closeDuplicatedTabs_label": { "message": "重複タブを閉じる(&U)" },
"tabContextMenu_closeMultipleTabs_label": { "message": "複数のタブを閉じる(&M)" },
"tabContextMenu_closeTabsToTop_label": { "message": "上のタブをすべて閉じる(&T)" },
"tabContextMenu_closeTabsToBottom_label": { "message": "下のタブをすべて閉じる(&B)" },
diff --git a/waterfox/browser/components/sidebar/background/api-tabs-listener.js b/waterfox/browser/components/sidebar/background/api-tabs-listener.js
index 8910e4241d8e0..b7a539d522392 100644
--- a/waterfox/browser/components/sidebar/background/api-tabs-listener.js
+++ b/waterfox/browser/components/sidebar/background/api-tabs-listener.js
@@ -138,15 +138,12 @@ async function onActivated(activeInfo) {
try {
const win = Window.init(activeInfo.windowId);
- const byInternalOperation = win.internalFocusCount > 0;
- if (byInternalOperation)
- win.internalFocusCount--;
- const byMouseOperation = win.internalByMouseFocusCount > 0;
- if (byMouseOperation)
- win.internalByMouseFocusCount--;
- const silently = win.internalSilentlyFocusCount > 0;
- if (silently)
- win.internalSilentlyFocusCount--;
+ const byInternalOperation = win.internallyFocusingTabs.has(activeInfo.tabId);
+ win.internallyFocusingTabs.delete(activeInfo.tabId);
+ const byMouseOperation = win.internallyFocusingByMouseTabs.has(activeInfo.tabId);
+ win.internallyFocusingByMouseTabs.delete(activeInfo.tabId);
+ const silently = win.internallyFocusingSilentlyTabs.has(activeInfo.tabId);
+ win.internallyFocusingSilentlyTabs.delete(activeInfo.tabId);
const byTabDuplication = parseInt(win.duplicatingTabsCount) > 0;
if (!Tab.isTracked(activeInfo.tabId))
@@ -272,6 +269,7 @@ async function onUpdated(tabId, changeInfo, tab) {
}
}
/*
+ Workaround for Firefox 130 and olders.
Updated openerTabId is not notified via tabs.onUpdated due to
https://bugzilla.mozilla.org/show_bug.cgi?id=1409262 , so it can be
notified with delay as a part of the complete tabs.Tab object,
@@ -284,8 +282,11 @@ async function onUpdated(tabId, changeInfo, tab) {
continue;
if ('key' in updatedTab)
oldState[key] = updatedTab[key];
- if (key == 'openerTabId')
+ if (key == 'openerTabId') {
+ if (changeInfo.openerTabId == updatedTab.openerTabId) // already processed
+ continue;
log(`openerTabId of ${tabId} is changed by someone (notified via changeInfo)!: ${updatedTab.openerTabId} (original) => ${changeInfo[key]} (changed by someone)`, configs.debug && new Error().stack);
+ }
updatedTab[key] = changeInfo[key];
}
if (changeInfo.url ||
@@ -788,10 +789,8 @@ async function onNewTabTracked(tab, info) {
}
tab.$TST.memorizeNeighbors('newly tracked');
- if (tab.$TST.unsafePreviousTab)
- tab.$TST.unsafePreviousTab.$TST.memorizeNeighbors('unsafePreviousTab');
- if (tab.$TST.unsafeNextTab)
- tab.$TST.unsafeNextTab.$TST.memorizeNeighbors('unsafeNextTab');
+ tab.$TST.unsafePreviousTab?.$TST?.memorizeNeighbors('unsafePreviousTab');
+ tab.$TST.unsafeNextTab?.$TST?.memorizeNeighbors('unsafeNextTab');
Tree.onAttached.removeListener(onTreeModified);
metric.add('Tree.onAttached proceeded');
@@ -839,11 +838,18 @@ async function onRemoved(tabId, removeInfo) {
log('tabs.onRemoved: ', tabId, removeInfo);
const win = Window.init(removeInfo.windowId);
const byInternalOperation = win.internalClosingTabs.has(tabId);
- if (byInternalOperation)
- win.internalClosingTabs.delete(tabId);
const preventEntireTreeBehavior = win.keepDescendantsTabs.has(tabId);
- if (preventEntireTreeBehavior)
- win.keepDescendantsTabs.delete(tabId);
+
+ win.internalMovingTabs.delete(tabId);
+ win.alreadyMovedTabs.delete(tabId);
+ win.internalClosingTabs.delete(tabId);
+ win.keepDescendantsTabs.delete(tabId);
+ win.highlightingTabs.delete(tabId);
+ win.tabsToBeHighlightedAlone.delete(tabId);
+
+ win.internallyFocusingTabs.delete(tabId);
+ win.internallyFocusingByMouseTabs.delete(tabId);
+ win.internallyFocusingSilentlyTabs.delete(tabId);
if (Tab.needToWaitTracked(removeInfo.windowId))
await Tab.waitUntilTrackedAll(removeInfo.windowId);
@@ -925,9 +931,7 @@ async function onRemoved(tabId, removeInfo) {
oldTab.$TST.destroy();
for (const tab of nearestTabs) {
- if (!tab || !tab.$TST)
- continue;
- tab.$TST.memorizeNeighbors('neighbor of closed tab');
+ tab?.$TST?.memorizeNeighbors('neighbor of closed tab');
}
onCompleted();
@@ -956,7 +960,10 @@ async function onMoved(tabId, moveInfo) {
// and other fixup operations around tabs moved by foreign triggers, on such
// cases. Don't mind, the tab will be rearranged again by delayed
// TabsMove.syncTabsPositionToApiTabs() anyway!
- const maybeInternalOperation = win.internalMovingTabs.has(tabId);
+ const internalExpectedIndex = win.internalMovingTabs.get(tabId);
+ const maybeInternalOperation = internalExpectedIndex < 0 || internalExpectedIndex == moveInfo.toIndex;
+ if (maybeInternalOperation)
+ log(`tabs.onMoved: ${tabId} is detected as moved internally`);
if (!Tab.isTracked(tabId))
await Tab.waitUntilTracked(tabId);
@@ -979,7 +986,7 @@ async function onMoved(tabId, moveInfo) {
do following processes after the tab is completely pinned. */
const movedTab = Tab.get(tabId);
if (!movedTab) {
- if (maybeInternalOperation)
+ if (win.internalMovingTabs.has(tabId))
win.internalMovingTabs.delete(tabId);
completelyMoved();
warnTabDestroyedWhileWaiting(tabId, movedTab);
@@ -989,22 +996,21 @@ async function onMoved(tabId, moveInfo) {
let oldPreviousTab = movedTab.hidden ? movedTab.$TST.unsafePreviousTab : movedTab.$TST.previousTab;
let oldNextTab = movedTab.hidden ? movedTab.$TST.unsafeNextTab : movedTab.$TST.nextTab;
if (movedTab.index != moveInfo.toIndex ||
- (oldPreviousTab && oldPreviousTab.index == movedTab.index - 1) ||
- (oldNextTab && oldNextTab.index == movedTab.index + 1)) {
+ (oldPreviousTab?.index == movedTab.index - 1) ||
+ (oldNextTab?.index == movedTab.index + 1)) {
// already moved
oldPreviousTab = Tab.getTabAt(moveInfo.windowId, moveInfo.toIndex < moveInfo.fromIndex ? moveInfo.fromIndex : moveInfo.fromIndex - 1);
oldNextTab = Tab.getTabAt(moveInfo.windowId, moveInfo.toIndex < moveInfo.fromIndex ? moveInfo.fromIndex + 1 : moveInfo.fromIndex);
- if (oldPreviousTab && oldPreviousTab.id == movedTab.id)
+ if (oldPreviousTab?.id == movedTab.id)
oldPreviousTab = Tab.getTabAt(moveInfo.windowId, moveInfo.toIndex < moveInfo.fromIndex ? moveInfo.fromIndex - 1 : moveInfo.fromIndex - 2);
- if (oldNextTab && oldNextTab.id == movedTab.id)
+ if (oldNextTab?.id == movedTab.id)
oldNextTab = Tab.getTabAt(moveInfo.windowId, moveInfo.toIndex < moveInfo.fromIndex ? moveInfo.fromIndex : moveInfo.fromIndex - 1);
}
- let alreadyMoved = false;
- if (win.alreadyMovedTabs.has(tabId)) {
+ const expectedIndex = win.alreadyMovedTabs.get(tabId);
+ const alreadyMoved = expectedIndex < 0 || expectedIndex == moveInfo.toIndex;
+ if (win.alreadyMovedTabs.has(tabId))
win.alreadyMovedTabs.delete(tabId);
- alreadyMoved = true;
- }
const extendedMoveInfo = {
...moveInfo,
@@ -1012,7 +1018,10 @@ async function onMoved(tabId, moveInfo) {
alreadyMoved,
oldPreviousTab,
oldNextTab,
- isSubstantiallyMoved: movedTab.$TST.isSubstantiallyMoved
+ // Multiselected tabs can be moved together in bulk, by drag and drop
+ // in the horizontal tab bar, or addons like
+ // https://addons.mozilla.org/firefox/addon/move-tab-hotkeys/
+ movedInBulk: !maybeInternalOperation && (movedTab.$TST.multiselected || movedTab.$TST.movedInBulk),
};
log('tabs.onMoved: ', movedTab, extendedMoveInfo);
@@ -1060,20 +1069,16 @@ async function onMoved(tabId, moveInfo) {
nextTabId: nextTab && nextTab.id
});
}
- if (maybeInternalOperation)
+ if (win.internalMovingTabs.has(tabId))
win.internalMovingTabs.delete(tabId);
completelyMoved();
movedTab.$TST.memorizeNeighbors('moved');
- if (movedTab.$TST.unsafePreviousTab)
- movedTab.$TST.unsafePreviousTab.$TST.memorizeNeighbors('unsafePreviousTab');
- if (movedTab.$TST.unsafeNextTab)
- movedTab.$TST.unsafeNextTab.$TST.memorizeNeighbors('unsafeNextTab');
-
- if (oldPreviousTab)
- oldPreviousTab.$TST.memorizeNeighbors('oldPreviousTab');
- if (oldNextTab)
- oldNextTab.$TST.memorizeNeighbors('oldNextTab');
+ movedTab.$TST.unsafePreviousTab?.$TST?.memorizeNeighbors('unsafePreviousTab');
+ movedTab.$TST.unsafeNextTab?.$TST?.memorizeNeighbors('unsafeNextTab');
+
+ oldPreviousTab?.$TST?.memorizeNeighbors('oldPreviousTab');
+ oldNextTab?.$TST?.memorizeNeighbors('oldNextTab');
}
catch(e) {
console.log(e);
diff --git a/waterfox/browser/components/sidebar/background/background-cache.js b/waterfox/browser/components/sidebar/background/background-cache.js
index bbbf21c92c765..1561ce5eff62b 100644
--- a/waterfox/browser/components/sidebar/background/background-cache.js
+++ b/waterfox/browser/components/sidebar/background/background-cache.js
@@ -339,7 +339,8 @@ async function updateWindowCache(owner, key, value) {
});
return;
}
- catch(_error) {
+ catch(error) {
+ console.log(`BackgroundCache.updateWindowCache for ${owner.windowId}/${key} failed: `, error.message, error.stack, error);
}
}
@@ -372,7 +373,8 @@ async function getWindowCache(owner, key) {
});
return value;
}
- catch(_error) {
+ catch(error) {
+ console.log(`BackgroundCache.getWindowCache for ${owner.windowId}/${key} failed: `, error.message, error.stack, error);
}
}
diff --git a/waterfox/browser/components/sidebar/background/background.js b/waterfox/browser/components/sidebar/background/background.js
index 52bca30f1033d..419cd6d4ab7f9 100644
--- a/waterfox/browser/components/sidebar/background/background.js
+++ b/waterfox/browser/components/sidebar/background/background.js
@@ -310,12 +310,23 @@ async function rebuildAll(windows) {
try {
log(`build tabs for ${win.id} from scratch`);
Window.init(win.id);
+ const promises = [];
for (let tab of win.tabs) {
tab = Tab.get(tab.id);
tab.$TST.clear(); // clear dirty restored states
- TabsUpdate.updateTab(tab, tab, { forceApply: true });
+ promises.push(
+ tab.$TST.getPermanentStates()
+ .then(states => {
+ tab.$TST.states = new Set(states);
+ })
+ .catch(console.error)
+ .then(() => {
+ TabsUpdate.updateTab(tab, tab, { forceApply: true });
+ })
+ );
tryStartHandleAccelKeyOnTab(tab);
}
+ await Promise.all(promises);
}
catch(e) {
log(`failed to build tabs for ${win.id}`, e);
@@ -774,7 +785,7 @@ Tab.onTabInternallyMoved.addListener((tab, info = {}) => {
});
Tab.onMoved.addListener((tab, moveInfo) => {
- if (!moveInfo.isSubstantiallyMoved)
+ if (moveInfo.movedInBulk)
return;
reserveToUpdateInsertionPosition([
tab,
diff --git a/waterfox/browser/components/sidebar/background/browser-action-menu.js b/waterfox/browser/components/sidebar/background/browser-action-menu.js
index 72cf03cbdfbe4..af224b7f61135 100644
--- a/waterfox/browser/components/sidebar/background/browser-action-menu.js
+++ b/waterfox/browser/components/sidebar/background/browser-action-menu.js
@@ -1298,8 +1298,32 @@ const mItems = [
type: 'radio'
}
]
- }
- ]
+ },
+ {
+ title: indent() + browser.i18n.getMessage('config_groupTabTemporaryStateForAPI_label'),
+ expert: true,
+ children: [
+ {
+ title: browser.i18n.getMessage('config_groupTabTemporaryState_option_default'),
+ key: 'groupTabTemporaryStateForAPI',
+ value: Constants.kGROUP_TAB_TEMPORARY_STATE_NOTHING,
+ type: 'radio'
+ },
+ {
+ title: `${browser.i18n.getMessage('config_groupTabTemporaryState_option_checked_before')}${browser.i18n.getMessage('groupTab_temporary_label')}${browser.i18n.getMessage('config_groupTabTemporaryState_option_checked_after')}`,
+ key: 'groupTabTemporaryStateForAPI',
+ value: Constants.kGROUP_TAB_TEMPORARY_STATE_PASSIVE,
+ type: 'radio'
+ },
+ {
+ title: `${browser.i18n.getMessage('config_groupTabTemporaryState_option_checked_before')}${browser.i18n.getMessage('groupTab_temporaryAggressive_label')}${browser.i18n.getMessage('config_groupTabTemporaryState_option_checked_after')}`,
+ key: 'groupTabTemporaryStateForAPI',
+ value: Constants.kGROUP_TAB_TEMPORARY_STATE_AGGRESSIVE,
+ type: 'radio'
+ },
+ ],
+ },
+ ],
},
{
title: browser.i18n.getMessage('config_treeBehavior_caption'),
diff --git a/waterfox/browser/components/sidebar/background/handle-misc.js b/waterfox/browser/components/sidebar/background/handle-misc.js
index 77e8f8eff7ab9..b0de426fde4cb 100644
--- a/waterfox/browser/components/sidebar/background/handle-misc.js
+++ b/waterfox/browser/components/sidebar/background/handle-misc.js
@@ -264,7 +264,7 @@ async function onShortcutCommand(command) {
return;
case 'simulateUpOnTree':
- if (SidebarConnection.isSidebarOpen(activeTab.windowId)) {
+ if (SidebarConnection.isOpen(activeTab.windowId)) {
if (configs.faviconizePinnedTabs &&
(activeTab.pinned ||
activeTab == Tab.getFirstNormalTab(activeTab.windowId))) {
@@ -291,7 +291,7 @@ async function onShortcutCommand(command) {
}
return;
case 'simulateDownOnTree':
- if (SidebarConnection.isSidebarOpen(activeTab.windowId)) {
+ if (SidebarConnection.isOpen(activeTab.windowId)) {
if (configs.faviconizePinnedTabs &&
activeTab.pinned) {
const nextActiveId = await browser.runtime.sendMessage({
@@ -317,7 +317,7 @@ async function onShortcutCommand(command) {
}
return;
case 'simulateLeftOnTree':
- if (SidebarConnection.isSidebarOpen(activeTab.windowId)) {
+ if (SidebarConnection.isOpen(activeTab.windowId)) {
if (configs.faviconizePinnedTabs &&
activeTab.pinned) {
const nextActiveId = await browser.runtime.sendMessage({
@@ -342,7 +342,7 @@ async function onShortcutCommand(command) {
}
return;
case 'simulateRightOnTree':
- if (SidebarConnection.isSidebarOpen(activeTab.windowId)) {
+ if (SidebarConnection.isOpen(activeTab.windowId)) {
if (configs.faviconizePinnedTabs &&
activeTab.pinned) {
const nextActiveId = await browser.runtime.sendMessage({
@@ -453,12 +453,14 @@ async function onShortcutCommand(command) {
function focusPrevious(activeTab) {
const nextActive = activeTab.$TST.nearestVisiblePrecedingTab ||
+ (!SidebarConnection.isOpen(activeTab.windowId) && activeTab.$TST.previousTab) ||
Tab.getLastVisibleTab(activeTab.windowId);
TabsInternalOperation.activateTab(nextActive);
}
function focusPreviousSilently(activeTab) {
const nextActive = activeTab.$TST.nearestVisiblePrecedingTab ||
+ (!SidebarConnection.isOpen(activeTab.windowId) && activeTab.$TST.previousTab) ||
Tab.getLastVisibleTab(activeTab.windowId);
TabsInternalOperation.activateTab(nextActive, {
silently: true,
@@ -466,13 +468,15 @@ function focusPreviousSilently(activeTab) {
}
function focusNext(activeTab) {
- const nextActive = activeTab.$TST.nextVisibleTab ||
+ const nextActive = activeTab.$TST.nearestVisibleFollowingTab ||
+ (!SidebarConnection.isOpen(activeTab.windowId) && activeTab.$TST.nextTab) ||
Tab.getFirstVisibleTab(activeTab.windowId);
TabsInternalOperation.activateTab(nextActive);
}
function focusNextSilently(activeTab) {
const nextActive = activeTab.$TST.nearestVisibleFollowingTab ||
+ (!SidebarConnection.isOpen(activeTab.windowId) && activeTab.$TST.nextTab) ||
Tab.getFirstVisibleTab(activeTab.windowId);
TabsInternalOperation.activateTab(nextActive, {
silently: true,
@@ -953,9 +957,27 @@ function onMessageExternal(message, sender) {
case TSTAPI.kGROUP_TABS:
return (async () => {
const tabs = await TSTAPI.getTargetTabs(message, sender);
+ const temporaryStateParams = (message.temporary && !message.temporaryAggressive) ?
+ {
+ temporary: true,
+ temporaryAggressive: false,
+ } :
+ (!message.temporary && message.temporaryAggressive) ?
+ {
+ temporary: false,
+ temporaryAggressive: true,
+ } :
+ (message.temporaryAggressive === false && message.temporary === false) ?
+ {
+ temporary: false,
+ temporaryAggressive: false,
+ } :
+ {};
const tab = await TabsGroup.groupTabs(Array.from(tabs), {
title: message.title,
broadcast: true,
+ ...TabsGroup.temporaryStateParams(configs.groupTabTemporaryStateForAPI),
+ ...temporaryStateParams,
});
if (!tab)
return null;
diff --git a/waterfox/browser/components/sidebar/background/handle-moved-tabs.js b/waterfox/browser/components/sidebar/background/handle-moved-tabs.js
index b819f6134c693..9060dc748ee64 100644
--- a/waterfox/browser/components/sidebar/background/handle-moved-tabs.js
+++ b/waterfox/browser/components/sidebar/background/handle-moved-tabs.js
@@ -52,8 +52,7 @@ Tab.onCreated.addListener((tab, info = {}) => {
info.movedBySelfWhileCreation) &&
(tab.$TST.nearestCompletelyOpenedNormalFollowingTab ||
tab.$TST.nearestCompletelyOpenedNormalPrecedingTab ||
- (info.treeForActionDetection &&
- info.treeForActionDetection.target &&
+ (info.treeForActionDetection?.target &&
(info.treeForActionDetection.target.next ||
info.treeForActionDetection.target.previous)))) {
tryFixupTreeForInsertedTab(tab, {
@@ -73,21 +72,21 @@ Tab.onMoving.addListener((tab, moveInfo) => {
const win = TabsStore.windows.get(tab.windowId);
const isNewlyOpenedTab = win.openingTabs.has(tab.id);
const positionControlled = configs.insertNewChildAt != Constants.kINSERT_NO_CONTROL;
- if (isNewlyOpenedTab &&
- !moveInfo.byInternalOperation &&
- !moveInfo.alreadyMoved &&
- !moveInfo.isSubstantiallyMoved &&
- positionControlled) {
- const opener = tab.$TST.openerTab;
- // if there is no valid opener, it can be a restored initial tab in a restored window
- // and can be just moved as a part of window restoration process.
- if (opener) {
- log('onTabMove for new child tab: move back '+moveInfo.toIndex+' => '+moveInfo.fromIndex);
- moveBack(tab, moveInfo);
- return false;
- }
- }
- return true;
+ if (!isNewlyOpenedTab ||
+ !positionControlled ||
+ moveInfo.byInternalOperation ||
+ moveInfo.alreadyMoved ||
+ !moveInfo.movedInBulk)
+ return true;
+
+ // if there is no valid opener, it can be a restored initial tab in a restored window
+ // and can be just moved as a part of window restoration process.
+ if (!tab.$TST.openerTab)
+ return true;
+
+ log('onTabMove for new child tab: move back '+moveInfo.toIndex+' => '+moveInfo.fromIndex);
+ moveBack(tab, moveInfo);
+ return false;
});
async function tryFixupTreeForInsertedTab(tab, moveInfo = {}) {
@@ -96,6 +95,7 @@ async function tryFixupTreeForInsertedTab(tab, moveInfo = {}) {
...moveInfo,
});
log('tryFixupTreeForInsertedTab ', {
+ tab: tab.id,
parentTabOperationBehavior,
moveInfo,
childIds: tab.$TST.childIds,
@@ -118,9 +118,10 @@ async function tryFixupTreeForInsertedTab(tab, moveInfo = {}) {
broadcast: true
});
}
- await Tree.detachTab(tab, {
- broadcast: true
- });
+ if (tab.$TST.parentId)
+ await Tree.detachTab(tab, {
+ broadcast: true
+ });
// Pinned tab is moved at first, so Tab.onPinned handler cannot know tree information
// before the pinned tab was moved. Thus we cache tree information for the handler.
wait(100).then(() => {
@@ -134,11 +135,19 @@ async function tryFixupTreeForInsertedTab(tab, moveInfo = {}) {
isMovingByShortcut: mMaybeTabMovingByShortcut,
...moveInfo,
});
+ log(' => action: ', action);
if (!action.action) {
log('no action');
return;
}
+ // When multiple tabs are moved at once by outside of TST (e.g. moving of multiselected tabs)
+ // Tree.detectTabActionFromNewPosition() may be called for other tabs asynchronously
+ // before this operation finishes. Thus we need to memorize the calculated "parent"
+ // and Tree.detectTabActionFromNewPosition() will use it.
+ if (action.parent)
+ tab.$TST.temporaryMetadata.set('goingToBeAttachedTo', action.parent);
+
// notify event to helper addons with action and allow or deny
const cache = {};
const allowed = await TSTAPI.tryOperationAllowed(
@@ -155,22 +164,26 @@ async function tryFixupTreeForInsertedTab(tab, moveInfo = {}) {
{ tabProperties: ['tab', 'parent', 'insertBefore', 'insertAfter'], cache }
);
TSTAPI.clearCache(cache);
+
if (!allowed) {
log('no action - canceled by a helper addon');
- return;
}
+ else {
+ log('action: ', action);
+ switch (action.action) {
+ case 'invalid':
+ moveBack(tab, moveInfo);
+ break;
- log('action: ', action);
- switch (action.action) {
- case 'invalid':
- moveBack(tab, moveInfo);
- return;
-
- default:
- log('tryFixupTreeForInsertedTab: apply action for unattached tab: ', tab, action);
- await action.apply();
- return;
+ default:
+ log('tryFixupTreeForInsertedTab: apply action for unattached tab: ', tab, action);
+ await action.apply();
+ break;
+ }
}
+
+ if (tab.$TST.temporaryMetadata.get('goingToBeAttachedTo') == action.parent)
+ tab.$TST.temporaryMetadata.delete('goingToBeAttachedTo');
}
function reserveToEnsureRootTabVisible(tab) {
@@ -196,14 +209,14 @@ function reserveToEnsureRootTabVisible(tab) {
reserveToEnsureRootTabVisible.tabIds = new Set();
Tab.onMoved.addListener((tab, moveInfo = {}) => {
- if (!moveInfo.byInternalOperation &&
- !moveInfo.isSubstantiallyMoved &&
- !tab.$TST.duplicating) {
- log('process moved tab');
- tryFixupTreeForInsertedTab(tab, moveInfo);
+ if (moveInfo.byInternalOperation ||
+ !moveInfo.movedInBulk ||
+ tab.$TST.duplicating) {
+ log('internal move');
}
else {
- log('internal move');
+ log('process moved tab');
+ tryFixupTreeForInsertedTab(tab, moveInfo);
}
reserveToEnsureRootTabVisible(tab);
});
@@ -255,7 +268,8 @@ function moveBack(tab, moveInfo) {
log('Move back tab from unexpected move: ', dumpTab(tab), moveInfo);
const id = tab.id;
const win = TabsStore.windows.get(tab.windowId);
- win.internalMovingTabs.add(id);
+ const index = moveInfo.fromIndex;
+ win.internalMovingTabs.set(id, index);
logApiTabs(`handle-moved-tabs:moveBack: browser.tabs.move() `, tab.id, {
windowId: moveInfo.windowId,
index: moveInfo.fromIndex
@@ -266,7 +280,7 @@ function moveBack(tab, moveInfo) {
windowId: moveInfo.windowId,
index: moveInfo.fromIndex
}).catch(ApiTabs.createErrorHandler(e => {
- if (win.internalMovingTabs.has(id))
+ if (win.internalMovingTabs.get(id) == index)
win.internalMovingTabs.delete(id);
ApiTabs.handleMissingTabError(e);
}));
diff --git a/waterfox/browser/components/sidebar/background/handle-tab-bunches.js b/waterfox/browser/components/sidebar/background/handle-tab-bunches.js
index edb58d022b7dd..52483a8fc0a37 100644
--- a/waterfox/browser/components/sidebar/background/handle-tab-bunches.js
+++ b/waterfox/browser/components/sidebar/background/handle-tab-bunches.js
@@ -104,9 +104,14 @@ async function tryDetectTabBunches(win) {
}
if (tabReferences.length > 1) {
- for (const tabReference of tabReferences) {
- Tab.get(tabReference.id).$TST.temporaryMetadata.set('openedWithOthers', true);
- }
+ await Promise.all(tabReferences.map(tabReference => {
+ const tab = Tab.get(tabReference.id);
+ tab.$TST.temporaryMetadata.set('openedWithOthers', true);
+ // We need to wait until all tabs are handlede completely.
+ // Otherwise `tab.$TST.needToBeGroupedSiblings` may contain unrelated tabs
+ // (tabs opened from any other parent tab) unexpectedly.
+ return tab.$TST.opened;
+ }));
}
if (areTabsFromOtherDeviceWithInsertAfterCurrent(tabReferences) &&
diff --git a/waterfox/browser/components/sidebar/background/prefs.js b/waterfox/browser/components/sidebar/background/prefs.js
index 780dccd36de83..7ae0b866800aa 100644
--- a/waterfox/browser/components/sidebar/background/prefs.js
+++ b/waterfox/browser/components/sidebar/background/prefs.js
@@ -11,8 +11,6 @@ import {
} from '/common/common.js';
import * as Constants from '/common/constants.js';
-import Tab from '/common/Tab.js';
-
export const onChanged = new EventListenerManager();
if (Constants.IS_BACKGROUND) {
@@ -100,16 +98,3 @@ configs.$addObserver(async name => {
mNamesSyncToChrome.delete(name);
});
});
-
-browser.waterfoxBridge.onHoverPreviewChanged.addListener(enabled => {
- if (enabled) {
- for (const tab of Tab.getAllTabs()) {
- tab.$TST.registerTooltipText(browser.runtime.id, '', true);
- }
- }
- else {
- for (const tab of Tab.getAllTabs()) {
- tab.$TST.unregisterTooltipText(browser.runtime.id);
- }
- }
-});
diff --git a/waterfox/browser/components/sidebar/background/successor-tab.js b/waterfox/browser/components/sidebar/background/successor-tab.js
index 3b80bab5d50d7..b931a5a735cc3 100644
--- a/waterfox/browser/components/sidebar/background/successor-tab.js
+++ b/waterfox/browser/components/sidebar/background/successor-tab.js
@@ -14,6 +14,7 @@ import {
import * as ApiTabs from '/common/api-tabs.js';
import * as Constants from '/common/constants.js';
import * as SidebarConnection from '/common/sidebar-connection.js';
+import * as TabsInternalOperation from '/common/tabs-internal-operation.js';
import * as TabsStore from '/common/tabs-store.js';
import * as TreeBehavior from '/common/tree-behavior.js';
@@ -42,6 +43,17 @@ browser.tabs.onUpdated.addListener((tabId, updateInfo, _tab) => {
// properties: ['successorTabId'],
});
+TabsInternalOperation.onBeforeTabsRemove.addListener(async tabs => {
+ let activeTab = null;
+ const tabIds = tabs.map(tab => {
+ if (tab.active)
+ activeTab = tab;
+ return tab.id;
+ });
+ if (activeTab)
+ await updateInternal(activeTab.id, tabIds);
+});
+
function setSuccessor(tabId, successorTabId = -1) {
const tab = Tab.get(tabId);
const successorTab = Tab.get(successorTabId);
@@ -133,7 +145,7 @@ function update(tabId) {
}
}, 0);
}
-async function updateInternal(tabId) {
+async function updateInternal(tabId, excludeTabIds = []) {
// tabs.onActivated can be notified before the tab is completely tracked...
await Tab.waitUntilTracked(tabId);
const tab = Tab.get(tabId);
@@ -190,6 +202,15 @@ async function updateInternal(tabId) {
let successor = null;
if (renewedTab.active) {
log('it is active, so reset successor');
+ const excludeTabIdsSet = new Set(excludeTabIds);
+ const findSuccessor = (...candidates) => {
+ for (const candidate of candidates) {
+ if (!excludeTabIdsSet.has(candidate?.id) &&
+ candidate)
+ return candidate;
+ }
+ return null;
+ };
if (configs.successorTabControlLevel == Constants.kSUCCESSOR_TAB_CONTROL_IN_TREE) {
const closeParentBehavior = TreeBehavior.getParentTabOperationBehavior(tab, {
context: Constants.kPARENT_TAB_OPERATION_CONTEXT_CLOSE,
@@ -203,7 +224,11 @@ async function updateInternal(tabId) {
const firstChild = collapsedChildSuccessorAllowed ? tab.$TST.firstChild : tab.$TST.firstVisibleChild;
const nextVisibleSibling = tab.$TST.nextVisibleSiblingTab;
const nearestVisiblePreceding = tab.$TST.nearestVisiblePrecedingTab;
- successor = firstChild || nextVisibleSibling || nearestVisiblePreceding;
+ successor = findSuccessor(
+ firstChild,
+ nextVisibleSibling,
+ nearestVisiblePreceding
+ );
log(` possible successor: ${dumpTab(tab)}: `, successor, {
closeParentBehavior,
collapsedChildSuccessorAllowed,
@@ -216,22 +241,29 @@ async function updateInternal(tabId) {
successor.discarded &&
configs.avoidDiscardedTabToBeActivatedIfPossible) {
log(` ${dumpTab(successor)} is discarded.`);
- successor = tab.$TST.nearestLoadedSiblingTab ||
- tab.$TST.nearestLoadedTabInTree ||
- tab.$TST.nearestLoadedTab ||
- successor;
+ successor = findSuccessor(
+ tab.$TST.nearestLoadedSiblingTab,
+ tab.$TST.nearestLoadedTabInTree,
+ tab.$TST.nearestLoadedTab,
+ successor
+ );
log(` => redirected successor is: ${dumpTab(successor)}`);
}
}
else {
- successor = tab.$TST.nearestVisibleFollowingTab || tab.$TST.nearestVisiblePrecedingTab;
+ successor = findSuccessor(
+ tab.$TST.nearestVisibleFollowingTab,
+ tab.$TST.nearestVisiblePrecedingTab
+ );
log(` possible successor: ${dumpTab(tab)}`);
if (successor &&
successor.discarded &&
configs.avoidDiscardedTabToBeActivatedIfPossible) {
log(` ${dumpTab(successor)} is discarded.`);
- successor = tab.$TST.nearestLoadedTab ||
- successor;
+ successor = findSuccessor(
+ tab.$TST.nearestLoadedTab,
+ successor
+ );
log(` => redirected successor is: ${dumpTab(successor)}`);
}
}
@@ -243,8 +275,7 @@ async function updateInternal(tabId) {
}
else {
log(` ${dumpTab(tab)} is out of control.`, {
- active: renewedTab.active,
- successor: successor && successor.id
+ active: renewedTab.active,
});
clearSuccessor(renewedTab.id);
}
diff --git a/waterfox/browser/components/sidebar/background/tab-context-menu.js b/waterfox/browser/components/sidebar/background/tab-context-menu.js
index 33ae764db4a1f..faeb564c9afd4 100644
--- a/waterfox/browser/components/sidebar/background/tab-context-menu.js
+++ b/waterfox/browser/components/sidebar/background/tab-context-menu.js
@@ -197,6 +197,9 @@ const mItemsById = {
title: browser.i18n.getMessage('tabContextMenu_close_label'),
titleMultiselected: browser.i18n.getMessage('tabContextMenu_close_label_multiselected')
},
+ 'context_closeDuplicatedTabs': {
+ title: browser.i18n.getMessage('tabContextMenu_closeDuplicatedTabs_label')
+ },
'context_closeMultipleTabs': {
title: browser.i18n.getMessage('tabContextMenu_closeMultipleTabs_label')
},
@@ -694,6 +697,7 @@ async function onShown(info, contextTab) {
const previousSiblingTab = contextTab && contextTab.$TST.previousSiblingTab;
const nextTab = contextTab && contextTab.$TST.nextTab;
const nextSiblingTab = contextTab && contextTab.$TST.nextSiblingTab;
+ const hasDuplicatedTabs = Tab.hasDuplicatedTabs(windowId);
const hasMultipleTabs = Tab.hasMultipleTabs(windowId);
const hasMultipleNormalTabs = Tab.hasMultipleTabs(windowId, { normal: true });
const multiselected = contextTab && contextTab.$TST.multiselected;
@@ -908,6 +912,11 @@ async function onShown(info, contextTab) {
multiselected
}) && modifiedItemsCount++;
+ updateItem('context_closeDuplicatedTabs', {
+ visible: emulate && !!contextTab,
+ enabled: hasDuplicatedTabs,
+ multiselected
+ }) && modifiedItemsCount++;
updateItem('context_closeMultipleTabs', {
visible: emulate && !!contextTab,
enabled: hasMultipleNormalTabs,
@@ -1251,6 +1260,28 @@ async function onClick(info, contextTab) {
case 'context_bookmarkSelected':
Commands.bookmarkTab(contextTab || activeTab);
break;
+ case 'context_closeDuplicatedTabs': {
+ const tabs = await browser.tabs.query({ windowId }).catch(ApiTabs.createErrorHandler());
+ tabs.sort((a, b) => b.lastAccessed - a.lastAccessed);
+ const tabKeys = new Set();
+ const closeTabs = [];
+ for (const tab of tabs) {
+ const key = `${tab.cookieStoreId}\n${tab.url}`;
+ if (tabKeys.has(key)) {
+ closeTabs.push(Tab.get(tab.id));
+ continue;
+ }
+ tabKeys.add(key);
+ }
+ const canceled = (await browser.runtime.sendMessage({
+ type: Constants.kCOMMAND_NOTIFY_TABS_CLOSING,
+ tabs: closeTabs.map(tab => tab.$TST.sanitized),
+ windowId,
+ }).catch(ApiTabs.createErrorHandler())) === false;
+ if (canceled)
+ break;
+ TabsInternalOperation.removeTabs(closeTabs);
+ } break;
case 'context_closeTabsToTheStart': {
const tabs = await browser.tabs.query({ windowId }).catch(ApiTabs.createErrorHandler());
const closeTabs = [];
diff --git a/waterfox/browser/components/sidebar/background/tabs-group.js b/waterfox/browser/components/sidebar/background/tabs-group.js
index 977594b7075db..cfdfc0601e2a3 100644
--- a/waterfox/browser/components/sidebar/background/tabs-group.js
+++ b/waterfox/browser/components/sidebar/background/tabs-group.js
@@ -30,7 +30,7 @@ function log(...args) {
internalLogger('background/tabs-group', ...args);
}
-export function makeGroupTabURI({ title, temporary, temporaryAggressive, openerTabId, aliasTabId } = {}) {
+export function makeGroupTabURI({ title, temporary, temporaryAggressive, openerTabId, aliasTabId, replacedParentCount } = {}) {
const url = new URL(Constants.kGROUP_TAB_URI);
if (title)
@@ -47,19 +47,31 @@ export function makeGroupTabURI({ title, temporary, temporaryAggressive, openerT
if (aliasTabId)
url.searchParams.set('aliasTabId', aliasTabId);
+ if (replacedParentCount)
+ url.searchParams.set('replacedParentCount', replacedParentCount);
+
return url.href;
}
export function temporaryStateParams(state) {
switch (state) {
case Constants.kGROUP_TAB_TEMPORARY_STATE_PASSIVE:
- return { temporary: true };
+ return {
+ temporary: true,
+ temporaryAggressive: false,
+ };
case Constants.kGROUP_TAB_TEMPORARY_STATE_AGGRESSIVE:
- return { temporaryAggressive: true };
+ return {
+ temporary: false,
+ temporaryAggressive: true,
+ };
default:
break;
}
- return {};
+ return {
+ temporary: false,
+ temporaryAggressive: false,
+ };
}
export async function groupTabs(tabs, { broadcast, parent, withDescendants, ...groupTabOptions } = {}) {
@@ -67,7 +79,7 @@ export async function groupTabs(tabs, { broadcast, parent, withDescendants, ...g
if (rootTabs.length <= 0)
return null;
- log('groupTabs: ', () => tabs.map(dumpTab));
+ log('groupTabs: ', () => tabs.map(dumpTab), { broadcast, parent, withDescendants });
const uri = makeGroupTabURI({
title: browser.i18n.getMessage('groupTab_label', rootTabs[0].title),
@@ -81,10 +93,19 @@ export async function groupTabs(tabs, { broadcast, parent, withDescendants, ...g
inBackground: true
});
- if (!withDescendants)
+ if (!withDescendants) {
+ const structure = TreeBehavior.getTreeStructureFromTabs(tabs);
+
await Tree.detachTabsFromTree(tabs, {
- broadcast: !!broadcast
+ broadcast: !!broadcast,
});
+
+ log('structure: ', structure);
+ await Tree.applyTreeStructureToTabs(tabs, structure, {
+ broadcast: !!broadcast,
+ });
+ }
+
await TabsMove.moveTabsAfter(tabs.slice(1), tabs[0], {
broadcast: !!broadcast
});
@@ -92,7 +113,7 @@ export async function groupTabs(tabs, { broadcast, parent, withDescendants, ...g
await Tree.attachTabTo(tab, groupTab, {
forceExpand: true, // this is required to avoid the group tab itself is active from active tab in collapsed tree
dontMove: true,
- broadcast: !!broadcast
+ broadcast: !!broadcast,
});
}
return groupTab;
@@ -163,7 +184,8 @@ export async function tryReplaceTabWithGroup(tab, { windowId, parent, children,
const firstChild = children[0];
const uri = makeGroupTabURI({
title: browser.i18n.getMessage('groupTab_label', firstChild.title),
- ...temporaryStateParams(configs.groupTabTemporaryStateForOrphanedTabs)
+ ...temporaryStateParams(configs.groupTabTemporaryStateForOrphanedTabs),
+ replacedParentCount: (tab?.$TST?.replacedParentGroupTabCount || 0) + 1,
});
const win = TabsStore.windows.get(windowId);
win.toBeOpenedTabsWithPositions++;
diff --git a/waterfox/browser/components/sidebar/background/tabs-move.js b/waterfox/browser/components/sidebar/background/tabs-move.js
index 1edb53a9c6432..1de08e9d9c119 100644
--- a/waterfox/browser/components/sidebar/background/tabs-move.js
+++ b/waterfox/browser/components/sidebar/background/tabs-move.js
@@ -14,7 +14,7 @@
* The Original Code is the Tree Style Tab.
*
* The Initial Developer of the Original Code is YUKI "Piro" Hiroshi.
- * Portions created by the Initial Developer are Copyright (C) 2011-2023
+ * Portions created by the Initial Developer are Copyright (C) 2011-2024
* the Initial Developer. All Rights Reserved.
*
* Contributor(s): YUKI "Piro" Hiroshi
@@ -101,15 +101,15 @@ async function moveTabsInternallyBefore(tabs, referenceTab, options = {}) {
const oldNextTab = tab.$TST.unsafeNextTab;
if (oldNextTab && oldNextTab.id == referenceTab.id) // no move case
continue;
- if (SidebarConnection.isInitialized()) { // only on the background page
- win.internalMovingTabs.add(tab.id);
- win.alreadyMovedTabs.add(tab.id);
- }
const fromIndex = tab.index;
if (referenceTab.index > tab.index)
tab.index = referenceTab.index - 1;
else
tab.index = referenceTab.index;
+ if (SidebarConnection.isInitialized()) { // only on the background page
+ win.internalMovingTabs.set(tab.id, tab.index);
+ win.alreadyMovedTabs.set(tab.id, tab.index);
+ }
tab.reindexedBy = `moveTabsInternallyBefore (${tab.index})`;
Tab.track(tab);
movedTabs.push(tab);
@@ -211,10 +211,6 @@ async function moveTabsInternallyAfter(tabs, referenceTab, options = {}) {
if ((!oldNextTab && !nextTab) ||
(oldNextTab && nextTab && oldNextTab.id == nextTab.id)) // no move case
continue;
- if (SidebarConnection.isInitialized()) { // only on the background page
- win.internalMovingTabs.add(tab.id);
- win.alreadyMovedTabs.add(tab.id);
- }
const fromIndex = tab.index;
if (nextTab) {
if (nextTab.index > tab.index)
@@ -225,6 +221,10 @@ async function moveTabsInternallyAfter(tabs, referenceTab, options = {}) {
else {
tab.index = win.tabs.size - 1
}
+ if (SidebarConnection.isInitialized()) { // only on the background page
+ win.internalMovingTabs.set(tab.id, tab.index);
+ win.alreadyMovedTabs.set(tab.id, tab.index);
+ }
tab.reindexedBy = `moveTabsInternallyAfter (${tab.index})`;
Tab.track(tab);
movedTabs.push(tab);
@@ -363,8 +363,8 @@ async function syncToNativeTabsInternal(windowId) {
toIndex--;
log(`syncToNativeTabs(${windowId}): step1, move ${moveTabIds.join(',')} before ${referenceId} / from = ${fromIndex}, to = ${toIndex}`);
for (const movedId of moveTabIds) {
- win.internalMovingTabs.add(movedId);
- win.alreadyMovedTabs.add(movedId);
+ win.internalMovingTabs.set(movedId, -1);
+ win.alreadyMovedTabs.set(movedId, -1);
movedTabs.add(movedId);
}
logApiTabs(`tabs-move:syncToNativeTabs(${windowId}): step1, browser.tabs.move() `, moveTabIds, {
diff --git a/waterfox/browser/components/sidebar/background/tabs-open.js b/waterfox/browser/components/sidebar/background/tabs-open.js
index 5ab817910df55..22f2197b75bba 100644
--- a/waterfox/browser/components/sidebar/background/tabs-open.js
+++ b/waterfox/browser/components/sidebar/background/tabs-open.js
@@ -299,7 +299,8 @@ function onMessage(message, openerTab) {
windowId: message.windowId,
parent: Tab.get(message.parentId),
insertBefore: Tab.get(message.insertBeforeId),
- insertAfter: Tab.get(message.insertAfterId)
+ insertAfter: Tab.get(message.insertAfterId),
+ active: !!message.active,
});
});
break;
diff --git a/waterfox/browser/components/sidebar/background/tree.js b/waterfox/browser/components/sidebar/background/tree.js
index fd12819d6c10e..494b732546ba4 100644
--- a/waterfox/browser/components/sidebar/background/tree.js
+++ b/waterfox/browser/components/sidebar/background/tree.js
@@ -455,11 +455,11 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
case Constants.kINSERT_END:
default:
insertAfter = lastDescendant;
- log(` insert ${child && child.id} after lastDescendant ${insertAfter && insertAfter.id} (insertAt=kINSERT_END)`);
+ log(` insert ${child?.id} after lastDescendant ${insertAfter?.id} (insertAt=kINSERT_END)`);
break;
case Constants.kINSERT_TOP:
insertBefore = firstChild;
- log(` insert ${child && child.id} before firstChild ${insertBefore && insertBefore.id} (insertAt=kINSERT_TOP)`);
+ log(` insert ${child?.id} before firstChild ${insertBefore?.id} (insertAt=kINSERT_TOP)`);
break;
case Constants.kINSERT_NEAREST: {
const allTabs = Tab.getOtherTabs((child || parent).windowId, ignoreTabs);
@@ -468,11 +468,11 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
if (index < allTabs.indexOf(firstChild)) {
insertBefore = firstChild;
insertAfter = parent;
- log(` insert ${child && child.id} between parent ${insertAfter && insertAfter.id} and firstChild ${insertBefore && insertBefore.id} (insertAt=kINSERT_NEAREST)`);
+ log(` insert ${child?.id} between parent ${insertAfter?.id} and firstChild ${insertBefore?.id} (insertAt=kINSERT_NEAREST)`);
}
else if (index > allTabs.indexOf(lastDescendant)) {
insertAfter = lastDescendant;
- log(` insert ${child && child.id} after lastDescendant ${insertAfter && insertAfter.id} (insertAt=kINSERT_NEAREST)`);
+ log(` insert ${child?.id} after lastDescendant ${insertAfter?.id} (insertAt=kINSERT_NEAREST)`);
}
else { // inside the tree
if (parent && !children)
@@ -483,12 +483,12 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
if (index > allTabs.indexOf(child))
continue;
insertBefore = child;
- log(` insert ${child && child.id} before nearest following child ${insertBefore && insertBefore.id} (insertAt=kINSERT_NEAREST)`);
+ log(` insert ${child?.id} before nearest following child ${insertBefore?.id} (insertAt=kINSERT_NEAREST)`);
break;
}
if (!insertBefore) {
insertAfter = lastDescendant;
- log(` insert ${child && child.id} after lastDescendant ${insertAfter && insertAfter.id} (insertAt=kINSERT_NEAREST)`);
+ log(` insert ${child?.id} after lastDescendant ${insertAfter?.id} (insertAt=kINSERT_NEAREST)`);
}
}
}; break;
@@ -500,11 +500,11 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
lastRelatedTab = child && parent.$TST.lastRelatedTabId == child.id ? parent.$TST.previousLastRelatedTab : parent.$TST.lastRelatedTab; // it could be updated already...
if (lastRelatedTab) {
insertAfter = lastRelatedTab.$TST.lastDescendant || lastRelatedTab;
- log(` insert ${child && child.id} after lastRelatedTab ${lastRelatedTab.id} (insertAt=kINSERT_NEXT_TO_LAST_RELATED_TAB)`);
+ log(` insert ${child?.id} after lastRelatedTab ${lastRelatedTab.id} (insertAt=kINSERT_NEXT_TO_LAST_RELATED_TAB)`);
}
else {
insertBefore = firstChild;
- log(` insert ${child && child.id} before firstChild (insertAt=kINSERT_NEXT_TO_LAST_RELATED_TAB)`);
+ log(` insert ${child?.id} before firstChild (insertAt=kINSERT_NEXT_TO_LAST_RELATED_TAB)`);
}
}; break;
case Constants.kINSERT_NO_CONTROL:
@@ -513,23 +513,23 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
}
else {
insertAfter = parent;
- log(` insert ${child && child.id} after parent`);
+ log(` insert ${child?.id} after parent`);
}
if (insertBefore == child) {
// Return unsafe tab, to avoid placing the child after hidden tabs
// (too far from the place it should be.)
- insertBefore = insertBefore && insertBefore.$TST.unsafeNextTab;
- log(` => insert ${child && child.id} before next tab ${insertBefore && insertBefore.id} of the child tab itelf`);
+ insertBefore = insertBefore?.$TST.unsafeNextTab;
+ log(` => insert ${child?.id} before next tab ${insertBefore?.id} of the child tab itelf`);
}
if (insertAfter == child) {
- insertAfter = insertAfter && insertAfter.$TST.previousTab;
- log(` => insert ${child && child.id} after previous tab ${insertAfter && insertAfter.id} of the child tab itelf`);
+ insertAfter = insertAfter?.$TST.previousTab;
+ log(` => insert ${child?.id} after previous tab ${insertAfter?.id} of the child tab itelf`);
}
// disallow to place tab in invalid position
if (insertBefore) {
if (parent && insertBefore.index <= parent.index) {
insertBefore = null;
- log(` => do not put ${child && child.id} before a tab preceding to the parent`);
+ log(` => do not put ${child?.id} before a tab preceding to the parent`);
}
//TODO: we need to reject more cases...
}
@@ -541,7 +541,7 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
if (lastMember != insertAfter &&
insertAfter.index >= lastMember.index) {
insertAfter = lastMember;
- log(` => do not put ${child && child.id} after the last tab ${insertAfter && insertAfter.id} in the tree`);
+ log(` => do not put ${child?.id} after the last tab ${insertAfter?.id} in the tree`);
}
//TODO: we need to reject more cases...
}
@@ -551,8 +551,7 @@ export function getReferenceTabsForNewChild(child, parent, { insertAt, ignoreTab
export function getReferenceTabsForNewNextSibling(base, options = {}) {
log('getReferenceTabsForNewNextSibling ', base);
let insertBefore = base.$TST.nextSiblingTab;
- if (insertBefore &&
- insertBefore.pinned &&
+ if (insertBefore?.pinned &&
!options.pinned) {
insertBefore = Tab.getFirstNormalTab(base.windowId);
}
@@ -634,6 +633,7 @@ export async function detachTabsFromTree(tabs, options = {}) {
promisedAttach.push(detachAllChildren(tab, {
...options,
behavior: Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD,
+ ignoreTabs: tabs,
}));
}
if (promisedAttach.length > 0)
@@ -642,7 +642,7 @@ export async function detachTabsFromTree(tabs, options = {}) {
export async function detachAllChildren(
tab = null,
- { windowId, children, descendants, parent, nearestFollowingRootTab, newParent, behavior, dontExpand, dontSyncParentToOpenerTab,
+ { windowId, children, descendants, parent, nearestFollowingRootTab, newParent, ignoreTabs, behavior, dontExpand, dontSyncParentToOpenerTab,
...options } = {}
) {
if (tab) {
@@ -652,12 +652,14 @@ export async function detachAllChildren(
descendants = tab.$TST.descendants;
}
log('detachAllChildren: ',
- tab && tab.id,
+ tab?.id,
{ children, parent, nearestFollowingRootTab, newParent, behavior, dontExpand, dontSyncParentToOpenerTab },
options);
// the "children" option is used for removing tab.
children = children ? children.map(TabsStore.ensureLivingTab) : tab.$TST.children;
+ const ignoreTabsSet = new Set(ignoreTabs || []);
+
if (behavior == Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD &&
newParent &&
!children.includes(newParent))
@@ -675,9 +677,11 @@ export async function detachAllChildren(
options.dontUpdateInsertionPositionInfo = true;
// the "parent" option is used for removing tab.
- parent = TabsStore.ensureLivingTab(parent) || (tab && tab.$TST.parent);
- if (tab &&
- tab.$TST.isGroupTab &&
+ parent = TabsStore.ensureLivingTab(parent) || tab?.$TST.parent;
+ while (ignoreTabsSet.has(parent)) {
+ parent = parent.$TST.parent;
+ }
+ if (tab?.$TST.isGroupTab &&
Tab.getRemovingTabs(tab.windowId).length == children.length) {
behavior = Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_ALL_CHILDREN;
options.dontUpdateIndent = false;
@@ -689,7 +693,7 @@ export async function detachAllChildren(
!configs.moveTabsToBottomWhenDetachedFromClosedParent) {
nextTab = nearestFollowingRootTab !== undefined ?
nearestFollowingRootTab :
- tab && tab.$TST.nearestFollowingRootTab;
+ tab?.$TST.nearestFollowingRootTab;
previousTab = nextTab ?
nextTab.$TST.previousTab :
Tab.getLastTab(windowId || tab.windowId);
@@ -734,9 +738,9 @@ export async function detachAllChildren(
promises.push(detachTab(child, { ...options, dontSyncParentToOpenerTab }));
// reference tabs can be closed while waiting...
- if (nextTab && nextTab.$TST.removing)
+ if (nextTab?.$TST.removing)
nextTab = null;
- if (previousTab && previousTab.$TST.removing)
+ if (previousTab?.$TST.removing)
previousTab = null;
if (nextTab) {
@@ -801,8 +805,7 @@ export async function behaveAutoAttachedTab(
baseTab = baseTab || Tab.getActiveTab(TabsStore.getCurrentWindowId() || tab.windowId);
log('behaveAutoAttachedTab ', tab.id, baseTab.id, { baseTab, behavior });
- if (baseTab &&
- baseTab.$TST.ancestors.includes(tab)) {
+ if (baseTab?.$TST.ancestors.includes(tab)) {
log(' => ignore possibly restored ancestor tab to avoid cyclic references');
return false;
}
@@ -1095,7 +1098,7 @@ async function collapseExpandSubtreeInternal(tab, params = {}) {
tabId: tab.id,
collapsed: !!params.collapsed,
justNow: params.justNow,
- anchorId: anchor && anchor.id,
+ anchorId: anchor?.id,
visibilityChangedTabIds,
last: true
});
@@ -1231,7 +1234,7 @@ export async function collapseExpandTab(tab, params = {}) {
type: Constants.kCOMMAND_NOTIFY_TAB_COLLAPSED_STATE_CHANGED,
windowId: tab.windowId,
tabId: tab.id,
- anchorId: collapseExpandInfo.anchor && collapseExpandInfo.anchor.id,
+ anchorId: collapseExpandInfo.anchor?.id,
justNow: params.justNow,
collapsed: params.collapsed,
last,
@@ -1257,55 +1260,60 @@ export async function collapseExpandTreesIntelligentlyFor(tab, options = {}) {
}
win.doingIntelligentlyCollapseExpandCount++;
- const expandedAncestors = [tab.id]
- .concat(tab.$TST.ancestors.map(ancestor => ancestor.id))
- .concat(tab.$TST.descendants.map(descendant => descendant.id));
- const collapseTabs = Tab.getSubtreeCollapsedTabs(tab.windowId, {
- '!id': expandedAncestors
- });
- logCollapseExpand(`${collapseTabs.length} tabs can be collapsed, ancestors: `, expandedAncestors);
- const allowedToCollapse = new Set();
- await Promise.all(collapseTabs.map(async tab => {
- const allowed = await TSTAPI.tryOperationAllowed(
- TSTAPI.kNOTIFY_TRY_COLLAPSE_TREE_FROM_OTHER_EXPANSION,
- { tab },
- { tabProperties: ['tab'] }
- );
- if (allowed)
- allowedToCollapse.add(tab);
- }));
- for (const collapseTab of collapseTabs) {
- if (!allowedToCollapse.has(collapseTab))
- continue;
- let dontCollapse = false;
- const parentTab = collapseTab.$TST.parent;
- if (parentTab) {
- dontCollapse = true;
- if (!parentTab.$TST.subtreeCollapsed) {
- for (const ancestor of collapseTab.$TST.ancestors) {
- if (!expandedAncestors.includes(ancestor.id))
- continue;
- dontCollapse = false;
- break;
+ try {
+ const expandedAncestors = [tab.id]
+ .concat(tab.$TST.ancestors.map(ancestor => ancestor.id))
+ .concat(tab.$TST.descendants.map(descendant => descendant.id));
+ const collapseTabs = Tab.getSubtreeCollapsedTabs(tab.windowId, {
+ '!id': expandedAncestors
+ });
+ logCollapseExpand(`${collapseTabs.length} tabs can be collapsed, ancestors: `, expandedAncestors);
+ const allowedToCollapse = new Set();
+ await Promise.all(collapseTabs.map(async tab => {
+ const allowed = await TSTAPI.tryOperationAllowed(
+ TSTAPI.kNOTIFY_TRY_COLLAPSE_TREE_FROM_OTHER_EXPANSION,
+ { tab },
+ { tabProperties: ['tab'] }
+ );
+ if (allowed)
+ allowedToCollapse.add(tab);
+ }));
+ for (const collapseTab of collapseTabs) {
+ if (!allowedToCollapse.has(collapseTab))
+ continue;
+ let dontCollapse = false;
+ const parentTab = collapseTab.$TST.parent;
+ if (parentTab) {
+ dontCollapse = true;
+ if (!parentTab.$TST.subtreeCollapsed) {
+ for (const ancestor of collapseTab.$TST.ancestors) {
+ if (!expandedAncestors.includes(ancestor.id))
+ continue;
+ dontCollapse = false;
+ break;
+ }
}
}
+ logCollapseExpand(`${collapseTab.id}: dontCollapse = ${dontCollapse}`);
+
+ const manuallyExpanded = collapseTab.$TST.states.has(Constants.kTAB_STATE_SUBTREE_EXPANDED_MANUALLY);
+ if (!dontCollapse &&
+ !manuallyExpanded &&
+ collapseTab.$TST.descendants.every(tab => !tab.$TST.canBecomeSticky))
+ collapseExpandSubtree(collapseTab, {
+ ...options,
+ collapsed: true
+ });
}
- logCollapseExpand(`${collapseTab.id}: dontCollapse = ${dontCollapse}`);
- const manuallyExpanded = collapseTab.$TST.states.has(Constants.kTAB_STATE_SUBTREE_EXPANDED_MANUALLY);
- if (!dontCollapse &&
- !manuallyExpanded &&
- collapseTab.$TST.descendants.every(tab => !tab.$TST.canBecomeSticky))
- collapseExpandSubtree(collapseTab, {
- ...options,
- collapsed: true
- });
+ collapseExpandSubtree(tab, {
+ ...options,
+ collapsed: false
+ });
+ }
+ catch(error) {
+ log(`failed to collapse/expand tree under ${tab.id}: ${String(error)}`, error);
}
-
- collapseExpandSubtree(tab, {
- ...options,
- collapsed: false
- });
win.doingIntelligentlyCollapseExpandCount--;
}
@@ -1345,12 +1353,12 @@ export async function fixupSubtreeCollapsedState(tab, options = {}) {
export async function moveTabSubtreeBefore(tab, nextTab, options = {}) {
if (!tab)
return;
- if (nextTab && nextTab.$TST.isAllPlacedBeforeSelf([tab].concat(tab.$TST.descendants))) {
+ if (nextTab?.$TST.isAllPlacedBeforeSelf([tab].concat(tab.$TST.descendants))) {
log('moveTabSubtreeBefore:no need to move');
return;
}
- log('moveTabSubtreeBefore: ', tab.id, nextTab && nextTab.id);
+ log('moveTabSubtreeBefore: ', tab.id, nextTab?.id);
const win = TabsStore.windows.get(tab.windowId);
win.subTreeMovingCount++;
try {
@@ -1359,8 +1367,8 @@ export async function moveTabSubtreeBefore(tab, nextTab, options = {}) {
throw new Error('the tab was removed before moving of descendants');
await followDescendantsToMovedRoot(tab, options);
}
- catch(e) {
- log(`failed to move subtree: ${String(e)}`);
+ catch(error) {
+ log(`failed to move subtree: ${String(error)}`, error);
}
await wait(0);
win.subTreeMovingCount--;
@@ -1370,8 +1378,8 @@ export async function moveTabSubtreeAfter(tab, previousTab, options = {}) {
if (!tab)
return;
- log('moveTabSubtreeAfter: ', tab.id, previousTab && previousTab.id);
- if (previousTab && previousTab.$TST.isAllPlacedAfterSelf([tab].concat(tab.$TST.descendants))) {
+ log('moveTabSubtreeAfter: ', tab.id, previousTab?.id);
+ if (previousTab?.$TST.isAllPlacedAfterSelf([tab].concat(tab.$TST.descendants))) {
log(' => no need to move');
return;
}
@@ -1384,8 +1392,8 @@ export async function moveTabSubtreeAfter(tab, previousTab, options = {}) {
throw new Error('the tab was removed before moving of descendants');
await followDescendantsToMovedRoot(tab, options);
}
- catch(e) {
- log(`failed to move subtree: ${String(e)}`);
+ catch(error) {
+ log(`failed to move subtree: ${String(error)}`, error);
}
await wait(0);
win.subTreeMovingCount--;
@@ -1399,7 +1407,12 @@ async function followDescendantsToMovedRoot(tab, options = {}) {
const win = TabsStore.windows.get(tab.windowId);
win.subTreeChildrenMovingCount++;
win.subTreeMovingCount++;
- await TabsMove.moveTabsAfter(tab.$TST.descendants, tab, options);
+ try {
+ await TabsMove.moveTabsAfter(tab.$TST.descendants, tab, options);
+ }
+ catch(error) {
+ log(`failed to move descendants of ${tab.id}: ${String(error)}`, error);
+ }
win.subTreeChildrenMovingCount--;
win.subTreeMovingCount--;
}
@@ -1529,8 +1542,7 @@ export async function moveTabs(tabs, options = {}) {
log('moveTabs: all windows and tabs are ready, ', movedTabIds, destinationWindowId);
let toIndex = (tabs.some(tab => tab.pinned) ? Tab.getPinnedTabs(destinationWindowId) : Tab.getAllTabs(destinationWindowId)).length;
log('toIndex = ', toIndex);
- if (options.insertBefore &&
- options.insertBefore.windowId == destinationWindowId) {
+ if (options.insertBefore?.windowId == destinationWindowId) {
try {
toIndex = Tab.get(options.insertBefore.id).index;
}
@@ -1539,8 +1551,7 @@ export async function moveTabs(tabs, options = {}) {
log('options.insertBefore is unavailable');
}
}
- else if (options.insertAfter &&
- options.insertAfter.windowId == destinationWindowId) {
+ else if (options.insertAfter?.windowId == destinationWindowId) {
try {
toIndex = Tab.get(options.insertAfter.id).index + 1;
}
@@ -1886,8 +1897,8 @@ class TabActionForNewPosition {
}
export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
- const isTabCreating = moveInfo && !!moveInfo.isTabCreating;
- const isMovingByShortcut = moveInfo && !!moveInfo.isMovingByShortcut;
+ const isTabCreating = !!moveInfo?.isTabCreating;
+ const isMovingByShortcut = !!moveInfo?.isMovingByShortcut;
if (tab.pinned)
return new TabActionForNewPosition(tab.$TST.parentId ? 'detach' : 'move', {
@@ -1910,18 +1921,48 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
const prevTab = tree.tabsById[target.previous];
const nextTab = tree.tabsById[target.next];
- log('prevTab: ', dumpTab(prevTab));
- log('nextTab: ', dumpTab(nextTab));
- const prevParent = prevTab && tree.tabsById[prevTab.parent];
- const nextParent = nextTab && tree.tabsById[nextTab.parent];
+ // When multiple tabs are moved at once by outside of TST (e.g. moving of multiselected tabs)
+ // this method may be called multiple times asynchronously before previous operation finishes.
+ // Thus we need to refer the calculated "parent" if it is given.
+ const futurePrevParent = Tab.get(Tab.get(prevTab?.id)?.$TST?.temporaryMetadata.get('goingToBeAttachedTo'));
+ const futureNextParent = Tab.get(Tab.get(nextTab?.id)?.$TST?.temporaryMetadata.get('goingToBeAttachedTo'));
+
+ const prevParent = prevTab && tree.tabsById[prevTab.parent] ||
+ snapshotTab(Tab.get(prevTab?.parent)) || // Given treeForActionDetection may not contain the parent tab, so failsafe
+ snapshotTab(futurePrevParent);
+ const nextParent = nextTab && tree.tabsById[nextTab.parent] ||
+ snapshotTab(Tab.get(nextTab?.parent)) || // Given treeForActionDetection may not contain the parent tab, so failsafe
+ snapshotTab(futureNextParent);
+ if (prevParent)
+ tree.tabsById[prevParent.id] = prevParent;
+ if (nextParent)
+ tree.tabsById[nextParent.id] = nextParent;
+
+ // Given treeForActionDetection may not contain the parent tab, so we fixup the information.
+ if (prevTab &&
+ !prevTab.parent &&
+ prevParent) {
+ prevTab.parent = prevParent.id;
+ prevTab.level = prevParent.level + 1;
+ }
+ if (nextTab &&
+ !nextTab.parent &&
+ nextParent) {
+ nextTab.parent = nextParent.id;
+ nextTab.level = nextParent.level + 1;
+ }
+ log('prevTab: ', dumpTab(prevTab), `parent: ${prevTab?.parent}`);
+ log('nextTab: ', dumpTab(nextTab), `parent: ${nextTab?.parent}`);
const prevLevel = prevTab ? prevTab.level : -1 ;
const nextLevel = nextTab ? nextTab.level : -1 ;
log('prevLevel: '+prevLevel);
log('nextLevel: '+nextLevel);
- const oldParent = tree.tabsById[target.parent];
+ const oldParent = tree.tabsById[target.parent] || snapshotTab(Tab.get(target.parent));
+ if (oldParent)
+ tree.tabsById[oldParent.id] = oldParent;
let newParent = null;
let mustToApply = false;
@@ -1936,7 +1977,7 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
tab,
isTabCreating,
isMovingByShortcut,
- insertAfter: prevTab && prevTab.id,
+ insertAfter: prevTab?.id,
mustToApply,
});
}
@@ -1948,7 +1989,7 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
}
else if (oldParent &&
prevTab &&
- oldParent == prevTab) {
+ oldParent?.id == prevTab?.id) {
log('=> no need to fix case');
newParent = oldParent;
}
@@ -1961,7 +2002,7 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
log('=> moved to last position');
let ancestor = oldParent;
while (ancestor) {
- if (ancestor == prevParent) {
+ if (ancestor.id == prevParent?.id) {
log(' => moving in related tree: keep it attached in existing tree');
newParent = prevParent;
break;
@@ -1971,15 +2012,15 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
if (!newParent) {
log(' => moving from other tree: keep it orphaned');
}
- mustToApply = !!oldParent && newParent != oldParent;
+ mustToApply = !!oldParent && newParent?.id != oldParent.id;
}
- else if (prevParent == nextParent) {
+ else if (prevParent?.id == nextParent?.id) {
log('=> moved into existing tree');
newParent = prevParent;
- mustToApply = !oldParent || newParent != oldParent;
+ mustToApply = !oldParent || newParent?.id != oldParent.id;
}
else if (prevLevel > nextLevel &&
- nextTab.parent != tab.id) {
+ nextTab?.parent != tab.id) {
log('=> moved to end of existing tree');
if (!target.active &&
target.children.length == 0 &&
@@ -1992,35 +2033,35 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
const realDelta = Math.abs(toIndex - fromIndex);
newParent = realDelta < 2 ? prevParent : (oldParent || nextParent) ;
}
- while (newParent && newParent.collapsed) {
+ while (newParent?.collapsed) {
log('=> the tree is collapsed, up to parent tree')
newParent = tree.tabsById[newParent.parent];
}
- mustToApply = !!oldParent && newParent != oldParent;
+ mustToApply = !!oldParent && newParent?.id != oldParent.id;
}
else if (prevLevel < nextLevel &&
- nextTab.parent == prevTab.id) {
+ nextTab?.parent == prevTab?.id) {
log('=> moved to first child position of existing tree');
newParent = prevTab || oldParent || nextParent;
- mustToApply = !!oldParent && newParent != oldParent;
+ mustToApply = !!oldParent && newParent?.id != oldParent.id;
}
log('calculated parent: ', {
- old: oldParent && oldParent.id,
- new: newParent && newParent.id
+ old: oldParent?.id,
+ new: newParent?.id
});
if (newParent) {
let ancestor = newParent;
while (ancestor) {
- if (ancestor == target) {
+ if (ancestor.id == target.id) {
if (moveInfo.toIndex - moveInfo.fromIndex == 1) {
log('=> maybe move-down by keyboard shortcut or something.');
let nearestForeigner = tab.$TST.nearestFollowingForeignerTab;
if (nearestForeigner &&
nearestForeigner == tab)
nearestForeigner = nearestForeigner.$TST.nextTab;
- log('nearest foreigner tab: ', nearestForeigner && nearestForeigner.id);
+ log('nearest foreigner tab: ', nearestForeigner?.id);
if (nearestForeigner) {
if (nearestForeigner.$TST.hasChild)
return new TabActionForNewPosition('attach', {
@@ -2054,8 +2095,8 @@ export function detectTabActionFromNewPosition(tab, moveInfo = {}) {
isTabCreating,
isMovingByShortcut,
parent: newParent.id,
- insertBefore: nextTab && nextTab.id,
- insertAfter: prevTab && prevTab.id,
+ insertBefore: nextTab?.id,
+ insertAfter: prevTab?.id,
mustToApply,
});
}
@@ -2085,11 +2126,11 @@ export function snapshotForActionDetection(targetTab) {
const prevTab = targetTab.$TST.nearestCompletelyOpenedNormalPrecedingTab;
const nextTab = targetTab.$TST.nearestCompletelyOpenedNormalFollowingTab;
const tabs = Array.from(new Set([
- ...(prevTab && prevTab.$TST.ancestors || []),
+ ...(prevTab?.$TST?.ancestors || []),
prevTab,
targetTab,
nextTab,
- targetTab.$TST.parent
+ targetTab.$TST.parent,
]))
.filter(TabsStore.ensureLivingTab)
.sort((a, b) => a.index - b.index);
@@ -2103,37 +2144,40 @@ function snapshotTree(targetTab, tabs) {
function snapshotChild(tab) {
if (!TabsStore.ensureLivingTab(tab) || tab.pinned)
return null;
- return snapshotById[tab.id] = {
- id: tab.id,
- url: tab.url,
- cookieStoreId: tab.cookieStoreId,
- active: tab.active,
- children: tab.$TST.children.map(child => child.id),
- collapsed: tab.$TST.subtreeCollapsed,
- pinned: tab.pinned,
- level: tab.$TST.ancestorIds.length, // parseInt(tab.$TST.getAttribute(Constants.kLEVEL) || 0), // we need to use the number of real ancestors instead of a cached "level", because it will be updated with delay
- trackedAt: tab.$TST.trackedAt,
- mayBeReplacedWithContainer: tab.$TST.mayBeReplacedWithContainer
- };
+ return snapshotById[tab.id] = snapshotTab(tab);
}
const snapshotArray = allTabs.map(tab => snapshotChild(tab));
for (const tab of allTabs) {
const item = snapshotById[tab.id];
if (!item)
continue;
- const parent = tab.$TST.parent;
- item.parent = parent && parent.id;
- const next = tab.$TST.nearestCompletelyOpenedNormalFollowingTab;
- item.next = next && next.id;
- const previous = tab.$TST.nearestCompletelyOpenedNormalPrecedingTab;
- item.previous = previous && previous.id;
+ item.parent = tab.$TST.parent?.id;
+ item.next = tab.$TST.nearestCompletelyOpenedNormalFollowingTab?.id;
+ item.previous = tab.$TST.nearestCompletelyOpenedNormalPrecedingTab?.id;
}
const activeTab = Tab.getActiveTab(targetTab.windowId);
return {
target: snapshotById[targetTab.id],
active: activeTab && snapshotById[activeTab.id],
tabs: snapshotArray,
- tabsById: snapshotById
+ tabsById: snapshotById,
+ };
+}
+
+function snapshotTab(tab) {
+ if (!tab)
+ return null;
+ return {
+ id: tab.id,
+ url: tab.url,
+ cookieStoreId: tab.cookieStoreId,
+ active: tab.active,
+ children: tab.$TST.children.map(child => child.id),
+ collapsed: tab.$TST.subtreeCollapsed,
+ pinned: tab.pinned,
+ level: tab.$TST.level, // parseInt(tab.$TST.getAttribute(Constants.kLEVEL) || 0), // we need to use the number of real ancestors instead of a cached "level", because it will be updated with delay
+ trackedAt: tab.$TST.trackedAt,
+ mayBeReplacedWithContainer: tab.$TST.mayBeReplacedWithContainer,
};
}
diff --git a/waterfox/browser/components/sidebar/common/Tab.js b/waterfox/browser/components/sidebar/common/Tab.js
index 4c9a484c9c4f2..84c3bfa9d8800 100644
--- a/waterfox/browser/components/sidebar/common/Tab.js
+++ b/waterfox/browser/components/sidebar/common/Tab.js
@@ -389,6 +389,10 @@ export default class Tab {
return this.states.has(Constants.kTAB_STATE_STICKY);
}
+ get stuck() {
+ return this.element?.parentNode?.classList.contains('sticky-tabs-container');
+ }
+
get isNewTabCommandTab() {
if (!this.tab ||
!configs.guessNewOrphanTabAsOpenedByNewTabCommand)
@@ -444,6 +448,13 @@ export default class Tab {
return (new URL(this.tab.url)).searchParams.get('temporaryAggressive') == 'true';
}
+ get replacedParentGroupTabCount() {
+ if (!this.tab || !this.isGroupTab)
+ return 0;
+ const count = parseInt((new URL(this.tab.url)).searchParams.get('replacedParentCount'));
+ return isNaN(count) ? 0 : count;
+ }
+
// Firefox Multi-Account Containers
// https://addons.mozilla.org/firefox/addon/multi-account-containers/
// Temporary Containers
@@ -962,6 +973,10 @@ export default class Tab {
return ancestors;
}
+ get level() {
+ return this.ancestorIds.length;
+ }
+
invalidateCachedAncestors() {
this.cachedAncestorIds = null;
for (const child of this.children) {
@@ -1176,13 +1191,16 @@ export default class Tab {
get needToBeGroupedSiblings() {
if (!this.tab)
return [];
+ const openerTabUniqueId = this.getAttribute(Constants.kPERSISTENT_ORIGINAL_OPENER_TAB_ID);
+ if (!openerTabUniqueId)
+ return [];
return TabsStore.queryAll({
windowId: this.tab.windowId,
tabs: TabsStore.toBeGroupedTabsInWindow.get(this.tab.windowId),
normal: true,
'!id': this.id,
attributes: [
- Constants.kPERSISTENT_ORIGINAL_OPENER_TAB_ID, this.getAttribute(Constants.kPERSISTENT_ORIGINAL_OPENER_TAB_ID),
+ Constants.kPERSISTENT_ORIGINAL_OPENER_TAB_ID, openerTabUniqueId,
Constants.kPERSISTENT_ALREADY_GROUPED_FOR_PINNED_OPENER, ''
],
ordered: true
@@ -1908,30 +1926,27 @@ export default class Tab {
if (!this.tab) // already closed tab
return;
log(`memorizeNeighbors ${this.tab.id} as ${hint}`);
-
- const previousTab = this.unsafePreviousTab;
- this.lastPreviousTabId = previousTab && previousTab.id;
-
- const nextTab = this.unsafeNextTab;
- this.lastNextTabId = nextTab && nextTab.id;
+ this.lastPreviousTabId = this.unsafePreviousTab?.id;
+ this.lastNextTabId = this.unsafeNextTab?.id;
}
- get isSubstantiallyMoved() {
+ // https://github.com/piroor/treestyletab/issues/2309#issuecomment-518583824
+ get movedInBulk() {
const previousTab = this.unsafePreviousTab;
if (this.lastPreviousTabId &&
- this.lastPreviousTabId != (previousTab && previousTab.id)) {
- log(`isSubstantiallyMoved lastPreviousTabId=${this.lastNextTabId}, previousTab=${previousTab && previousTab.id}`);
- return true;
+ this.lastPreviousTabId != previousTab?.id) {
+ log(`not bulkMoved lastPreviousTabId=${this.lastNextTabId}, previousTab=${previousTab?.id}`);
+ return false;
}
const nextTab = this.unsafeNextTab;
if (this.lastNextTabId &&
- this.lastNextTabId != (nextTab && nextTab.id)) {
- log(`isSubstantiallyMoved lastNextTabId=${this.lastNextTabId}, nextTab=${nextTab && nextTab.id}`);
- return true;
+ this.lastNextTabId != nextTab?.id) {
+ log(`not bulkMoved lastNextTabId=${this.lastNextTabId}, nextTab=${nextTab?.id}`);
+ return false;
}
- return false;
+ return true;
}
get sanitized() {
@@ -2040,6 +2055,8 @@ export default class Tab {
ancestorTabIds: this.tab.$TST.ancestorIds,
bundledTabId: this.tab.$TST.bundledTabId,
};
+ if (this.stuck)
+ exportedTab.states.push(Constants.kTAB_STATE_STUCK);
if (configs.cacheAPITreeItems && light)
this.$exportedForAPI = exportedTab;
}
@@ -2893,6 +2910,24 @@ Tab.hasLoadingTab = windowId => {
});
};
+Tab.hasDuplicatedTabs = (windowId, options = {}) => {
+ const tabs = TabsStore.queryAll({
+ windowId,
+ tabs: TabsStore.getTabsMap(TabsStore.livingTabsInWindow, windowId),
+ living: true,
+ ...options,
+ iterator: true
+ });
+ const tabKeys = new Set();
+ for (const tab of tabs) {
+ const key = `${tab.cookieStoreId}\n${tab.url}`;
+ if (tabKeys.has(key))
+ return true;
+ tabKeys.add(key);
+ }
+ return false;
+};
+
Tab.hasMultipleTabs = (windowId, options = {}) => {
const tabs = TabsStore.queryAll({
windowId,
diff --git a/waterfox/browser/components/sidebar/common/Window.js b/waterfox/browser/components/sidebar/common/Window.js
index a32e2d6325741..a30f4bc86f0b6 100644
--- a/waterfox/browser/components/sidebar/common/Window.js
+++ b/waterfox/browser/components/sidebar/common/Window.js
@@ -35,8 +35,8 @@ export default class Window {
this.containerClassList = null;
this.pinnedContainerElement = null;
- this.internalMovingTabs = new Set();
- this.alreadyMovedTabs = new Set();
+ this.internalMovingTabs = new Map();
+ this.alreadyMovedTabs = new Map();
this.internalClosingTabs = new Set();
this.keepDescendantsTabs = new Set();
this.highlightingTabs = new Set();
@@ -45,11 +45,12 @@ export default class Window {
this.subTreeMovingCount =
this.subTreeChildrenMovingCount =
this.doingIntelligentlyCollapseExpandCount =
- this.internalFocusCount =
- this.internalSilentlyFocusCount =
- this.internalByMouseFocusCount =
this.duplicatingTabsCount = 0;
+ this.internallyFocusingTabs = new Set();
+ this.internallyFocusingByMouseTabs = new Set();
+ this.internallyFocusingSilentlyTabs = new Set();
+
this.preventToDetectTabBunchesUntil = Date.now() + configs.tabBunchesDetectionDelayOnNewWindow;
this.openingTabs = new Set();
diff --git a/waterfox/browser/components/sidebar/common/api-tabs.js b/waterfox/browser/components/sidebar/common/api-tabs.js
index 9fe3d36c9c4ec..ea19a513b8a84 100644
--- a/waterfox/browser/components/sidebar/common/api-tabs.js
+++ b/waterfox/browser/components/sidebar/common/api-tabs.js
@@ -29,55 +29,6 @@ export async function getIndexes(...queriedTabIds) {
return indexes.map(tab => tab ? tab.index : -1);
}
-export async function blur(tab, unactivatableTabs = []) {
- const unactivatableIndices = Array.from(new Set([tab, ...unactivatableTabs].map(tab => tab.index)));
- const minUnactivatableIndex = Math.min.apply(null, unactivatableIndices);
- const maxUnactivatableIndex = Math.max.apply(null, unactivatableIndices);
- const unactivatableTabById = new Map();
- for (const tab of unactivatableTabs) {
- unactivatableTabById.set(tab.id, tab);
- }
-
- const allTabs = await browser.tabs.query({ windowId: tab.windowId });
- const unactivatableTabIds = new Set([tab.id, ...unactivatableTabs.map(tab => tab.id)]);
- if (allTabs.length == unactivatableTabIds.size)
- return; // there is no other focusible tab!
-
- const restTabs = allTabs.filter(tab => unactivatableTabIds.has(tab.id));
- const middleTabs = restTabs.filter(tab => tab.index > minUnactivatableIndex || tab.index < maxUnactivatableIndex);
- const previousTab = minUnactivatableIndex > 0 && allTabs[minUnactivatableIndex - 1];
- const nextTab = maxUnactivatableIndex < allTabs.length - 1 && allTabs[maxUnactivatableIndex + 1];
- const allTabById = new Map();
- for (const tab of allTabs) {
- allTabById.set(tab.id, tab);
- }
-
- const scannedTabIds = new Set();
- let successorTab = tab;
- do {
- if (scannedTabIds.has(successorTab.id))
- break; // prevent infinite loop!
- scannedTabIds.add(successorTab.id);
- let nextSuccessorTab = unactivatableTabById.get(successorTab.successorTabId);
- if (nextSuccessorTab) {
- successorTab = nextSuccessorTab;
- continue;
- }
- nextSuccessorTab = allTabById.get(successorTab.successorTabId) ||
- nextTab ||
- (middleTabs.length > 0 && middleTabs[0]) ||
- previousTab ||
- restTabs[0];
- if (unactivatableTabById.has(nextSuccessorTab.id)) {
- successorTab = nextSuccessorTab;
- continue;
- }
- await browser.tabs.update(nextSuccessorTab.id, { active: true });
- break;
- }
- while (successorTab);
-}
-
export function isMissingTabError(error) {
return (
error &&
diff --git a/waterfox/browser/components/sidebar/common/bookmark.js b/waterfox/browser/components/sidebar/common/bookmark.js
index 75f693ff5adf7..836b0adc6d574 100644
--- a/waterfox/browser/components/sidebar/common/bookmark.js
+++ b/waterfox/browser/components/sidebar/common/bookmark.js
@@ -20,6 +20,7 @@ import {
isLinux,
} from './common.js';
import * as ApiTabs from './api-tabs.js';
+import * as TreeBehavior from './tree-behavior.js';
import * as Constants from './constants.js';
import * as ContextualIdentities from './contextual-identities.js';
import * as Dialog from './dialog.js';
@@ -637,6 +638,11 @@ reserveToGroupCreatedBookmarks.retryCount = 0;
async function tryGroupCreatedBookmarks() {
log('tryGroupCreatedBookmarks ', mCreatedBookmarks);
+ if (!configs.autoCreateFolderForBookmarksFromTree) {
+ log(' => autoCreateFolderForBookmarksFromTree is false');
+ return;
+ }
+
const lastDraggedTabs = configs.lastDraggedTabs;
if (lastDraggedTabs &&
lastDraggedTabs.tabIds.length > mCreatedBookmarks.length) {
@@ -677,37 +683,69 @@ async function tryGroupCreatedBookmarks() {
for (const bookmark of bookmarks) {
parentIds.add(bookmark.parentId);
}
+ log('parentIds: ', parentIds);
if (parentIds.size > 1) {
log(' => ignore bookmarks created under multiple folders');
return;
}
}
+ const tabs = lastDraggedTabs ?
+ lastDraggedTabs.tabIds.map(id => Tab.get(id)) :
+ (await Promise.all(bookmarks.map(async bookmark => {
+ const tabs = await browser.tabs.query({ url: bookmark.url });
+ if (tabs.length == 0)
+ return null;
+ const tab = tabs.find(tab => tab.highlighted) || tabs[0];
+ return Tab.get(tab);
+ }))).filter(tab => !!tab);
+ log('tabs: ', tabs);
+ if (tabs.length != bookmarks.length) {
+ log(' => ignore bookmarks created from non-tab sources');
+ return;
+ }
+
+ const treeStructure = TreeBehavior.getTreeStructureFromTabs(tabs);
+ log('treeStructure: ', treeStructure);
+ const topLevelTabsCount = treeStructure.filter(item => item.parent < 0).length;
+ if (topLevelTabsCount == treeStructure.length) {
+ log(' => no need to group bookmarks from dragged flat tabs');
+ return;
+ }
+
+ let titles = getTitlesWithTreeStructure(tabs);
+ if (tabs[0].$TST.isGroupTab &&
+ titles.filter(title => !/^>/.test(title)).length == 1) {
+ log('delete needless bookmark for a group tab');
+ browser.bookmarks.remove(bookmarks[0].id);
+ tabs.shift();
+ bookmarks.shift();
+ titles = getTitlesWithTreeStructure(tabs);
+ }
+ log('titles: ', titles);
+
+ log('save tree structure to bookmarks');
+ for (let i = 0, maxi = bookmarks.length; i < maxi; i++) {
+ const title = titles[i];
+ if (title == tabs[i].title)
+ continue;
+ browser.bookmarks.update(bookmarks[i].id, { title });
+ }
+
+ log('ready to group bookmarks under a folder');
+
const parentId = bookmarks[0].parentId;
{
// Do nothing if all bookmarks are created under a new
// blank folder.
const allChildren = await browser.bookmarks.getChildren(parentId);
+ log('allChildren.length vs bookmarks.length: ', allChildren.length, bookmarks.length);
if (allChildren.length == bookmarks.length) {
- log(' => ignore bookmarks created under a new blank folder');
+ log(' => no need to create folder for bookmarks under a new blank folder');
return;
}
}
- const possibleSourceTabs = (await Promise.all(bookmarks.map(async bookmark => {
- const tabs = await browser.tabs.query({ url: bookmark.url });
- if (tabs.length == 0)
- return null;
- return tabs[0];
- }))).filter(tab => !!tab);
- console.log('possibleSourceTabs ', possibleSourceTabs);
- if (possibleSourceTabs.length != bookmarks.length) {
- log(' => ignore bookmarks created from non-tab sources');
- return;
- }
-
- log('ready to group bookmarks under a folder');
-
log('create a folder for grouping');
mCreatingCount++;
const folder = await browser.bookmarks.create({
@@ -728,28 +766,6 @@ async function tryGroupCreatedBookmarks() {
index: movedCount++
});
}
-
- if (!lastDraggedTabs)
- return;
-
- const tabs = lastDraggedTabs.tabIds.map(id => Tab.get(id));
- let titles = getTitlesWithTreeStructure(tabs);
- if (tabs[0].$TST.isGroupTab &&
- titles.filter(title => !/^>/.test(title)).length == 1) {
- log('delete needless bookmark for a group tab');
- browser.bookmarks.remove(bookmarks[0].id);
- tabs.shift();
- bookmarks.shift();
- titles = getTitlesWithTreeStructure(tabs);
- }
-
- log('save tree structure to bookmarks');
- for (let i = 0, maxi = bookmarks.length; i < maxi; i++) {
- const title = titles[i];
- if (title == tabs[i].title)
- continue;
- browser.bookmarks.update(bookmarks[i].id, { title });
- }
}
if (Constants.IS_BACKGROUND &&
diff --git a/waterfox/browser/components/sidebar/common/cache-storage.js b/waterfox/browser/components/sidebar/common/cache-storage.js
index 40227939b7605..e70805047ba14 100644
--- a/waterfox/browser/components/sidebar/common/cache-storage.js
+++ b/waterfox/browser/components/sidebar/common/cache-storage.js
@@ -6,10 +6,14 @@
'use strict';
import * as UniqueId from '/common/unique-id.js';
+import {
+ asyncRunWithTimeout,
+} from '/common/common.js';
const DB_NAME = 'PermanentStorage';
const DB_VERSION = 3;
const EXPIRATION_TIME_IN_MSEC = 7 * 24 * 60 * 60 * 1000; // 7 days
+const TIMEOUT_IN_MSEC = 1000 * 5; // 5 sec
export const BACKGROUND = 'backgroundCaches';
const SIDEBAR = 'sidebarCaches'; // obsolete, but left here to delete old storage
@@ -70,28 +74,42 @@ export async function setValue({ windowId, key, value } = {}) {
const store = BACKGROUND;
const cacheKey = `${windowUniqueId}-${key}`;
- const timestamp = Date.now();
- try {
- const transaction = db.transaction([store], 'readwrite');
- const cacheStore = transaction.objectStore(store);
-
- cacheStore.put({
- key: cacheKey,
- windowId: windowUniqueId,
- value,
- timestamp,
- });
-
- transaction.oncomplete = () => {
- //db.close();
- windowId = undefined;
- key = undefined;
- value = undefined;
- };
- }
- catch(error) {
- console.error(`Failed to store cache ${cacheKey} in the store ${store}`, error);
- }
+ asyncRunWithTimeout({
+ task: () => new Promise((resolve, reject) => {
+ const timestamp = Date.now();
+ try {
+ const transaction = db.transaction([store], 'readwrite');
+ const cacheStore = transaction.objectStore(store);
+ const cacheRequest = cacheStore.put({
+ key: cacheKey,
+ windowId: windowUniqueId,
+ value,
+ timestamp,
+ });
+
+ transaction.oncomplete = () => {
+ //db.close();
+ windowId = undefined;
+ key = undefined;
+ value = undefined;
+ resolve();
+ };
+
+ cacheRequest.onerror = event => {
+ console.error(`Failed to store cache ${cacheKey} in the store ${store}`, event);
+ reject(event);
+ };
+ }
+ catch(error) {
+ console.error(`Failed to store cache ${cacheKey} in the store ${store}`, error);
+ reject(error);
+ }
+ }),
+ timeout: TIMEOUT_IN_MSEC,
+ onTimedOut() {
+ throw new Error(`CacheStorage.setValue for {windowId}/key timed out`);
+ },
+ });
}
export async function deleteValue({ windowId, key } = {}) {
@@ -107,22 +125,47 @@ export async function deleteValue({ windowId, key } = {}) {
const store = BACKGROUND;
const cacheKey = `${windowUniqueId}-${key}`;
- try {
- const transaction = db.transaction([store], 'readwrite');
- const cacheStore = transaction.objectStore(store);
- cacheStore.delete(cacheKey);
- transaction.oncomplete = () => {
- //db.close();
- windowId = undefined;
- key = undefined;
- };
- }
- catch(error) {
- console.error(`Failed to delete cache ${cacheKey} in the store ${store}`, error);
- }
+ asyncRunWithTimeout({
+ task: () => new Promise((resolve, reject) => {
+ try {
+ const transaction = db.transaction([store], 'readwrite');
+ const cacheStore = transaction.objectStore(store);
+ const cacheRequest = cacheStore.delete(cacheKey);
+
+ transaction.oncomplete = () => {
+ //db.close();
+ windowId = undefined;
+ key = undefined;
+ resolve();
+ };
+
+ cacheRequest.onerror = event => {
+ console.error(`Failed to delete cache ${cacheKey} in the store ${store}`, event);
+ reject(event);
+ };
+ }
+ catch(error) {
+ console.error(`Failed to delete cache ${cacheKey} in the store ${store}`, error);
+ reject(error);
+ }
+ }),
+ timeout: TIMEOUT_IN_MSEC,
+ onTimedOut() {
+ throw new Error(`CacheStorage.deleteValue for {windowId}/key timed out`);
+ },
+ });
}
export async function getValue({ windowId, key } = {}) {
+ return asyncRunWithTimeout({
+ task: () => getValueInternal({ windowId, key }),
+ timeout: TIMEOUT_IN_MSEC,
+ onTimedOut() {
+ throw new Error(`CacheStorage.getValue for {windowId}/${key} timed out`);
+ },
+ });
+}
+async function getValueInternal({ windowId, key } = {}) {
return new Promise(async (resolve, _reject) => {
const [db, windowUniqueId] = await Promise.all([
openDB(),
@@ -163,6 +206,11 @@ export async function getValue({ windowId, key } = {}) {
cache.value = undefined;
};
+ cacheRequest.onerror = event => {
+ console.error('Failed to get from cache:', event);
+ resolve(null);
+ };
+
transaction.oncomplete = () => {
//db.close();
windowId = undefined;
@@ -177,6 +225,15 @@ export async function getValue({ windowId, key } = {}) {
}
export async function clearForWindow(windowId) {
+ return asyncRunWithTimeout({
+ task: () => clearForWindowInternal(windowId),
+ timeout: TIMEOUT_IN_MSEC,
+ onTimedOut() {
+ throw new Error(`CacheStorage.clearForWindow for {windowId} timed out`);
+ },
+ });
+}
+async function clearForWindowInternal(windowId) {
reserveToExpireOldEntries();
return new Promise(async (resolve, reject) => {
const [db, windowUniqueId] = await Promise.all([
diff --git a/waterfox/browser/components/sidebar/common/common.js b/waterfox/browser/components/sidebar/common/common.js
index 12fd4e2024c1e..cf2b929546d2e 100644
--- a/waterfox/browser/components/sidebar/common/common.js
+++ b/waterfox/browser/components/sidebar/common/common.js
@@ -13,7 +13,6 @@ import * as Constants from './constants.js';
const WATERFOX_SPECIFIC_VALUES = {
hideHorizontalTabsWhileActive: true,
showTabPreview: true,
- hoverTabPreviewDelayMs: 500,
sidebarPosition: Constants.kTABBAR_POSITION_LEFT,
suppressGapFromShownOrHiddenToolbarOnlyOnMouseOperation: false,
@@ -303,6 +302,8 @@ export const configs = new Configs({
autoExpandOnLongHoverDelay: 500,
autoExpandOnLongHoverRestoreIniitalState: true,
+ autoCreateFolderForBookmarksFromTree: true,
+
accelKey: '',
skipCollapsedTabsForTabSwitchingShortcuts: false,
@@ -341,11 +342,13 @@ export const configs = new Configs({
groupTabTemporaryStateForChildrenOfPinned: Constants.kGROUP_TAB_TEMPORARY_STATE_PASSIVE,
groupTabTemporaryStateForChildrenOfFirefoxView: Constants.kGROUP_TAB_TEMPORARY_STATE_PASSIVE,
groupTabTemporaryStateForOrphanedTabs: Constants.kGROUP_TAB_TEMPORARY_STATE_AGGRESSIVE,
+ groupTabTemporaryStateForAPI: Constants.kGROUP_TAB_TEMPORARY_STATE_NOTHING,
renderTreeInGroupTabs: true,
warnOnAutoGroupNewTabs: true,
warnOnAutoGroupNewTabsWithListing: true,
warnOnAutoGroupNewTabsWithListingMaxRows: 5,
showAutoGroupOptionHint: true,
+ showAutoGroupOptionHintWithOpener: true,
// behavior around newly opened tabs
@@ -392,11 +395,13 @@ export const configs = new Configs({
moveParentBehavior_outsideSidebar_expanded: Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD,
moveParentBehavior_noSidebar_collapsed: Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD,
moveParentBehavior_noSidebar_expanded: Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD,
+ closeParentBehavior_replaceWithGroup_thresholdToPrevent: 1, // negative value means "never prevent"
moveTabsToBottomWhenDetachedFromClosedParent: false,
promoteAllChildrenWhenClosedParentIsLastChild: true,
successorTabControlLevel: Constants.kSUCCESSOR_TAB_CONTROL_IN_TREE,
simulateSelectOwnerOnClose: true,
simulateLockTabSizing: true,
+ deferScrollingToOutOfViewportSuccessor: true,
simulateTabsLoadInBackgroundInverted: false,
supportTabsMultiselect: typeof browser.menus.overrideContext == 'function',
warnOnCloseTabs: true,
@@ -847,6 +852,20 @@ export function nextFrame() {
});
}
+export async function asyncRunWithTimeout({ task, timeout, onTimedOut }) {
+ let succeeded = false;
+ return Promise.race([
+ task().then(result => {
+ succeeded = true;
+ return result;
+ }),
+ wait(timeout).then(() => {
+ if (!succeeded)
+ return onTimedOut();
+ }),
+ ]);
+}
+
const mNotificationTasks = new Map();
@@ -1067,7 +1086,7 @@ export function watchOverflowStateChange({ target, moreResizeTargets, onOverflow
};
let resizeObserver/*, mutationObserver*/;
- if (useLegacyOverflowEvents) {
+ if (!useLegacyOverflowEvents) {
const resizeTargets = new Set([target, ...(moreResizeTargets || [])]);
resizeObserver = new ResizeObserver(entries => {
for (const entry of entries) {
diff --git a/waterfox/browser/components/sidebar/common/constants.js b/waterfox/browser/components/sidebar/common/constants.js
index 8a728934f27ee..65f14d6b0ee92 100644
--- a/waterfox/browser/components/sidebar/common/constants.js
+++ b/waterfox/browser/components/sidebar/common/constants.js
@@ -217,6 +217,7 @@ export const kTAB_STATE_GROUP_TAB = 'group-tab';
export const kTAB_STATE_NEW_TAB_COMMAND_TAB = 'newtab-command-tab';
export const kTAB_STATE_OPENED_FOR_SAME_WEBSITE = 'opened-for-same-website';
export const kTAB_STATE_STICKY = 'sticky';
+export const kTAB_STATE_STUCK = 'stuck'; // virtual state
export const kTAB_INTERNAL_STATES = new Set([ // TST specific states
'tab',
kTAB_STATE_LAST_ROW,
@@ -239,6 +240,7 @@ export const kTAB_INTERNAL_STATES = new Set([ // TST specific states
kTAB_STATE_FROM_FIREFOX_VIEW,
kTAB_STATE_OPENED_FOR_SAME_WEBSITE,
kTAB_STATE_STICKY,
+ kTAB_STATE_STUCK,
]);
export const kTAB_TEMPORARY_STATES = new Set([ // states not trigger updating of cache
kTAB_STATE_CREATING,
@@ -269,6 +271,7 @@ export const kTAB_SAFE_STATES = new Set([ // exportable via API
kTAB_STATE_FROM_FIREFOX_VIEW,
kTAB_STATE_OPENED_FOR_SAME_WEBSITE,
kTAB_STATE_STICKY,
+ kTAB_STATE_STUCK,
]);
export const kTAB_SAFE_STATES_ARRAY = Array.from(kTAB_SAFE_STATES);
diff --git a/waterfox/browser/components/sidebar/common/permissions.js b/waterfox/browser/components/sidebar/common/permissions.js
index 5df2a8a294972..c7f2396e3b8f8 100644
--- a/waterfox/browser/components/sidebar/common/permissions.js
+++ b/waterfox/browser/components/sidebar/common/permissions.js
@@ -71,9 +71,10 @@ browser.runtime.onMessage.addListener((message, _sender) => {
for (const request of requests) {
const { onChanged, checkbox } = destroyRequest(request);
- if (onChanged)
- onChanged(true);
- checkbox.checked = true;
+ const checked =onChanged ?
+ onChanged(true) :
+ undefined;
+ checkbox.checked = checked !== undefined ? !!checked : true;
}
});
@@ -114,7 +115,10 @@ export function bindToCheckbox(permissions, checkbox, options = {}) {
isGranted(permissions)
.then(granted => {
- checkbox.checked = granted;
+ const checked = options.onInitialized ?
+ options.onInitialized(granted) :
+ undefined;
+ checkbox.checked = checked !== undefined ? !!checked : granted;
})
.catch(_error => {
checkbox.setAttribute('readonly', true);
@@ -168,11 +172,12 @@ export function bindToCheckbox(permissions, checkbox, options = {}) {
return;
if (granted) {
+ const checked = options.onChanged ?
+ options.onChanged(true) :
+ undefined;
for (const checkbox of checkboxes) {
- checkbox.checked = true;
+ checkbox.checked = checked !== undefined ? !!checked : true;
}
- if (options.onChanged)
- options.onChanged(true);
browser.runtime.sendMessage({
type: Constants.kCOMMAND_NOTIFY_PERMISSIONS_GRANTED,
permissions
diff --git a/waterfox/browser/components/sidebar/common/retrieve-url.js b/waterfox/browser/components/sidebar/common/retrieve-url.js
index 6c8c13942336b..a73c730f3b8af 100644
--- a/waterfox/browser/components/sidebar/common/retrieve-url.js
+++ b/waterfox/browser/components/sidebar/common/retrieve-url.js
@@ -14,6 +14,7 @@ function log(...args) {
internalLogger('common/retrieve-url', ...args);
}
+export const kTYPE_PLAIN_TEXT = 'text/plain';
export const kTYPE_X_MOZ_URL = 'text/x-moz-url';
export const kTYPE_URI_LIST = 'text/uri-list';
export const kTYPE_MOZ_TEXT_INTERNAL = 'text/x-moz-text-internal';
@@ -23,7 +24,7 @@ const ACCEPTABLE_DATA_TYPES = [
kTYPE_URI_LIST,
kTYPE_X_MOZ_URL,
kTYPE_MOZ_TEXT_INTERNAL,
- 'text/plain'
+ kTYPE_PLAIN_TEXT,
];
let mFileURLResolver = null;
@@ -102,7 +103,7 @@ export async function fromClipboard({ selection } = {}) {
if (await mSelectionClipboardProvider.isAvailable()) {
const maybeUrlString = await mSelectionClipboardProvider.getTextData();
if (maybeUrlString)
- urls.push(...fromData(maybeUrlString, 'text/plain'));
+ urls.push(...fromData(maybeUrlString, kTYPE_PLAIN_TEXT));
return sanitizeURLs(urls);
}
}
@@ -172,7 +173,7 @@ function fromData(data, type) {
.trim()
.split('\n');
- case 'text/plain':
+ case kTYPE_PLAIN_TEXT:
return data
.replace(/\r/g, '\n')
.replace(/\n\n+/g, '\n')
diff --git a/waterfox/browser/components/sidebar/common/tabs-internal-operation.js b/waterfox/browser/components/sidebar/common/tabs-internal-operation.js
index 3d47da4683a1f..44477b9e15695 100644
--- a/waterfox/browser/components/sidebar/common/tabs-internal-operation.js
+++ b/waterfox/browser/components/sidebar/common/tabs-internal-operation.js
@@ -7,6 +7,8 @@
// internal operations means operations bypassing WebExtensions' tabs APIs.
+import EventListenerManager from '/extlib/EventListenerManager.js';
+
import {
log as internalLogger,
dumpTab,
@@ -26,6 +28,8 @@ function log(...args) {
internalLogger('common/tabs-internal-operation', ...args);
}
+export const onBeforeTabsRemove = new EventListenerManager();
+
export async function activateTab(tab, { byMouseOperation, keepMultiselection, silently } = {}) {
if (!Constants.IS_BACKGROUND)
throw new Error('Error: TabsInternalOperation.activateTab is available only on the background page, use a `kCOMMAND_ACTIVATE_TAB` message instead.');
@@ -35,17 +39,15 @@ export async function activateTab(tab, { byMouseOperation, keepMultiselection, s
return;
log('activateTab: ', dumpTab(tab));
const win = TabsStore.windows.get(tab.windowId);
- win.internalFocusCount++;
+ win.internallyFocusingTabs.add(tab.id);
if (byMouseOperation)
- win.internalByMouseFocusCount++;
+ win.internallyFocusingByMouseTabs.add(tab.id);
if (silently)
- win.internalSilentlyFocusCount++;
+ win.internallyFocusingSilentlyTabs.add(tab.id);
const onError = (e) => {
- win.internalFocusCount--;
- if (byMouseOperation)
- win.internalByMouseFocusCount--;
- if (silently)
- win.internalSilentlyFocusCount--;
+ win.internallyFocusingTabs.delete(tab.id);
+ win.internallyFocusingByMouseTabs.delete(tab.id);
+ win.internallyFocusingSilentlyTabs.delete(tab.id);
ApiTabs.handleMissingTabError(e);
};
if (configs.supportTabsMultiselect &&
@@ -77,30 +79,52 @@ export async function blurTab(bluredTabs, { windowId, silently } = {}) {
const bluredTabIds = new Set(Array.from(bluredTabs || [], tab => tab.id || tab));
- let bluredTabsFound = false;
- let nextActiveTab = null;
- for (const tab of Tab.getVisibleTabs(windowId || bluredTabs[0].windowId)) {
- const blured = bluredTabIds.has(tab.id);
- if (blured)
- bluredTabsFound = true;
- if (!bluredTabsFound)
- nextActiveTab = tab;
- if (bluredTabsFound &&
- !blured) {
- nextActiveTab = tab;
+ // First, try to find successor based on successorTabId from left tabs.
+ let successorTab = Tab.get(bluredTabs.find(tab => tab.active)?.successorTabId);
+ const scannedTabIds = new Set();
+ while (successorTab && bluredTabIds.has(successorTab.id)) {
+ if (scannedTabIds.has(successorTab.id))
+ break; // prevent infinite loop!
+ scannedTabIds.add(successorTab.id);
+ const nextSuccessorTab = (successorTab.successorTabId > 0 && successorTab.successorTabId != successorTab.id) ?
+ Tab.get(successorTab.successorTabId) :
+ null;
+ if (!nextSuccessorTab)
break;
+ successorTab = nextSuccessorTab;
+ }
+ log('blurTab/step 1: found successor = ', successorTab?.id);
+
+ // Second, try to detect successor based on their order.
+ if (!successorTab || bluredTabIds.has(successorTab.id)) {
+ if (successorTab)
+ log(' => it cannot become the successor, find again');
+ let bluredTabsFound = false;
+ for (const tab of Tab.getVisibleTabs(windowId || bluredTabs[0].windowId)) {
+ const blured = bluredTabIds.has(tab.id);
+ if (blured)
+ bluredTabsFound = true;
+ if (!bluredTabsFound)
+ successorTab = tab;
+ if (bluredTabsFound &&
+ !blured) {
+ successorTab = tab;
+ break;
+ }
}
+ log('blurTab/step 2: found successor = ', successorTab?.id);
}
- if (nextActiveTab)
- await activateTab(nextActiveTab, { silently });
- return nextActiveTab;
+
+ if (successorTab)
+ await activateTab(successorTab, { silently });
+ return successorTab;
}
export function removeTab(tab) {
return removeTabs([tab]);
}
-export function removeTabs(tabs, { keepDescendants, byMouseOperation, originalStructure, triggerTab } = {}) {
+export async function removeTabs(tabs, { keepDescendants, byMouseOperation, originalStructure, triggerTab } = {}) {
if (!Constants.IS_BACKGROUND)
throw new Error('TabsInternalOperation.removeTabs is available only on the background page, use a `kCOMMAND_REMOVE_TABS_INTERNALLY` message instead.');
@@ -108,6 +132,8 @@ export function removeTabs(tabs, { keepDescendants, byMouseOperation, originalSt
if (tabs.length == 0)
return;
+ await onBeforeTabsRemove.dispatch(tabs);
+
const win = TabsStore.windows.get(tabs[0].windowId);
const tabIds = [];
let willChangeFocus = false;
@@ -136,13 +162,12 @@ export function removeTabs(tabs, { keepDescendants, byMouseOperation, originalSt
clearCache(tab);
if (keepDescendants)
win.keepDescendantsTabs.add(tab.id);
- }
- if (willChangeFocus && byMouseOperation) {
- win.internalByMouseFocusCount++;
- setTimeout(() => { // the operation can be canceled
- if (win.internalByMouseFocusCount > 0)
- win.internalByMouseFocusCount--;
- }, 250);
+ if (willChangeFocus && byMouseOperation) {
+ win.internallyFocusingByMouseTabs.add(tab.id);
+ setTimeout(() => { // the operation can be canceled
+ win.internallyFocusingByMouseTabs.delete(tab.id);
+ }, 250);
+ }
}
}
diff --git a/waterfox/browser/components/sidebar/common/tree-behavior.js b/waterfox/browser/components/sidebar/common/tree-behavior.js
index 17f53194be618..f6d4a0c644733 100644
--- a/waterfox/browser/components/sidebar/common/tree-behavior.js
+++ b/waterfox/browser/components/sidebar/common/tree-behavior.js
@@ -82,15 +82,24 @@ export function getParentTabOperationBehavior(tab, { context, byInternalOperatio
log(' => behavior: ', behavior);
- if (behavior == Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY) {
- behavior = parentTab ? Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_ALL_CHILDREN : Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD;
- log(' => intelligent behavior: ', behavior);
+ const replacedParentCount = tab?.$TST?.replacedParentGroupTabCount;
+ if (behavior == Constants.kPARENT_TAB_OPERATION_BEHAVIOR_REPLACE_WITH_GROUP_TAB &&
+ configs.closeParentBehavior_replaceWithGroup_thresholdToPrevent >= 0 &&
+ replacedParentCount &&
+ replacedParentCount >= configs.closeParentBehavior_replaceWithGroup_thresholdToPrevent) {
+ behavior = Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY;
+ log(' => the group tab is already replaced parent, fallback to kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY');
}
if (behavior == Constants.kPARENT_TAB_OPERATION_BEHAVIOR_ENTIRE_TREE &&
preventEntireTreeBehavior) {
+ behavior = Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY;
+ log(' => preventEntireTreeBehavior behavior, fallback to kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY');
+ }
+
+ if (behavior == Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_INTELLIGENTLY) {
behavior = parentTab ? Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_ALL_CHILDREN : Constants.kPARENT_TAB_OPERATION_BEHAVIOR_PROMOTE_FIRST_CHILD;
- log(' => preventEntireTreeBehavior behavior: ', behavior);
+ log(' => intelligent behavior: ', behavior);
}
// Promote all children to upper level, if this is the last child of the parent.
diff --git a/waterfox/browser/components/sidebar/common/tst-api.js b/waterfox/browser/components/sidebar/common/tst-api.js
index b0a0a9a710c38..b139a8693d90d 100644
--- a/waterfox/browser/components/sidebar/common/tst-api.js
+++ b/waterfox/browser/components/sidebar/common/tst-api.js
@@ -134,6 +134,7 @@ export const kNOTIFY_TRY_COLLAPSE_TREE_FROM_COLLAPSE_COMMAND = 'try-collapse-tre
export const kNOTIFY_TRY_COLLAPSE_TREE_FROM_COLLAPSE_ALL_COMMAND = 'try-collapse-tree-from-collapse-all-command';
export const kNOTIFY_TRY_FIXUP_TREE_ON_TAB_MOVED = 'try-fixup-tree-on-tab-moved';
export const kNOTIFY_TRY_HANDLE_NEWTAB = 'try-handle-newtab';
+export const kNOTIFY_TRY_SCROLL_TO_ACTIVATED_TAB = 'try-scroll-to-activated-tab';
export const kGET_TREE = 'get-tree';
export const kGET_LIGHT_TREE = 'get-light-tree';
export const kATTACH = 'attach';
@@ -272,6 +273,11 @@ export function clearCache(cache) {
// bacause instances of the class will be very short-life and increases RAM usage on
// massive tabs case.
export async function exportTab(sourceTab, { addonId, light, isContextTab, interval, permissions, cache, cacheKey } = {}) {
+ const normalizedSourceTab = Tab.get(sourceTab);
+ if (!normalizedSourceTab)
+ throw new Error(`Fatal error: tried to export not a tab. ${sourceTab}`);
+ sourceTab = normalizedSourceTab;
+
if (!interval)
interval = 0;
if (!cache)
diff --git a/waterfox/browser/components/sidebar/experiments/prefs.js b/waterfox/browser/components/sidebar/experiments/prefs.js
index e3af91d4abd32..38e7604f94599 100644
--- a/waterfox/browser/components/sidebar/experiments/prefs.js
+++ b/waterfox/browser/components/sidebar/experiments/prefs.js
@@ -36,7 +36,7 @@ function initSidebarCategory(document, { locale, BASE_URL, BASE_PREF }) {
if (document.querySelector('#category-tabsSidebar'))
return true;
- const generalItem = document.querySelector('#category-search');
+ const generalItem = document.querySelector('#category-general');
if (!generalItem)
return false;
@@ -859,9 +859,25 @@ function initSidebarCategory(document, { locale, BASE_URL, BASE_PREF }) {
document.defaultView.register_module('paneTabsSidebar', document.defaultView.gSidebarPage);
if (document.URL.endsWith('#tabsSidebar')) {
- document.querySelector('#categories').selectItem(document.querySelector('#category-tabsSidebar'));
- document.defaultView.gotoPref('paneTabsSidebar');
- document.defaultView.gCategoryInits.get('paneTabsSidebar').init();
+ let done = false;
+ const initialize = () => {
+ document.querySelector('#categories').selectItem(document.querySelector('#category-tabsSidebar'));
+ document.defaultView.gotoPref('paneTabsSidebar');
+ document.defaultView.gCategoryInits.get('paneTabsSidebar').init();
+ done = true;
+ };
+ if (document.defaultView.gSearchResultsPane.inited) {
+ initialize();
+ }
+ else {
+ document.addEventListener('DOMContentLoaded', initialize, { once: true });
+ document.defaultView.setTimeout(1000, () => {
+ if (done)
+ return;
+ document.removeEventListener('DOMContentLoaded', initialize, { once: true });
+ initialize();
+ });
+ }
}
}
catch (error) {
@@ -995,7 +1011,7 @@ const AboutPreferencesWatcher = {
const startAt = Date.now();
const topWin = loadInfo.browsingContext.topChromeWindow;
const timer = topWin.setInterval(() => {
- if (Date.now() - startAt > 1000) {
+ if (Date.now() - startAt > 5000) {
// timeout
topWin.clearInterval(timer);
return;
diff --git a/waterfox/browser/components/sidebar/experiments/waterfoxBridge.js b/waterfox/browser/components/sidebar/experiments/waterfoxBridge.js
index e7d216415897b..8c15bbee132cb 100644
--- a/waterfox/browser/components/sidebar/experiments/waterfoxBridge.js
+++ b/waterfox/browser/components/sidebar/experiments/waterfoxBridge.js
@@ -71,7 +71,7 @@ const BrowserWindowWatcher = {
if (win.location.href.startsWith('chrome://browser/content/browser.xhtml')) {
const installed = this.installTabsSidebar(win);
if (installed) {
- this.patchToTabPreviewModules(win);
+ this.patchToTabHoverPreviewModules(win);
this.patchToPlacesModules(win);
this.patchToFullScreenModules(win);
win.addEventListener('DOMAudioPlaybackBlockStarted', this, { capture: true });
@@ -96,7 +96,7 @@ const BrowserWindowWatcher = {
if (win.location.href.startsWith('chrome://browser/content/browser.xhtml')) {
this.uninstallTabsSidebar(win);
try {
- this.unpatchTabPreviewModules(win);
+ this.unpatchTabHoverPreviewModules(win);
this.unpatchPlacesModules(win);
this.unpatchFullScreenModules(win);
win.removeEventListener('DOMAudioPlaybackBlockStarted', this, { capture: true });
@@ -216,9 +216,9 @@ const BrowserWindowWatcher = {
:root.tabs-sidebar-right #tabs-sidebar-moveRight {
display: none;
}
- @supports not -moz-bool-pref("userChrome.icon.disabled") {
- @supports -moz-bool-pref("userChrome.icon.menu") {
- @supports -moz-bool-pref("userChrome.icon.context_menu") {
+ @media not (-moz-bool-pref: "userChrome.icon.disabled") {
+ @media (-moz-bool-pref: "userChrome.icon.menu") {
+ @media (-moz-bool-pref: "userChrome.icon.context_menu") {
#tabs-sidebar-newTab {
list-style-image: var(--uc-new-tab-icon);
}
@@ -327,18 +327,24 @@ const BrowserWindowWatcher = {
transition: 0.8s margin-left ease-out,
0.8s margin-right ease-out;
}
- @supports -moz-bool-pref("userChrome.fullscreen.overlap") {
- @supports -moz-bool-pref("browser.fullscreen.autohide") {
+ @media (-moz-bool-pref: "userChrome.fullscreen.overlap") {
+ @media (-moz-bool-pref: "browser.fullscreen.autohide") {
:root[sizemode="fullscreen"] #tabs-sidebar-box,
:root[sizemode="fullscreen"] #tabs-sidebar-splitter {
+ height: 100% !important;
position: fixed !important;
z-index: 900 !important;
- height: 100% !important;
}
:root[sizemode="fullscreen"] #tabs-sidebar-splitter {
width: 5px !important;
z-index: 890 !important;
}
+ :root[sizemode="fullscreen"]:not(.tabs-sidebar-right) #sidebar-box:not([positionend="true"]) ~ #tabs-sidebar-splitter,
+ :root[sizemode="fullscreen"].tabs-sidebar-right #sidebar-box:not([positionend="true"]) ~ #tabs-sidebar-splitter,
+ :root[sizemode="fullscreen"]:not(.tabs-sidebar-right) #sidebar-box[positionend="true"] ~ #tabs-sidebar-splitter,
+ :root[sizemode="fullscreen"].tabs-sidebar-right #sidebar-box[positionend="true"] ~ #tabs-sidebar-splitter {
+ order: unset !important;
+ }
:root[sizemode="fullscreen"]:not(.tabs-sidebar-right) #tabs-sidebar-box,
:root[sizemode="fullscreen"]:not(.tabs-sidebar-right) #tabs-sidebar-splitter {
left: 0;
@@ -351,7 +357,7 @@ const BrowserWindowWatcher = {
}
@media (prefers-reduced-motion: no-preference) {
- @supports -moz-bool-pref("userChrome.decoration.animate") {
+ @media (-moz-bool-pref: "userChrome.decoration.animate") {
:root[sizemode="fullscreen"] #tabs-sidebar-box {
transition: margin-left 1.3s var(--animation-easing-function) 50ms,
margin-right 1.3s var(--animation-easing-function) 50ms !important;
@@ -366,11 +372,14 @@ const BrowserWindowWatcher = {
document.querySelector('#mainCommandSet').appendChild(element(document, XUL, 'command', {
id: 'toggle-tabs-sidebar-command',
}));
- document.querySelector('#mainKeyset').appendChild(element(document, XUL, 'key', {
- id: 'toggle-tabs-sidebar-key',
- keycode: 'VK_F1',
- command: 'toggle-tabs-sidebar-command',
- }));
+ // So, as a workaround we don't remove the key element on uninstallation and reuse existing key element.
+ if (!document.querySelector('#toggle-tabs-sidebar-key')) {
+ document.querySelector('#mainKeyset').appendChild(element(document, XUL, 'key', {
+ id: 'toggle-tabs-sidebar-key',
+ keycode: 'VK_F1',
+ command: 'toggle-tabs-sidebar-command',
+ }));
+ }
document.querySelector('#viewSidebarMenu').appendChild(element(document, XUL, 'menuseparator', {
id: 'viewmenu-tabs-sidebar-separator',
@@ -655,7 +664,8 @@ const BrowserWindowWatcher = {
for (const node of document.querySelectorAll(`
#tabs-sidebar-splitter,
#tabs-sidebar-box,
- #toggle-tabs-sidebar-key,
+ /* Don't remove key element because only the first element added is effective.*/
+ /*#toggle-tabs-sidebar-key,*/
#toggle-tabs-sidebar-command,
#viewmenu-tabs-sidebar-separator,
#viewmenu-toggle-tabs-sidebar,
@@ -697,17 +707,27 @@ const BrowserWindowWatcher = {
}
},
- patchToTabPreviewModules(win) {
- const tabPreview = win.document.querySelector('#tabbrowser-tab-preview');
+ patchToTabHoverPreviewModules(win) {
+ const tabs = win.document.getElementById('tabbrowser-tabs');
+ if (!tabs)
+ return;
+
+ if (!tabs._previewPanel) {
+ // https://searchfox.org/mozilla-esr128/rev/7998c47697fb3ba3e380eda1281c8280da81a0b1/browser/components/tabbrowser/content/tabs.js#187
+ const TabHoverPreviewPanel = ChromeUtils.importESModule('chrome://browser/content/tabbrowser/tab-hover-preview.mjs').default;
+ tabs._previewPanel = new TabHoverPreviewPanel(win.document.getElementById('tab-preview-panel'));
+ }
+
+ const tabPreview = tabs._previewPanel;
if (!tabPreview ||
- !tabPreview.showPreview)
+ !tabPreview.activate)
return;
tabPreview.__ws__calculateCoordinates = () => {
const tabsSidebar = win.document.querySelector('#tabs-sidebar');
const tabsSidebarRect = tabsSidebar.getBoundingClientRect();
const align = win.document.documentElement.classList.contains('tabs-sidebar-right') ? 'right' : 'left';
- const previewRect = tabPreview.panel.getBoundingClientRect();
+ const previewRect = tabPreview._panel.getBoundingClientRect();
const x = align == 'left' ?
tabsSidebar.screenX + tabsSidebarRect.width - 2 :
tabsSidebar.screenX - previewRect.width + 2;
@@ -715,32 +735,50 @@ const BrowserWindowWatcher = {
return [x, y];
};
- if (!tabPreview.__ws_orig__showPreview)
- tabPreview.__ws_orig__showPreview = tabPreview.showPreview;
- tabPreview.showPreview = function() {
- if (typeof this.__ws__top == 'number') {
- const [x, y] = tabPreview.__ws__calculateCoordinates();
- this.panel.openPopupAtScreen(x, y, false);
+ if (!tabPreview.__ws_orig__activate)
+ tabPreview.__ws_orig__activate = tabPreview.activate;
+ // https://searchfox.org/mozilla-esr128/rev/7998c47697fb3ba3e380eda1281c8280da81a0b1/browser/components/tabbrowser/content/tab-hover-preview.mjs#97
+ const self = this;
+ tabPreview.activate = function(tab) {
+ if (this._isDisabled()) {
+ return;
}
- else {
- this.panel.openPopup(this.tab, {
- position: 'bottomleft topleft',
- y: -2,
- isContextMenu: false,
- });
+ this._tab = tab;
+ this._movePanel();
+
+ this._thumbnailElement = null;
+ this._maybeRequestThumbnail();
+ if (this._panel.state == 'open') {
+ this._updatePreview();
}
- win.addEventListener('wheel', this, {
- capture: true,
- passive: true,
- });
- win.addEventListener('TabSelect', this);
- this.panel.addEventListener('popuphidden', this);
+ if (this._timer) {
+ return;
+ }
+ this._timer = this._win.setTimeout(() => {
+ this._timer = null;
+ if (self.shouldShowSidebar(win.document) &&
+ typeof this.__ws__top == 'number') {
+ const [x, y] = tabPreview.__ws__calculateCoordinates();
+ this._panel.openPopupAtScreen(x, y, false);
+ }
+ else {
+ this._panel.openPopup(this._tab, {
+ // POPUP_OPTIONS https://searchfox.org/mozilla-esr128/rev/7998c47697fb3ba3e380eda1281c8280da81a0b1/browser/components/tabbrowser/content/tab-hover-preview.mjs#9
+ position: 'bottomleft topleft',
+ x: 0,
+ y: -2,
+ });
+ }
+ }, this._prefPreviewDelay);
+ this._win.addEventListener('TabSelect', this);
+ this._panel.addEventListener('popupshowing', this);
};
- if (!tabPreview.panel.__ws_orig__moveToAnchor)
- tabPreview.panel.__ws_orig__moveToAnchor = tabPreview.panel.moveToAnchor;
- tabPreview.panel.moveToAnchor = function(...args) {
- if (typeof tabPreview.__ws__top == 'number') {
+ if (!tabPreview._panel.__ws_orig__moveToAnchor)
+ tabPreview._panel.__ws_orig__moveToAnchor = tabPreview._panel.moveToAnchor;
+ tabPreview._panel.moveToAnchor = function(...args) {
+ if (self.shouldShowSidebar(win.document) &&
+ typeof tabPreview.__ws__top == 'number') {
const [x, y] = tabPreview.__ws__calculateCoordinates();
this.moveTo(x, y);
return;
@@ -749,22 +787,29 @@ const BrowserWindowWatcher = {
};
},
- unpatchTabPreviewModules(win) {
- const tabPreview = win.document.querySelector('#tabbrowser-tab-preview');
+ unpatchTabHoverPreviewModules(win) {
+ const tabPreview = win.document.getElementById('tabbrowser-tabs')?._previewPanel;
if (!tabPreview ||
- !tabPreview.showPreview)
+ !tabPreview.activate)
return;
- if (tabPreview.__ws_orig__showPreview) {
- tabPreview.showPreview = tabPreview.__ws_orig__showPreview;
- tabPreview.__ws_orig__showPreview = null;
+ if (tabPreview.__ws_orig__activate) {
+ tabPreview.activate = tabPreview.__ws_orig__activate;
+ tabPreview.__ws_orig__activate = null;
+ }
+
+ if (tabPreview._panel?.__ws_orig__moveToAnchor) {
+ tabPreview._panel.moveToAnchor = tabPreview._panel.__ws_orig__moveToAnchor;
+ tabPreview._panel.__ws_orig__moveToAnchor = null;
}
},
patchToFullScreenModules(win) {
const FullScreen = win.FullScreen;
- const sidebarBox = win.document.querySelector('#tabs-sidebar-box');
- const sidebarSplitter = win.document.querySelector('#tabs-sidebar-splitter');
+ const sidebarMainBox = win.document.querySelector('#sidebar-main');
+ const sidebarBox = win.document.querySelector('#sidebar-box');
+ const tabsSidebarBox = win.document.querySelector('#tabs-sidebar-box');
+ const tabsSidebarSplitter = win.document.querySelector('#tabs-sidebar-splitter');
if (!FullScreen.__ws_orig__toggle)
FullScreen.__ws_orig__toggle = FullScreen.toggle;
@@ -775,7 +820,7 @@ const BrowserWindowWatcher = {
if (enterFS) {
if (!win.document.fullscreenElement) {
if (win.gNavToolbox.getAttribute('fullscreenShouldAnimate') == 'true')
- sidebarBox.setAttribute('fullscreenShouldAnimate', true);
+ tabsSidebarBox.setAttribute('fullscreenShouldAnimate', true);
FullScreen.__ws_sidebar.hide();
}
@@ -809,30 +854,32 @@ const BrowserWindowWatcher = {
updateMouseTargetRect() {
const contentsAreaRect = win.gBrowser.tabpanels.getBoundingClientRect();
+ const tabsSidebarBoxRect = tabsSidebarBox.getBoundingClientRect();
+ const sidebarMainBoxRect = sidebarMainBox.getBoundingClientRect();
const sidebarBoxRect = sidebarBox.getBoundingClientRect();
const isRight = this.isRightSide();
this._mouseTargetRect = {
top: contentsAreaRect.top,
bottom: contentsAreaRect.bottom,
- left: contentsAreaRect.left + (isRight ? 0 : sidebarBoxRect.width + 50),
- right: contentsAreaRect.right - (isRight ? sidebarBoxRect.width + 50 : 0),
+ left: Math.min(sidebarMainBoxRect.left, sidebarBoxRect.left, contentsAreaRect.left) + (isRight ? 0 : tabsSidebarBoxRect.width + 50),
+ right: Math.max(sidebarMainBoxRect.right, sidebarBoxRect.right, contentsAreaRect.right) - (isRight ? tabsSidebarBoxRect.width + 50 : 0),
};
win.MousePosTracker.addListener(this);
},
hide() {
if (this.isRightSide())
- sidebarBox.style.marginRight = `-${sidebarBox.getBoundingClientRect().width}px`;
+ tabsSidebarBox.style.marginRight = `-${tabsSidebarBox.getBoundingClientRect().width}px`;
else
- sidebarBox.style.marginLeft = `-${sidebarBox.getBoundingClientRect().width}px`;
+ tabsSidebarBox.style.marginLeft = `-${tabsSidebarBox.getBoundingClientRect().width}px`;
win.MousePosTracker.removeListener(this);
this.startListenToShow();
},
show({ exitting } = {}) {
- sidebarBox.removeAttribute('fullscreenShouldAnimate');
- sidebarBox.style.marginLeft = sidebarBox.style.marginRight = '';
+ tabsSidebarBox.removeAttribute('fullscreenShouldAnimate');
+ tabsSidebarBox.style.marginLeft = tabsSidebarBox.style.marginRight = '';
this.endListenToShow();
if (!exitting) {
@@ -849,9 +896,9 @@ const BrowserWindowWatcher = {
if (this.listeningToShow)
return;
this.listeningToShow = true;
- sidebarSplitter.addEventListener('mouseover', FullScreen.__ws_sidebar);
- sidebarSplitter.addEventListener('dragenter', FullScreen.__ws_sidebar);
- sidebarSplitter.addEventListener('touchmove', FullScreen.__ws_sidebar, {
+ tabsSidebarSplitter.addEventListener('mouseover', FullScreen.__ws_sidebar);
+ tabsSidebarSplitter.addEventListener('dragenter', FullScreen.__ws_sidebar);
+ tabsSidebarSplitter.addEventListener('touchmove', FullScreen.__ws_sidebar, {
passive: true,
});
},
@@ -860,9 +907,9 @@ const BrowserWindowWatcher = {
if (!this.listeningToShow)
return;
this.listeningToShow = false;
- sidebarSplitter.removeEventListener('mouseover', FullScreen.__ws_sidebar);
- sidebarSplitter.removeEventListener('dragenter', FullScreen.__ws_sidebar);
- sidebarSplitter.removeEventListener('touchmove', FullScreen.__ws_sidebar, {
+ tabsSidebarSplitter.removeEventListener('mouseover', FullScreen.__ws_sidebar);
+ tabsSidebarSplitter.removeEventListener('dragenter', FullScreen.__ws_sidebar);
+ tabsSidebarSplitter.removeEventListener('touchmove', FullScreen.__ws_sidebar, {
passive: true,
});
},
@@ -1070,10 +1117,12 @@ const BrowserWindowWatcher = {
break;
case 'tabs-sidebar-bookmarkTab':
- case 'tabs-sidebar-bookmarkTabs':
- win.PlacesUIUtils.showBookmarkPagesDialog(win.PlacesCommandHook.uniqueSelectedPages); // same to #toolbar-context-bookmarkSelectedTab / #toolbar-context-bookmarkSelectedTabs
+ case 'tabs-sidebar-bookmarkTabs': {
+ const pages = win.PlacesCommandHook.getUniquePages(win.gBrowser.selectedTabs)
+ .map(page => Object.assign(page, { uri: Services.io.createExposableURI(page.uri) }));
+ win.PlacesUIUtils.showBookmarkPagesDialog(pages); // same to #toolbar-context-bookmarkSelectedTab / #toolbar-context-bookmarkSelectedTabs
this.tryHidePopup(event);
- break;
+ }; break;
case 'tabs-sidebar-selectAll':
win.gBrowser.selectAllTabs(); // same to #toolbar-context-selectAllTabs
@@ -1130,16 +1179,22 @@ const BrowserWindowWatcher = {
moveToolbarButtonToDefaultPosition() {
try {
- const state = JSON.parse(Services.prefs.getStringPref('browser.uiCustomization.state', '{}'));
+ const rawState = Services.prefs.getStringPref('browser.uiCustomization.state', '{}');
+ if (!rawState)
+ return false;
+
+ const state = JSON.parse(rawState);
if (!state?.placements)
return false;
let foundInNavBar = false;
+ let found = false;
for (const [name, items] of Object.entries(state.placements)) {
if (!items.includes(this.id))
continue;
if (name == 'nav-bar')
foundInNavBar = true;
+ found = true;
break;
}
@@ -1154,9 +1209,17 @@ const BrowserWindowWatcher = {
if (foundInNavBar)
lazy.CustomizableUI.moveWidgetWithinArea(this.id, index);
- else
+ else if (!found)
lazy.CustomizableUI.addWidgetToArea(this.id, this.defaultArea, index);
+ const win = Services.wm.getMostRecentBrowserWindow();
+ win.setTimeout(() => {
+ if (win.document.getElementById(this.id))
+ return;
+ console.log('failed to insert tabs sidebar button due to unhandled error in CustomizableUI module: retrying with reset');
+ win.gCustomizeMode.reset();
+ }, 250); // for safety, this delay need to be large enough
+
return true;
}
catch (error) {
@@ -1223,13 +1286,18 @@ const BrowserWindowWatcher = {
}
}; break;
- case 'browser.uiCustomization.state':
+ case 'browser.uiCustomization.state': {
if (!Services.prefs.getStringPref(name)) { // resetting!
- Services.wm.getMostRecentBrowserWindow().setTimeout(() => {
+ const tryInsertButton = () => {
+ if (!Services.prefs.getStringPref(name)) {
+ Services.wm.getMostRecentBrowserWindow().setTimeout(tryInsertButton, 10);
+ return;
+ }
this.moveToolbarButtonToDefaultPosition();
- }, 100);
+ };
+ tryInsertButton();
}
- break;
+ }; break;
}
},
@@ -1546,17 +1614,15 @@ this.waterfoxBridge = class extends ExtensionAPI {
async showPreviewPanel(tabId, top) {
const tab = tabId && context.extension.tabManager.get(tabId);
- if (!tab ||
- !Services.prefs.getBoolPref('browser.tabs.cardPreview.enabled', false))
+ if (!tab)
return;
const document = tab.nativeTab.ownerDocument;
- const tabPreview = document.querySelector('#tabbrowser-tab-preview');
+ const tabPreview = document.getElementById('tabbrowser-tabs')?._previewPanel;
if (!tabPreview)
return;
tabPreview.__ws__top = top;
- tabPreview.tab = tab.nativeTab;
- tabPreview.showPreview();
+ tabPreview.activate(tab.nativeTab);
},
async hidePreviewPanel(windowId) {
@@ -1564,28 +1630,12 @@ this.waterfoxBridge = class extends ExtensionAPI {
if (!win)
return;
- const tabPreview = win.window.document.querySelector('#tabbrowser-tab-preview');
+ const tabPreview = document.getElementById('tabbrowser-tabs')?._previewPanel;
if (!tabPreview)
return;
tabPreview.__ws__top = null;
- tabPreview.tab = null;
},
- onHoverPreviewChanged: new EventManager({
- context,
- name: 'waterfoxBridge.onHoverPreviewChanged',
- register: (fire) => {
- const observe = (_subject, _topic, data) => {
- fire.async(Services.prefs.getBoolPref('browser.tabs.cardPreview.enabled', false)).catch(() => {}); // ignore Message Manager disconnects
- };
- Services.prefs.addObserver('browser.tabs.cardPreview.enabled', observe);
- observe();
- return () => {
- Services.prefs.removeObserver('browser.tabs.cardPreview.enabled', observe);
- };
- },
- }).api(),
-
async openPreferences() {
BrowserWindowWatcher.openOptions();
},
diff --git a/waterfox/browser/components/sidebar/experiments/waterfoxBridge.json b/waterfox/browser/components/sidebar/experiments/waterfoxBridge.json
index 692a383fdb73a..38cb4c8760291 100644
--- a/waterfox/browser/components/sidebar/experiments/waterfoxBridge.json
+++ b/waterfox/browser/components/sidebar/experiments/waterfoxBridge.json
@@ -434,17 +434,6 @@
}
]
},
- {
- "name": "onHoverPreviewChanged",
- "description": "Notified when hover preview is activated or deactivated.",
- "type": "function",
- "parameters": [
- {
- "type": "boolean",
- "name": "hoverPreviewEnabled"
- }
- ]
- },
{
"name": "onAutoplayBlocked",
"description": "Notified when a tab's autoplay is blocked.",
diff --git a/waterfox/browser/components/sidebar/manifest.json b/waterfox/browser/components/sidebar/manifest.json
index f86cfe07d445a..8466e134ec2a3 100644
--- a/waterfox/browser/components/sidebar/manifest.json
+++ b/waterfox/browser/components/sidebar/manifest.json
@@ -1,291 +1,272 @@
{
- "manifest_version": 2,
- "name": "__MSG_extensionName__",
- "version": "1.0.2.1",
- "author": "Waterfox",
- "hidden": true,
- "description": "__MSG_extensionDescription__",
- "permissions": [
- "activeTab",
- "bookmarks",
- "clipboardRead",
- "contextualIdentities",
- "cookies",
- "menus",
- "menus.overrideContext",
- "notifications",
- "scripting",
- "search",
- "sessions",
- "storage",
- "tabs",
- "theme"
- ],
- "optional_permissions": [
- ""
- ],
- "background": {
- "page": "background/background.html"
- },
- "commands": {
- "reloadTree": {
- "description": "__MSG_context_reloadTree_command__"
- },
- "reloadDescendants": {
- "description": "__MSG_context_reloadDescendants_command__"
- },
- "unblockAutoplayTree": {
- "description": "__MSG_context_unblockAutoplayTree_command__"
- },
- "unblockAutoplayDescendants": {
- "description": "__MSG_context_unblockAutoplayDescendants_command__"
- },
- "toggleMuteTree": {
- "description": "__MSG_context_toggleMuteTree_command__"
- },
- "toggleMuteDescendants": {
- "description": "__MSG_context_toggleMuteDescendants_command__"
- },
- "closeTree": {
- "description": "__MSG_context_closeTree_command__"
- },
- "closeDescendants": {
- "description": "__MSG_context_closeDescendants_command__"
- },
- "closeOthers": {
- "description": "__MSG_context_closeOthers_command__"
- },
- "toggleSticky": {
- "description": "__MSG_context_toggleSticky_command__"
- },
- "collapseTree": {
- "description": "__MSG_context_collapseTree_command__"
- },
- "collapseTreeRecursively": {
- "description": "__MSG_context_collapseTreeRecursively_command__"
- },
- "collapseAll": {
- "description": "__MSG_context_collapseAll_command__"
- },
- "expandTree": {
- "description": "__MSG_context_expandTree_command__"
- },
- "expandTreeRecursively": {
- "description": "__MSG_context_expandTreeRecursively_command__"
- },
- "expandAll": {
- "description": "__MSG_context_expandAll_command__"
- },
- "toggleTreeCollapsed": {
- "description": "__MSG_command_toggleTreeCollapsed__"
- },
- "toggleTreeCollapsedRecursively": {
- "description": "__MSG_command_toggleTreeCollapsedRecursively__"
- },
- "bookmarkTree": {
- "description": "__MSG_context_bookmarkTree_command__"
- },
- "newIndependentTab": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_independent_command__"
- },
- "newChildTab": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_child_command__"
- },
- "newChildTabTop": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_childTop_command__"
- },
- "newChildTabEnd": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_childEnd_command__"
- },
- "newSiblingTab": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_sibling_command__"
- },
- "newNextSiblingTab": {
- "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_nextSibling_command__"
- },
- "newContainerTab": {
- "description": "__MSG_tabbar_newTabWithContexualIdentity_tooltip__"
- },
- "tabMoveUp": {
- "description": "__MSG_command_tabMoveUp__"
- },
- "treeMoveUp": {
- "description": "__MSG_command_treeMoveUp__"
- },
- "tabMoveDown": {
- "description": "__MSG_command_tabMoveDown__"
- },
- "treeMoveDown": {
- "description": "__MSG_command_treeMoveDown__"
- },
- "focusPrevious": {
- "description": "__MSG_command_focusPrevious__"
- },
- "focusPreviousSilently": {
- "description": "__MSG_command_focusPreviousSilently__"
- },
- "focusNext": {
- "description": "__MSG_command_focusNext__"
- },
- "focusNextSilently": {
- "description": "__MSG_command_focusNextSilently__"
- },
- "focusParent": {
- "description": "__MSG_command_focusParent__"
- },
- "focusParentOrCollapse": {
- "description": "__MSG_command_focusParentOrCollapse__"
- },
- "focusFirstChild": {
- "description": "__MSG_command_focusFirstChild__"
- },
- "focusFirstChildOrExpand": {
- "description": "__MSG_command_focusFirstChildOrExpand__"
- },
- "focusLastChild": {
- "description": "__MSG_command_focusLastChild__"
- },
- "focusPreviousSibling": {
- "description": "__MSG_command_focusPreviousSibling__"
- },
- "focusNextSibling": {
- "description": "__MSG_command_focusNextSibling__"
- },
- "simulateUpOnTree": {
- "description": "__MSG_command_simulateUpOnTree__",
- "suggested_key": {
- "default": "Alt+Shift+Up"
- }
- },
- "simulateDownOnTree": {
- "description": "__MSG_command_simulateDownOnTree__",
- "suggested_key": {
- "default": "Alt+Shift+Down"
- }
- },
- "simulateLeftOnTree": {
- "description": "__MSG_command_simulateLeftOnTree__",
- "suggested_key": {
- "default": "Alt+Shift+Left"
- }
- },
- "simulateRightOnTree": {
- "description": "__MSG_command_simulateRightOnTree__",
- "suggested_key": {
- "default": "Alt+Shift+Right"
- }
- },
- "tabbarUp": {
- "description": "__MSG_command_tabbarUp__",
- "suggested_key": {
- "default": "Alt+Up"
- }
- },
- "tabbarPageUp": {
- "description": "__MSG_command_tabbarPageUp__",
- "suggested_key": {
- "default": "Alt+PageUp"
- }
- },
- "tabbarHome": {
- "description": "__MSG_command_tabbarHome__",
- "suggested_key": {
- "default": "Alt+Shift+Home"
- }
- },
- "tabbarDown": {
- "description": "__MSG_command_tabbarDown__",
- "suggested_key": {
- "default": "Alt+Down"
- }
- },
- "tabbarPageDown": {
- "description": "__MSG_command_tabbarPageDown__",
- "suggested_key": {
- "default": "Alt+PageDown"
- }
- },
- "tabbarEnd": {
- "description": "__MSG_command_tabbarEnd__",
- "suggested_key": {
- "default": "Alt+End"
- }
- },
- "toggleSubPanel": {
- "description": "__MSG_command_toggleSubPanel__",
- "suggested_key": {
- "default": "F2"
- }
- },
- "switchSubPanel": {
- "description": "__MSG_command_switchSubPanel__"
- },
- "increaseSubPanel": {
- "description": "__MSG_command_increaseSubPanel__"
- },
- "decreaseSubPanel": {
- "description": "__MSG_command_decreaseSubPanel__"
- }
- },
- "web_accessible_resources": [
- "/resources/group-tab.html*"
- ],
- "protocol_handlers": [
- {
- "protocol": "ext+ws",
- "name": "Waterfox",
- "uriTemplate": "/resources/protocol-handler.html?%s"
- }
- ],
- "experiment_apis": {
- "prefs": {
- "schema": "experiments/prefs.json",
- "parent": {
- "scopes": [
- "addon_parent"
- ],
- "paths": [
- [
- "prefs"
- ]
- ],
- "script": "experiments/prefs.js"
- }
- },
- "syncPrefs": {
- "schema": "experiments/syncPrefs.json",
- "child": {
- "scopes": [
- "addon_child"
- ],
- "paths": [
- [
- "syncPrefs"
- ]
- ],
- "script": "experiments/syncPrefs.js"
- }
- },
- "waterfoxBridge": {
- "schema": "experiments/waterfoxBridge.json",
- "parent": {
- "scopes": [
- "addon_parent"
- ],
- "paths": [
- [
- "waterfoxBridge"
- ]
- ],
- "script": "experiments/waterfoxBridge.js"
- }
- }
- },
- "default_locale": "en",
- "browser_specific_settings": {
- "gecko": {
- "id": "sidebar@waterfox.net",
- "strict_min_version": "115.0"
- }
- }
+ "manifest_version": 2,
+ "name": "__MSG_extensionName__",
+ "version": "1.0.3",
+ "author": "Waterfox",
+ "hidden": true,
+ "description": "__MSG_extensionDescription__",
+ "permissions": [
+ "activeTab",
+ "bookmarks",
+ "clipboardRead",
+ "contextualIdentities",
+ "cookies",
+ "menus",
+ "menus.overrideContext",
+ "notifications",
+ "search",
+ "sessions",
+ "storage",
+ "tabs",
+ "theme",
+ ""
+ ],
+ "background": {
+ "page": "background/background.html"
+ },
+ "commands": {
+ "reloadTree": {
+ "description": "__MSG_context_reloadTree_command__"
+ },
+ "reloadDescendants": {
+ "description": "__MSG_context_reloadDescendants_command__"
+ },
+ "unblockAutoplayTree": {
+ "description": "__MSG_context_unblockAutoplayTree_command__"
+ },
+ "unblockAutoplayDescendants": {
+ "description": "__MSG_context_unblockAutoplayDescendants_command__"
+ },
+ "toggleMuteTree": {
+ "description": "__MSG_context_toggleMuteTree_command__"
+ },
+ "toggleMuteDescendants": {
+ "description": "__MSG_context_toggleMuteDescendants_command__"
+ },
+ "closeTree": {
+ "description": "__MSG_context_closeTree_command__"
+ },
+ "closeDescendants": {
+ "description": "__MSG_context_closeDescendants_command__"
+ },
+ "closeOthers": {
+ "description": "__MSG_context_closeOthers_command__"
+ },
+ "toggleSticky": {
+ "description": "__MSG_context_toggleSticky_command__"
+ },
+ "collapseTree": {
+ "description": "__MSG_context_collapseTree_command__"
+ },
+ "collapseTreeRecursively": {
+ "description": "__MSG_context_collapseTreeRecursively_command__"
+ },
+ "collapseAll": {
+ "description": "__MSG_context_collapseAll_command__"
+ },
+ "expandTree": {
+ "description": "__MSG_context_expandTree_command__"
+ },
+ "expandTreeRecursively": {
+ "description": "__MSG_context_expandTreeRecursively_command__"
+ },
+ "expandAll": {
+ "description": "__MSG_context_expandAll_command__"
+ },
+ "toggleTreeCollapsed": {
+ "description": "__MSG_command_toggleTreeCollapsed__"
+ },
+ "toggleTreeCollapsedRecursively": {
+ "description": "__MSG_command_toggleTreeCollapsedRecursively__"
+ },
+ "bookmarkTree": {
+ "description": "__MSG_context_bookmarkTree_command__"
+ },
+ "newIndependentTab": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_independent_command__"
+ },
+ "newChildTab": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_child_command__"
+ },
+ "newChildTabTop": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_childTop_command__"
+ },
+ "newChildTabEnd": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_childEnd_command__"
+ },
+ "newSiblingTab": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_sibling_command__"
+ },
+ "newNextSiblingTab": {
+ "description": "__MSG_tabbar_newTabButton_tooltip__: __MSG_tabbar_newTabAction_nextSibling_command__"
+ },
+ "newContainerTab": {
+ "description": "__MSG_tabbar_newTabWithContexualIdentity_tooltip__"
+ },
+ "tabMoveUp": {
+ "description": "__MSG_command_tabMoveUp__"
+ },
+ "treeMoveUp": {
+ "description": "__MSG_command_treeMoveUp__"
+ },
+ "tabMoveDown": {
+ "description": "__MSG_command_tabMoveDown__"
+ },
+ "treeMoveDown": {
+ "description": "__MSG_command_treeMoveDown__"
+ },
+ "focusPrevious": {
+ "description": "__MSG_command_focusPrevious__"
+ },
+ "focusPreviousSilently": {
+ "description": "__MSG_command_focusPreviousSilently__"
+ },
+ "focusNext": {
+ "description": "__MSG_command_focusNext__"
+ },
+ "focusNextSilently": {
+ "description": "__MSG_command_focusNextSilently__"
+ },
+ "focusParent": {
+ "description": "__MSG_command_focusParent__"
+ },
+ "focusParentOrCollapse": {
+ "description": "__MSG_command_focusParentOrCollapse__"
+ },
+ "focusFirstChild": {
+ "description": "__MSG_command_focusFirstChild__"
+ },
+ "focusFirstChildOrExpand": {
+ "description": "__MSG_command_focusFirstChildOrExpand__"
+ },
+ "focusLastChild": {
+ "description": "__MSG_command_focusLastChild__"
+ },
+ "focusPreviousSibling": {
+ "description": "__MSG_command_focusPreviousSibling__"
+ },
+ "focusNextSibling": {
+ "description": "__MSG_command_focusNextSibling__"
+ },
+ "simulateUpOnTree": {
+ "description": "__MSG_command_simulateUpOnTree__",
+ "suggested_key": {
+ "default": "Alt+Shift+Up",
+ "mac": "MacCtrl+Shift+Up"
+ }
+ },
+ "simulateDownOnTree": {
+ "description": "__MSG_command_simulateDownOnTree__",
+ "suggested_key": {
+ "default": "Alt+Shift+Down",
+ "mac": "MacCtrl+Shift+Down"
+ }
+ },
+ "simulateLeftOnTree": {
+ "description": "__MSG_command_simulateLeftOnTree__",
+ "suggested_key": {
+ "default": "Alt+Shift+Left",
+ "mac": "MacCtrl+Shift+Left"
+ }
+ },
+ "simulateRightOnTree": {
+ "description": "__MSG_command_simulateRightOnTree__",
+ "suggested_key": {
+ "default": "Alt+Shift+Right",
+ "mac": "MacCtrl+Shift+Right"
+ }
+ },
+ "tabbarUp": {
+ "description": "__MSG_command_tabbarUp__",
+ "suggested_key": {
+ "default": "Alt+Up"
+ }
+ },
+ "tabbarPageUp": {
+ "description": "__MSG_command_tabbarPageUp__",
+ "suggested_key": {
+ "default": "Alt+PageUp"
+ }
+ },
+ "tabbarHome": {
+ "description": "__MSG_command_tabbarHome__",
+ "suggested_key": {
+ "default": "Alt+Shift+Home"
+ }
+ },
+ "tabbarDown": {
+ "description": "__MSG_command_tabbarDown__",
+ "suggested_key": {
+ "default": "Alt+Down"
+ }
+ },
+ "tabbarPageDown": {
+ "description": "__MSG_command_tabbarPageDown__",
+ "suggested_key": {
+ "default": "Alt+PageDown"
+ }
+ },
+ "tabbarEnd": {
+ "description": "__MSG_command_tabbarEnd__",
+ "suggested_key": {
+ "default": "Alt+End"
+ }
+ },
+ "toggleSubPanel": {
+ "description": "__MSG_command_toggleSubPanel__",
+ "suggested_key": {
+ "default": "F2"
+ }
+ },
+ "switchSubPanel": {
+ "description": "__MSG_command_switchSubPanel__"
+ },
+ "increaseSubPanel": {
+ "description": "__MSG_command_increaseSubPanel__"
+ },
+ "decreaseSubPanel": {
+ "description": "__MSG_command_decreaseSubPanel__"
+ }
+ },
+ "web_accessible_resources": ["/resources/group-tab.html*"],
+ "protocol_handlers": [
+ {
+ "protocol": "ext+ws",
+ "name": "Waterfox",
+ "uriTemplate": "/resources/protocol-handler.html?%s"
+ }
+ ],
+ "experiment_apis": {
+ "prefs": {
+ "schema": "experiments/prefs.json",
+ "parent": {
+ "scopes": ["addon_parent"],
+ "paths": [["prefs"]],
+ "script": "experiments/prefs.js"
+ }
+ },
+ "syncPrefs": {
+ "schema": "experiments/syncPrefs.json",
+ "child": {
+ "scopes": ["addon_child"],
+ "paths": [["syncPrefs"]],
+ "script": "experiments/syncPrefs.js"
+ }
+ },
+ "waterfoxBridge": {
+ "schema": "experiments/waterfoxBridge.json",
+ "parent": {
+ "scopes": ["addon_parent"],
+ "paths": [["waterfoxBridge"]],
+ "script": "experiments/waterfoxBridge.js"
+ }
+ }
+ },
+ "default_locale": "en",
+ "browser_specific_settings": {
+ "gecko": {
+ "id": "sidebar@waterfox.net",
+ "strict_min_version": "128.0"
+ }
+ }
}
diff --git a/waterfox/browser/components/sidebar/options/init.js b/waterfox/browser/components/sidebar/options/init.js
index 6b32944b93fed..cf988b81aa6fe 100644
--- a/waterfox/browser/components/sidebar/options/init.js
+++ b/waterfox/browser/components/sidebar/options/init.js
@@ -55,7 +55,8 @@ const options = new Options(configs, {
});
document.title = browser.i18n.getMessage('config_title');
-if ((location.hash && location.hash != '#') ||
+if ((location.hash &&
+ /^#!?$/.test(location.hash)) ||
/independent=true/.test(location.search))
document.body.classList.add('independent');
@@ -139,6 +140,8 @@ let mUserStyleRulesFieldEditor;
const mDarkModeMedia = window.matchMedia('(prefers-color-scheme: dark)');
+let mShowExpertOptionsTemporarily = false;
+
function onConfigChanged(key) {
const value = configs[key];
switch (key) {
@@ -181,19 +184,24 @@ function onConfigChanged(key) {
}
}; break;
- case 'showExpertOptions':
- document.documentElement.classList.toggle('show-expert-options', configs.showExpertOptions);
+ case 'showExpertOptions': {
+ if (mShowExpertOptionsTemporarily && !configs.showExpertOptions)
+ document.querySelector('#showExpertOptions').checked = true;
+ const show = mShowExpertOptionsTemporarily || configs.showExpertOptions;
+ document.documentElement.classList.toggle('show-expert-options', show);
for (const item of document.querySelectorAll('#parentTabOperationBehaviorModeGroup li li')) {
const radio = item.querySelector('input[type="radio"]');
- if (configs.showExpertOptions || radio.checked) {
+ if (show || radio.checked) {
item.style.display = '';
- radio.style.display = configs.showExpertOptions ? '' : 'none';
+ radio.style.display = show || radio.checked ? '' : 'none';
}
else {
- item.style.display = radio.style.display = 'none';
+ item.style.display = radio.style.display = 'none';
}
}
- break;
+ if (mShowExpertOptionsTemporarily && !configs.showExpertOptions)
+ mShowExpertOptionsTemporarily = false;
+ }; break;
case 'syncDeviceInfo': {
const name = (configs.syncDeviceInfo || {}).name || '';
@@ -614,6 +622,12 @@ window.addEventListener('DOMContentLoaded', async () => {
console.error(error);
}
+ mShowExpertOptionsTemporarily = !!(
+ location.hash &&
+ !/^#!?$/.test(location.hash) &&
+ document.querySelector(`.expert #${location.hash.replace(/^#!?/, '')}, .expert#${location.hash.replace(/^#!?/, '')}`)
+ );
+
try {
options.buildUIForAllConfigs(document.querySelector('#group-allConfigs'));
onConfigChanged('successorTabControlLevel');
@@ -801,6 +815,9 @@ function initPermissionOptions() {
Permissions.CLIPBOARD_READ,
document.querySelector('#clipboardReadPermissionGranted_middleClickPasteURLOnNewTabButton'),
{
+ onInitialized: (granted) => {
+ return granted && configs.middleClickPasteURLOnNewTabButton;
+ },
onChanged: (granted) => {
configs.middleClickPasteURLOnNewTabButton = granted;
}
diff --git a/waterfox/browser/components/sidebar/options/options.css b/waterfox/browser/components/sidebar/options/options.css
index ce60bc4ba183e..797fdde652ced 100644
--- a/waterfox/browser/components/sidebar/options/options.css
+++ b/waterfox/browser/components/sidebar/options/options.css
@@ -131,7 +131,7 @@ label.has-radio:not(.inline) {
label input[type="radio"] ~ img,
label input[type="checkbox"] ~ img {
- border: 1px solid ThreeDShadow;
+ border: 1px solid var(--ThreeDShadow);
vertical-align: middle;
margin-top: 0.15em;
margin-bottom: 0.15em;
diff --git a/waterfox/browser/components/sidebar/options/options.html b/waterfox/browser/components/sidebar/options/options.html
index b1d0951566d92..97bd776775a82 100644
--- a/waterfox/browser/components/sidebar/options/options.html
+++ b/waterfox/browser/components/sidebar/options/options.html
@@ -770,6 +770,16 @@ __MSG_config_newTab_caption__
__MSG_groupTab_temporaryAggressive_label__
__MSG_config_groupTabTemporaryState_option_checked_after__
+
@@ -1286,6 +1296,12 @@ __MSG_config_drag_caption__
__MSG_config_insertDroppedTabsAt_end__
+
+
+