always restore all tree
This commit is contained in:
parent
b8fd343863
commit
8d981a7d13
@ -559,15 +559,6 @@ pref("extensions.treestyletab.undoCloseTabSet.behavior", 3);
|
|||||||
*/
|
*/
|
||||||
pref("extensions.treestyletab.repositionStatusPanel", true);
|
pref("extensions.treestyletab.repositionStatusPanel", true);
|
||||||
|
|
||||||
/**
|
|
||||||
* "Fast restore" of tree structure on the startup. Recommended value is "1".
|
|
||||||
* 0 = Restore all trees based on SSTabRestoring.
|
|
||||||
* 1 = Restore trees in the current group (Panorama) before SSTabRestoring.
|
|
||||||
* Others (tabs in background groups) are restored based on SSTabRestoring.
|
|
||||||
* 2 = Restore all trees including background groups before SSTabRestoring.
|
|
||||||
*/
|
|
||||||
pref("extensions.treestyletab.fastRestoreTree.level", 1);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* TST overrides some internal prefs of Firefox itself, because they can
|
* TST overrides some internal prefs of Firefox itself, because they can
|
||||||
* conflict with TST features. They will be rolled back when TST is uninstalled.
|
* conflict with TST features. They will be rolled back when TST is uninstalled.
|
||||||
|
@ -783,6 +783,10 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
aTab.setAttribute(this.kALLOW_COLLAPSE, true);
|
aTab.setAttribute(this.kALLOW_COLLAPSE, true);
|
||||||
let self = this;
|
let self = this;
|
||||||
this.Deferred.next(function() {
|
this.Deferred.next(function() {
|
||||||
|
// changed by someone!
|
||||||
|
if (self.getAttribute(self.kID) != id)
|
||||||
|
return;
|
||||||
|
|
||||||
if (!self.getTabValue(aTab, self.kID)) {
|
if (!self.getTabValue(aTab, self.kID)) {
|
||||||
self.setTabValue(aTab, self.kID, id);
|
self.setTabValue(aTab, self.kID, id);
|
||||||
if (!(id in self.tabsHash))
|
if (!(id in self.tabsHash))
|
||||||
@ -2280,48 +2284,62 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
if (!this.window.__SS_tabsToRestore || this.window.__SS_tabsToRestore <= 1)
|
if (!this.window.__SS_tabsToRestore || this.window.__SS_tabsToRestore <= 1)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var level = this.getTreePref('fastRestoreTree.level');
|
|
||||||
if (level <= this.kFAST_RESTORE_NONE)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var restoreOnlyCurrentGroup = level < this.kFAST_RESTORE_ALL;
|
|
||||||
var tabs = this.getAllTabsArray(this.mTabBrowser);
|
var tabs = this.getAllTabsArray(this.mTabBrowser);
|
||||||
tabs.reverse().filter(function(aTab) {
|
tabs = tabs.filter(function(aTab) {
|
||||||
var id = this.getTabValue(aTab, this.kID);
|
var id = this.getTabValue(aTab, this.kID);
|
||||||
if (
|
if (
|
||||||
!id || // tabs opened by externals applications
|
!id || // tabs opened by externals applications
|
||||||
(restoreOnlyCurrentGroup && aTab.hidden) // tabs in background groups
|
!aTab.linkedBrowser.__SS_restoreState
|
||||||
)
|
)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
var alreadyRestored = id == aTab.getAttribute(this.kID);
|
var currentId = aTab.getAttribute(this.kID);
|
||||||
|
var restored = id == currentId;
|
||||||
aTab.setAttribute(this.kID, id);
|
if (!restored) {
|
||||||
this.tabsHash[id] = aTab;
|
delete this.tabsHash[id];
|
||||||
|
aTab.setAttribute(this.kID, id);
|
||||||
var subTreeCollapsed = this.getTabValue(aTab, this.kSUBTREE_COLLAPSED) == 'true';
|
this.tabsHash[id] = aTab;
|
||||||
var children = this.getTabValue(aTab, this.kCHILDREN);
|
|
||||||
if (children) {
|
|
||||||
subTreeCollapsed = this._restoreSubtreeCollapsedState(aTab, subTreeCollapsed);
|
|
||||||
children.split('|').forEach(function(aChild) {
|
|
||||||
aChild = this.getTabById(aChild);
|
|
||||||
if (aChild) {
|
|
||||||
this.attachTabTo(aChild, aTab, {
|
|
||||||
forceExpand : true, // to prevent to collapse the selected tab
|
|
||||||
dontAnimate : true,
|
|
||||||
insertBefore : this.getTabById(this.getTabValue(aChild, this.kINSERT_BEFORE))
|
|
||||||
});
|
|
||||||
this.collapseExpandTab(aChild, subTreeCollapsed, true);
|
|
||||||
}
|
|
||||||
}, this);
|
|
||||||
this.collapseExpandSubtree(aTab, subTreeCollapsed, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!alreadyRestored)
|
aTab.__treestyletab__structureRestored = true;
|
||||||
aTab.__treestyletab__structureRestored = true;
|
|
||||||
|
|
||||||
return true
|
aTab.removeAttribute(this.kPARENT);
|
||||||
}, this).forEach(this.updateInsertionPositionInfo, this);
|
aTab.removeAttribute(this.kCHILDREN);
|
||||||
|
aTab.removeAttribute(this.kSUBTREE_COLLAPSED);
|
||||||
|
aTab.removeAttribute(this.kCOLLAPSED);
|
||||||
|
aTab.removeAttribute(this.kNEST);
|
||||||
|
this.updateTabCollapsed(aTab, false, true);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}, this);
|
||||||
|
|
||||||
|
this.updateAllTabsIndent(true);
|
||||||
|
|
||||||
|
tabs.reverse()
|
||||||
|
.filter(this.restoreOneTab, this)
|
||||||
|
.forEach(this.updateInsertionPositionInfo, this);
|
||||||
|
},
|
||||||
|
restoreOneTab : function TSTBrowser_restoreOneTab(aTab)
|
||||||
|
{
|
||||||
|
let subTreeCollapsed = this.getTabValue(aTab, this.kSUBTREE_COLLAPSED) == 'true';
|
||||||
|
let children = this.getTabValue(aTab, this.kCHILDREN);
|
||||||
|
this.deleteTabValue(aTab, this.kCHILDREN);
|
||||||
|
if (children) {
|
||||||
|
subTreeCollapsed = this._restoreSubtreeCollapsedState(aTab, subTreeCollapsed);
|
||||||
|
children.split('|').forEach(function(aChild) {
|
||||||
|
aChild = this.getTabById(aChild);
|
||||||
|
if (aChild) {
|
||||||
|
this.attachTabTo(aChild, aTab, {
|
||||||
|
forceExpand : true, // to prevent to collapse the selected tab
|
||||||
|
dontAnimate : true,
|
||||||
|
insertBefore : this.getTabById(this.getTabValue(aChild, this.kINSERT_BEFORE))
|
||||||
|
});
|
||||||
|
this.collapseExpandTab(aChild, subTreeCollapsed, true);
|
||||||
|
}
|
||||||
|
}, this);
|
||||||
|
this.collapseExpandSubtree(aTab, subTreeCollapsed, true);
|
||||||
|
}
|
||||||
|
return true
|
||||||
},
|
},
|
||||||
|
|
||||||
/* DOM Event Handling */
|
/* DOM Event Handling */
|
||||||
@ -3193,7 +3211,7 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
|
|
||||||
onTabRestoring : function TSTBrowser_onTabRestoring(aEvent)
|
onTabRestoring : function TSTBrowser_onTabRestoring(aEvent)
|
||||||
{
|
{
|
||||||
this.restoreStructure(aEvent.originalTarget);
|
this.handleRestoredTab(aEvent.originalTarget);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Updating of the counter which is used to know how many tabs were
|
* Updating of the counter which is used to know how many tabs were
|
||||||
@ -3239,7 +3257,7 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
RESTORED_TREE_COLLAPSED_STATE_LAST_STATE : -1,
|
RESTORED_TREE_COLLAPSED_STATE_LAST_STATE : -1,
|
||||||
RESTORED_TREE_COLLAPSED_STATE_COLLAPSED : 0,
|
RESTORED_TREE_COLLAPSED_STATE_COLLAPSED : 0,
|
||||||
RESTORED_TREE_COLLAPSED_STATE_EXPANDED : 1,
|
RESTORED_TREE_COLLAPSED_STATE_EXPANDED : 1,
|
||||||
restoreStructure : function TSTBrowser_restoreStructure(aTab)
|
handleRestoredTab : function TSTBrowser_handleRestoredTab(aTab)
|
||||||
{
|
{
|
||||||
var [id, mayBeDuplicated] = this._restoreTabId(aTab);
|
var [id, mayBeDuplicated] = this._restoreTabId(aTab);
|
||||||
|
|
||||||
@ -3263,8 +3281,12 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
|
|
||||||
var closeSetId = !structureRestored && this._getCloseSetId(aTab, mayBeDuplicated);
|
var closeSetId = !structureRestored && this._getCloseSetId(aTab, mayBeDuplicated);
|
||||||
|
|
||||||
this.setTabValue(aTab, this.kID, id);
|
var currentId = aTab.getAttribute(this.kID);
|
||||||
this.tabsHash[id] = aTab;
|
if (id != currentId) {
|
||||||
|
delete this.tabsHash[currentId];
|
||||||
|
this.setTabValue(aTab, this.kID, id);
|
||||||
|
this.tabsHash[id] = aTab;
|
||||||
|
}
|
||||||
|
|
||||||
if (structureRestored) {
|
if (structureRestored) {
|
||||||
/**
|
/**
|
||||||
@ -3274,7 +3296,6 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
* values.
|
* values.
|
||||||
*/
|
*/
|
||||||
[
|
[
|
||||||
this.kPARENT,
|
|
||||||
this.kINSERT_BEFORE,
|
this.kINSERT_BEFORE,
|
||||||
this.kINSERT_AFTER
|
this.kINSERT_AFTER
|
||||||
].forEach(function(aKey) {
|
].forEach(function(aKey) {
|
||||||
@ -3283,21 +3304,24 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
this.setTabValue(aTab, aKey, tab);
|
this.setTabValue(aTab, aKey, tab);
|
||||||
}, this);
|
}, this);
|
||||||
|
|
||||||
let children = this.getTabValue(aTab, this.kCHILDREN);
|
let parentId = this.getTabValue(aTab, this.kPARENT);
|
||||||
if (children.split('|').every(function(aChild) {
|
let parentTab = this.getTabById(parentId);
|
||||||
return this.getTabById(aChild);
|
if (parentTab && parentTab._tPos < aTab._tPos)
|
||||||
}, this))
|
this.setTabValue(aTab, this.kPARENT, parentId);
|
||||||
this.setTabValue(aTab, this.kCHILDREN, children);
|
else
|
||||||
|
this.deleteTabValue(aTab, this.kPARENT);
|
||||||
|
|
||||||
[
|
let ancestors = [aTab].concat(this.getAncestorTabs(aTab));
|
||||||
this.kSUBTREE_COLLAPSED,
|
let children = this.getTabValue(aTab, this.kCHILDREN);
|
||||||
this.kCOLLAPSED,
|
children = children.split('|').filter(function(aChild) {
|
||||||
this.kCOLLAPSED_DONE
|
let tab = this.getTabById(aChild);
|
||||||
].forEach(function(aKey) {
|
return ancestors.indexOf(tab) < 0;
|
||||||
var storedValue = this.getTabValue(aTab, aKey);
|
|
||||||
if (storedValue)
|
|
||||||
this.setTabValue(aTab, aKey, storedValue);
|
|
||||||
}, this);
|
}, this);
|
||||||
|
this.setTabValue(aTab, this.kCHILDREN, children.join('|'));
|
||||||
|
|
||||||
|
let subtreeCollapsed = this.getTabValue(aTab, this.kSUBTREE_COLLAPSED);
|
||||||
|
if (subtreeCollapsed != aTab.getAttribute(this.kSUBTREE_COLLAPSED))
|
||||||
|
this.collapseExpandSubtree(aTab, subtreeCollapsed == 'true', true);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (closeSetId)
|
if (closeSetId)
|
||||||
@ -4374,18 +4398,16 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
currentParent = aParent;
|
// avoid recursive tree
|
||||||
do {
|
var ancestors = [aParent].concat(this.getAncestorTabs(aChild));
|
||||||
if (currentParent != aChild) continue;
|
if (ancestors.indexOf(aChild) > -1)
|
||||||
// this.fireAttachedEvent(aChild, aParent);
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
while (currentParent = this.getParentTab(currentParent));
|
|
||||||
|
|
||||||
shouldInheritIndent = (
|
currentParent = ancestors[ancestors.length-1];
|
||||||
!currentParent ||
|
var shouldInheritIndent = (
|
||||||
(currentParent.getAttribute(this.kNEST) == aParent.getAttribute(this.kNEST))
|
!currentParent ||
|
||||||
);
|
(currentParent.getAttribute(this.kNEST) == aParent.getAttribute(this.kNEST))
|
||||||
|
);
|
||||||
|
|
||||||
this.ensureTabInitialized(aChild);
|
this.ensureTabInitialized(aChild);
|
||||||
this.ensureTabInitialized(aParent);
|
this.ensureTabInitialized(aParent);
|
||||||
@ -5288,9 +5310,9 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
// !this.canCollapseSubtree(this.getParentTab(aTab))
|
// !this.canCollapseSubtree(this.getParentTab(aTab))
|
||||||
) {
|
) {
|
||||||
if (aCollapsed)
|
if (aCollapsed)
|
||||||
this.setTabValue(aTab, this.kCOLLAPSED_DONE, true);
|
aTab.setAttribute(this.kCOLLAPSED_DONE, true);
|
||||||
else
|
else
|
||||||
this.deleteTabValue(aTab, this.kCOLLAPSED_DONE);
|
aTab.removeAttribute(this.kCOLLAPSED_DONE);
|
||||||
aTab.removeAttribute(this.kCOLLAPSING_PHASE);
|
aTab.removeAttribute(this.kCOLLAPSING_PHASE);
|
||||||
|
|
||||||
if (CSSTransitionEnabled) {
|
if (CSSTransitionEnabled) {
|
||||||
@ -5315,7 +5337,7 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
|
|
||||||
if (!aCollapsed) {
|
if (!aCollapsed) {
|
||||||
aTab.setAttribute(offsetAttr, maxMargin);
|
aTab.setAttribute(offsetAttr, maxMargin);
|
||||||
this.deleteTabValue(aTab, this.kCOLLAPSED_DONE);
|
aTab.removeAttribute(this.kCOLLAPSED_DONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
var radian = 90 * Math.PI / 180;
|
var radian = 90 * Math.PI / 180;
|
||||||
@ -5342,7 +5364,7 @@ TreeStyleTabBrowser.prototype = {
|
|||||||
}
|
}
|
||||||
if (aTime >= aDuration || stopAnimation) {
|
if (aTime >= aDuration || stopAnimation) {
|
||||||
delete aTab.__treestyletab__updateTabCollapsedTask;
|
delete aTab.__treestyletab__updateTabCollapsedTask;
|
||||||
if (aCollapsed) self.setTabValue(aTab, self.kCOLLAPSED_DONE, true);
|
if (aCollapsed) aTab.setAttribute(self.kCOLLAPSED_DONE, true);
|
||||||
if (!CSSTransitionEnabled) {
|
if (!CSSTransitionEnabled) {
|
||||||
aTab.style.removeProperty(self.collapseCSSProp);
|
aTab.style.removeProperty(self.collapseCSSProp);
|
||||||
aTab.style.removeProperty('opacity');
|
aTab.style.removeProperty('opacity');
|
||||||
|
@ -243,10 +243,6 @@ var TreeStyleTabUtils = {
|
|||||||
kCLOSE_PARENT_BEHAVIOR_SIMPLY_DETACH_ALL_CHILDREN : 4,
|
kCLOSE_PARENT_BEHAVIOR_SIMPLY_DETACH_ALL_CHILDREN : 4,
|
||||||
kCLOSE_PARENT_BEHAVIOR_CLOSE_ALL_CHILDREN : 2, // onTabRemoved only
|
kCLOSE_PARENT_BEHAVIOR_CLOSE_ALL_CHILDREN : 2, // onTabRemoved only
|
||||||
|
|
||||||
kFAST_RESTORE_NONE : 0,
|
|
||||||
kFAST_RESTORE_CURRENT_GROUP : 1,
|
|
||||||
kFAST_RESTORE_ALL : 2,
|
|
||||||
|
|
||||||
MAX_TABBAR_SIZE_RATIO : 0.8,
|
MAX_TABBAR_SIZE_RATIO : 0.8,
|
||||||
DEFAULT_SHRUNKEN_WIDTH_RATIO : 0.67,
|
DEFAULT_SHRUNKEN_WIDTH_RATIO : 0.67,
|
||||||
|
|
||||||
@ -1908,27 +1904,53 @@ var TreeStyleTabUtils = {
|
|||||||
{
|
{
|
||||||
if (!aTab) return null;
|
if (!aTab) return null;
|
||||||
|
|
||||||
|
var parent;
|
||||||
if (this.tabsHash) { // XPath-less implementation
|
if (this.tabsHash) { // XPath-less implementation
|
||||||
let parent = this.getTabById(aTab.getAttribute(this.kPARENT));
|
parent = this.getTabById(aTab.getAttribute(this.kPARENT));
|
||||||
return (parent && parent != aTab) ? parent : null ;
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
return this.evaluateXPath(
|
parent = this.evaluateXPath(
|
||||||
'preceding-sibling::xul:tab[@'+this.kID+'="'+aTab.getAttribute(this.kPARENT)+'"][1]',
|
'preceding-sibling::xul:tab[@'+this.kID+'="'+aTab.getAttribute(this.kPARENT)+'"][1]',
|
||||||
aTab,
|
aTab,
|
||||||
Ci.nsIDOMXPathResult.FIRST_ORDERED_NODE_TYPE
|
Ci.nsIDOMXPathResult.FIRST_ORDERED_NODE_TYPE
|
||||||
).singleNodeValue;
|
).singleNodeValue;
|
||||||
|
}
|
||||||
|
return (parent && parent != aTab) ? parent : null ;
|
||||||
},
|
},
|
||||||
|
|
||||||
getAncestorTabs : function TSTUtils_getAncestorTabs(aTab) /* PUBLIC API */
|
getAncestorTabs : function TSTUtils_getAncestorTabs(aTab) /* PUBLIC API */
|
||||||
{
|
{
|
||||||
var tabs = [];
|
var tabs = [aTab];
|
||||||
var parentTab = aTab;
|
var parentTab = aTab;
|
||||||
while (parentTab = this.getParentTab(parentTab))
|
while (parentTab = this.getParentTab(parentTab))
|
||||||
{
|
{
|
||||||
|
if (tabs.indexOf(parentTab) > -1) {
|
||||||
|
let message = 'recursive tree detected!\n'+
|
||||||
|
tabs.concat([parentTab])
|
||||||
|
.reverse().map(function(aTab) {
|
||||||
|
return ' '+aTab._tPos+' : '+
|
||||||
|
aTab.label+'\n '+
|
||||||
|
aTab.getAttribute(this.kID);
|
||||||
|
}, this).join('\n');
|
||||||
|
dump(message+'\n');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (aTab._tPos < parentTab._tPos) {
|
||||||
|
let message = 'broken tree detected!\n'+
|
||||||
|
tabs.concat([parentTab])
|
||||||
|
.reverse().map(function(aTab) {
|
||||||
|
return ' '+aTab._tPos+' : '+
|
||||||
|
aTab.label+'\n '+
|
||||||
|
aTab.getAttribute(this.kID);
|
||||||
|
}, this).join('\n');
|
||||||
|
dump(message+'\n');
|
||||||
|
}
|
||||||
|
|
||||||
tabs.push(parentTab);
|
tabs.push(parentTab);
|
||||||
|
aTab = parentTab;
|
||||||
}
|
}
|
||||||
return tabs;
|
return tabs.slice(1);
|
||||||
},
|
},
|
||||||
|
|
||||||
getRootTab : function TSTUtils_getRootTab(aTab) /* PUBLIC API */
|
getRootTab : function TSTUtils_getRootTab(aTab) /* PUBLIC API */
|
||||||
@ -1936,13 +1958,8 @@ var TreeStyleTabUtils = {
|
|||||||
if (!aTab) return null;
|
if (!aTab) return null;
|
||||||
|
|
||||||
if (this.tabsHash) { // XPath-less implementation
|
if (this.tabsHash) { // XPath-less implementation
|
||||||
let parent = aTab;
|
let ancestors = this.getAncestorTabs(aTab);
|
||||||
let root = aTab;
|
return ancestors.length ? ancestors[ancestors.length-1] : aTab ;
|
||||||
while (parent = this.getParentTab(parent))
|
|
||||||
{
|
|
||||||
root = parent;
|
|
||||||
}
|
|
||||||
return root;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.evaluateXPath(
|
return this.evaluateXPath(
|
||||||
@ -2162,15 +2179,11 @@ var TreeStyleTabUtils = {
|
|||||||
if (this.tabsHash) { // XPath-less implementation
|
if (this.tabsHash) { // XPath-less implementation
|
||||||
let parent = this.getParentTab(aTab);
|
let parent = this.getParentTab(aTab);
|
||||||
if (!aParent || !parent || aParent != parent) {
|
if (!aParent || !parent || aParent != parent) {
|
||||||
parent = aTab;
|
let ancestors = this.getAncestorTabs(aTab);
|
||||||
while (parent && parent != aParent)
|
let index = ancestors.indexOf(aParent);
|
||||||
{
|
if (index < 1)
|
||||||
aTab = parent;
|
|
||||||
parent = this.getParentTab(parent);
|
|
||||||
}
|
|
||||||
if (parent != aParent)
|
|
||||||
return -1;
|
return -1;
|
||||||
aParent = parent;
|
aTab = ancestors[index-1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (aParent) {
|
if (aParent) {
|
||||||
|
@ -1199,7 +1199,7 @@ TreeStyleTabWindow.prototype = {
|
|||||||
this._restoringTabs.forEach(function(aTab) {
|
this._restoringTabs.forEach(function(aTab) {
|
||||||
try {
|
try {
|
||||||
var b = this.getTabBrowserFromChild(aTab);
|
var b = this.getTabBrowserFromChild(aTab);
|
||||||
if (b) b.treeStyleTab.restoreStructure(aTab, true);
|
if (b) b.treeStyleTab.handleRestoredTab(aTab);
|
||||||
}
|
}
|
||||||
catch(e) {
|
catch(e) {
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user