・ツリーのインデントを動的に変更するようにした
・ツリーの開閉に失敗することがあったのを修正 git-svn-id: http://www.cozmixng.org/repos/piro/treestyletab/trunk@1241 599a83e7-65a4-db11-8015-0010dcdd6dc2
This commit is contained in:
parent
d9d02b036c
commit
c1dbf1696d
@ -32,9 +32,11 @@ var TreeStyleTabService = {
|
|||||||
kACTION_ATTACH : 2,
|
kACTION_ATTACH : 2,
|
||||||
kACTION_PART : 4,
|
kACTION_PART : 4,
|
||||||
|
|
||||||
levelMargin : 12,
|
levelMargin : 12,
|
||||||
positionProp : 'screenY',
|
levelMarginProp : 'margin-left',
|
||||||
sizeProp : 'height',
|
positionProp : 'screenY',
|
||||||
|
sizeProp : 'height',
|
||||||
|
invertedSizeProp : 'width',
|
||||||
|
|
||||||
NSResolver : {
|
NSResolver : {
|
||||||
lookupNamespaceURI : function(aPrefix)
|
lookupNamespaceURI : function(aPrefix)
|
||||||
@ -62,6 +64,14 @@ var TreeStyleTabService = {
|
|||||||
},
|
},
|
||||||
_SessionStore : null,
|
_SessionStore : null,
|
||||||
|
|
||||||
|
get ObserverService() {
|
||||||
|
if (!this._ObserverService) {
|
||||||
|
this._ObserverService = Components.classes['@mozilla.org/observer-service;1'].getService(Components.interfaces.nsIObserverService);
|
||||||
|
}
|
||||||
|
return this._ObserverService;
|
||||||
|
},
|
||||||
|
_ObserverService : null,
|
||||||
|
|
||||||
/* Utilities */
|
/* Utilities */
|
||||||
|
|
||||||
isEventFiredOnTabIcon : function(aEvent)
|
isEventFiredOnTabIcon : function(aEvent)
|
||||||
@ -212,7 +222,7 @@ var TreeStyleTabService = {
|
|||||||
appcontent.addEventListener('SubBrowserRemoveRequest', this, false);
|
appcontent.addEventListener('SubBrowserRemoveRequest', this, false);
|
||||||
|
|
||||||
this.addPrefListener(this);
|
this.addPrefListener(this);
|
||||||
this.observe(null, 'nsPref:changed', 'extensions.treestyletab.');
|
this.observe(null, 'nsPref:changed', 'extensions.treestyletab.levelMargin');
|
||||||
|
|
||||||
eval('window.nsBrowserAccess.prototype.openURI = '+
|
eval('window.nsBrowserAccess.prototype.openURI = '+
|
||||||
window.nsBrowserAccess.prototype.openURI.toSource().replace(
|
window.nsBrowserAccess.prototype.openURI.toSource().replace(
|
||||||
@ -293,7 +303,7 @@ var TreeStyleTabService = {
|
|||||||
aTabBrowser.mTabContainer.addEventListener('select', this, true);
|
aTabBrowser.mTabContainer.addEventListener('select', this, true);
|
||||||
aTabBrowser.mPanelContainer.addEventListener('click', this, true);
|
aTabBrowser.mPanelContainer.addEventListener('click', this, true);
|
||||||
|
|
||||||
aTabBrowser.setAttribute(this.kVERTICAL, true);
|
aTabBrowser.__treestyletab__levelMargin = -1;
|
||||||
|
|
||||||
eval('aTabBrowser.mTabContainer.selectNewTab = '+
|
eval('aTabBrowser.mTabContainer.selectNewTab = '+
|
||||||
aTabBrowser.mTabContainer.selectNewTab.toSource().replace(
|
aTabBrowser.mTabContainer.selectNewTab.toSource().replace(
|
||||||
@ -572,6 +582,10 @@ catch(e) {
|
|||||||
this.initTab(tabs[i], aTabBrowser);
|
this.initTab(tabs[i], aTabBrowser);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
aTabBrowser.__treestyletab__observer = new TreeStyleTabBrowserObserver(aTabBrowser);
|
||||||
|
|
||||||
|
aTabBrowser.setAttribute(this.kVERTICAL, true);
|
||||||
|
|
||||||
delete addTabMethod;
|
delete addTabMethod;
|
||||||
delete removeTabMethod;
|
delete removeTabMethod;
|
||||||
delete i;
|
delete i;
|
||||||
@ -652,6 +666,9 @@ catch(e) {
|
|||||||
|
|
||||||
destroyTabBrowser : function(aTabBrowser)
|
destroyTabBrowser : function(aTabBrowser)
|
||||||
{
|
{
|
||||||
|
aTabBrowser.__treestyletab__observer.destroy();
|
||||||
|
delete aTabBrowser.__treestyletab__observer;
|
||||||
|
|
||||||
var tabs = aTabBrowser.mTabContainer.childNodes;
|
var tabs = aTabBrowser.mTabContainer.childNodes;
|
||||||
var parent;
|
var parent;
|
||||||
for (var i = 0, maxi = tabs.length; i < maxi; i++)
|
for (var i = 0, maxi = tabs.length; i < maxi; i++)
|
||||||
@ -781,7 +798,7 @@ catch(e) {
|
|||||||
var tab = aEvent.originalTarget;
|
var tab = aEvent.originalTarget;
|
||||||
var b = this.getTabBrowserFromChildren(tab);
|
var b = this.getTabBrowserFromChildren(tab);
|
||||||
|
|
||||||
if (tab.getAttribute(this.kSUBTREE_COLLAPSED)) {
|
if (tab.getAttribute(this.kSUBTREE_COLLAPSED) == 'true') {
|
||||||
var descendant = this.getDescendantTabs(tab);
|
var descendant = this.getDescendantTabs(tab);
|
||||||
for (var i = descendant.length-1; i > -1; i--)
|
for (var i = descendant.length-1; i > -1; i--)
|
||||||
{
|
{
|
||||||
@ -820,6 +837,7 @@ catch(e) {
|
|||||||
processTab(children[i]);
|
processTab(children[i]);
|
||||||
}
|
}
|
||||||
this.updateTabsIndent(children);
|
this.updateTabsIndent(children);
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
if (attach) {
|
if (attach) {
|
||||||
nextFocusedTab = firstChild;
|
nextFocusedTab = firstChild;
|
||||||
}
|
}
|
||||||
@ -847,6 +865,8 @@ catch(e) {
|
|||||||
|
|
||||||
if (nextFocusedTab && b.selectedTab == tab)
|
if (nextFocusedTab && b.selectedTab == tab)
|
||||||
b.selectedTab = nextFocusedTab;
|
b.selectedTab = nextFocusedTab;
|
||||||
|
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
},
|
},
|
||||||
|
|
||||||
onTabMove : function(aEvent)
|
onTabMove : function(aEvent)
|
||||||
@ -896,9 +916,11 @@ catch(e) {
|
|||||||
this.attachTabTo(tab, parent, { dontExpand : true, insertBefore : (before ? this.getTabById(before, b) : null ), dontUpdateIndent : true });
|
this.attachTabTo(tab, parent, { dontExpand : true, insertBefore : (before ? this.getTabById(before, b) : null ), dontUpdateIndent : true });
|
||||||
this.deleteTabValue(tab, this.kPARENT);
|
this.deleteTabValue(tab, this.kPARENT);
|
||||||
this.updateTabsIndent([tab]);
|
this.updateTabsIndent([tab]);
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
}
|
}
|
||||||
else if (children) {
|
else if (children) {
|
||||||
this.updateTabsIndent(tabs);
|
this.updateTabsIndent(tabs);
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isSubTreeCollapsed) {
|
if (isSubTreeCollapsed) {
|
||||||
@ -1021,6 +1043,14 @@ catch(e) {
|
|||||||
|
|
||||||
/* tree */
|
/* tree */
|
||||||
|
|
||||||
|
getRootTabs : function(aTabBrowser)
|
||||||
|
{
|
||||||
|
return this.evaluateXPath(
|
||||||
|
'child::xul:tab[not(@'+this.kNEST+') or @'+this.kNEST+'="0" or @'+this.kNEST+'=""]',
|
||||||
|
aTabBrowser.mTabContainer
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
getParentTab : function(aTab)
|
getParentTab : function(aTab)
|
||||||
{
|
{
|
||||||
var id = aTab.getAttribute(this.kID);
|
var id = aTab.getAttribute(this.kID);
|
||||||
@ -1345,7 +1375,11 @@ catch(e) {
|
|||||||
this.getPref('extensions.treestyletab.autoCollapseExpandSubTreeOnSelect')
|
this.getPref('extensions.treestyletab.autoCollapseExpandSubTreeOnSelect')
|
||||||
) {
|
) {
|
||||||
this.collapseExpandTreesIntelligentlyFor(aChild);
|
this.collapseExpandTreesIntelligentlyFor(aChild);
|
||||||
this.collapseExpandTabSubTree(aParent, false);
|
var p = aParent;
|
||||||
|
do {
|
||||||
|
this.collapseExpandTabSubTree(p, false);
|
||||||
|
}
|
||||||
|
while (p = this.getParentTab(p));
|
||||||
}
|
}
|
||||||
else if (aParent.getAttribute(this.kSUBTREE_COLLAPSED) == 'true') {
|
else if (aParent.getAttribute(this.kSUBTREE_COLLAPSED) == 'true') {
|
||||||
if (this.getPref('extensions.treestyletab.autoExpandSubTreeOnAppendChild')) {
|
if (this.getPref('extensions.treestyletab.autoExpandSubTreeOnAppendChild')) {
|
||||||
@ -1367,7 +1401,10 @@ catch(e) {
|
|||||||
this.collapseExpandTab(aChild, true);
|
this.collapseExpandTab(aChild, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!aInfo.dontUpdateIndent) this.updateTabsIndent([aChild]);
|
if (!aInfo.dontUpdateIndent) {
|
||||||
|
this.updateTabsIndent([aChild]);
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
partTab : function(aChild, aDontUpdateIndent)
|
partTab : function(aChild, aDontUpdateIndent)
|
||||||
@ -1384,7 +1421,10 @@ catch(e) {
|
|||||||
this.setTabValue(parentTab, this.kCHILDREN, children);
|
this.setTabValue(parentTab, this.kCHILDREN, children);
|
||||||
this.updateTabsCount(parentTab);
|
this.updateTabsCount(parentTab);
|
||||||
|
|
||||||
if (!aDontUpdateIndent) this.updateTabsIndent([aChild]);
|
if (!aDontUpdateIndent) {
|
||||||
|
this.updateTabsIndent([aChild]);
|
||||||
|
this.checkTabsIndentOverflow(b);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
updateTabsIndent : function(aTabs, aLevel, aProp)
|
updateTabsIndent : function(aTabs, aLevel, aProp)
|
||||||
@ -1401,18 +1441,76 @@ catch(e) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!aProp)
|
if (!aProp) aProp = this.levelMarginProp;
|
||||||
aProp = this.isTabVertical(aTabs[0]) ? 'margin-left' : 'margin-top' ;
|
|
||||||
|
var b = this.getTabBrowserFromChildren(aTabs[0]);
|
||||||
|
var margin = b.__treestyletab__levelMargin < 0 ? this.levelMargin : b.__treestyletab__levelMargin ;
|
||||||
|
var indent = margin * aLevel;
|
||||||
|
|
||||||
var indent = (this.levelMargin * aLevel)+'px';
|
|
||||||
for (var i = 0, maxi = aTabs.length; i < maxi; i++)
|
for (var i = 0, maxi = aTabs.length; i < maxi; i++)
|
||||||
{
|
{
|
||||||
aTabs[i].setAttribute('style', aTabs[i].getAttribute('style')+';'+aProp+':'+indent+' !important;');
|
aTabs[i].setAttribute('style', aTabs[i].getAttribute('style')+'; margin: 0 !important; '+aProp+':'+indent+'px !important;');
|
||||||
aTabs[i].setAttribute(this.kNEST, aLevel);
|
aTabs[i].setAttribute(this.kNEST, aLevel);
|
||||||
this.updateTabsIndent(this.getChildTabs(aTabs[i]), aLevel+1, aProp);
|
this.updateTabsIndent(this.getChildTabs(aTabs[i]), aLevel+1, aProp);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
updateAllTabsIndent : function(aTabBrowser)
|
||||||
|
{
|
||||||
|
var b = this.getTabBrowserFromChildren(aTabBrowser);
|
||||||
|
this.updateTabsIndent(
|
||||||
|
this.getArrayFromXPathResult(
|
||||||
|
this.getRootTabs(b)
|
||||||
|
),
|
||||||
|
0
|
||||||
|
);
|
||||||
|
// this.checkTabsIndentOverflow(b);
|
||||||
|
},
|
||||||
|
|
||||||
|
checkTabsIndentOverflow : function(aTabBrowser)
|
||||||
|
{
|
||||||
|
if (this.checkTabsIndentOverflowTimer) {
|
||||||
|
window.clearTimeout(this.checkTabsIndentOverflowTimer);
|
||||||
|
this.checkTabsIndentOverflowTimer = null;
|
||||||
|
}
|
||||||
|
this.checkTabsIndentOverflowTimer = window.setTimeout(function(aSelf, aTabBrowser) {
|
||||||
|
aSelf.checkTabsIndentOverflowCallback(aTabBrowser);
|
||||||
|
}, 100, this, aTabBrowser);
|
||||||
|
},
|
||||||
|
checkTabsIndentOverflowTimer : null,
|
||||||
|
checkTabsIndentOverflowCallback : function(aTabBrowser)
|
||||||
|
{
|
||||||
|
var b = aTabBrowser;
|
||||||
|
var tabs = this.getArrayFromXPathResult(this.evaluateXPath(
|
||||||
|
'child::xul:tab[@'+this.kNEST+' and not(@'+this.kNEST+'="0" or @'+this.kNEST+'="")]',
|
||||||
|
b.mTabContainer
|
||||||
|
));
|
||||||
|
if (!tabs.length) return;
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
tabs.sort(function(aA, aB) { return Number(aA.getAttribute(self.kNEST)) - Number(aB.getAttribute(self.kNEST)); });
|
||||||
|
var nest = tabs[tabs.length-1].getAttribute(self.kNEST);
|
||||||
|
if (!nest) return;
|
||||||
|
|
||||||
|
var oldMargin = b.__treestyletab__levelMargin;
|
||||||
|
var indent = (oldMargin < 0 ? this.levelMargin : oldMargin ) * nest;
|
||||||
|
var maxIndent = b.mTabContainer.childNodes[0].boxObject[this.invertedSizeProp] * 0.33;
|
||||||
|
|
||||||
|
var marginUnit = Math.max(Math.floor(maxIndent / nest), 1);
|
||||||
|
if (indent > maxIndent) {
|
||||||
|
b.__treestyletab__levelMargin = marginUnit;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
b.__treestyletab__levelMargin = -1;
|
||||||
|
if ((this.levelMargin * nest) > maxIndent)
|
||||||
|
b.__treestyletab__levelMargin = marginUnit;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (oldMargin != b.__treestyletab__levelMargin) {
|
||||||
|
this.updateAllTabsIndent(b);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
updateTabsCount : function(aTab)
|
updateTabsCount : function(aTab)
|
||||||
{
|
{
|
||||||
var count = document.getAnonymousElementByAttribute(aTab, 'class', this.kCOUNTER);
|
var count = document.getAnonymousElementByAttribute(aTab, 'class', this.kCOUNTER);
|
||||||
@ -1488,7 +1586,10 @@ catch(e) {
|
|||||||
{
|
{
|
||||||
if (!aTab) return;
|
if (!aTab) return;
|
||||||
|
|
||||||
if (aTab.getAttribute(this.kSUBTREE_COLLAPSED) == String(aCollapse)) return;
|
if ((aTab.getAttribute(this.kSUBTREE_COLLAPSED) == 'true') == aCollapse) return;
|
||||||
|
|
||||||
|
var b = this.getTabBrowserFromChildren(aTab);
|
||||||
|
b.__treestyletab__doingCollapseExpand = true;
|
||||||
|
|
||||||
this.setTabValue(aTab, this.kSUBTREE_COLLAPSED, aCollapse);
|
this.setTabValue(aTab, this.kSUBTREE_COLLAPSED, aCollapse);
|
||||||
|
|
||||||
@ -1497,6 +1598,8 @@ catch(e) {
|
|||||||
{
|
{
|
||||||
this.collapseExpandTab(tabs[i], aCollapse);
|
this.collapseExpandTab(tabs[i], aCollapse);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
b.__treestyletab__doingCollapseExpand = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
collapseExpandTab : function(aTab, aCollapse)
|
collapseExpandTab : function(aTab, aCollapse)
|
||||||
@ -1522,7 +1625,9 @@ catch(e) {
|
|||||||
|
|
||||||
collapseExpandTreesIntelligentlyFor : function(aTab)
|
collapseExpandTreesIntelligentlyFor : function(aTab)
|
||||||
{
|
{
|
||||||
var b = this.getTabBrowserFromChildren(aTab);
|
var b = this.getTabBrowserFromChildren(aTab);
|
||||||
|
if (b.__treestyletab__doingCollapseExpand) return;
|
||||||
|
|
||||||
var parent = this.getParentTab(aTab);
|
var parent = this.getParentTab(aTab);
|
||||||
var expandedParentTabs = [
|
var expandedParentTabs = [
|
||||||
aTab.getAttribute(this.kID)
|
aTab.getAttribute(this.kID)
|
||||||
@ -1604,7 +1709,9 @@ catch(e) {
|
|||||||
var value = this.getPref(aPrefName);
|
var value = this.getPref(aPrefName);
|
||||||
switch (aPrefName)
|
switch (aPrefName)
|
||||||
{
|
{
|
||||||
case 'extensions.treestyletab.':
|
case 'extensions.treestyletab.levelMargin':
|
||||||
|
this.levelMargin = value;
|
||||||
|
this.ObserverService.notifyObservers(null, 'TreeStyleTab:levelMarginModified', value);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -1711,3 +1818,38 @@ catch(e) {
|
|||||||
window.addEventListener('load', TreeStyleTabService, false);
|
window.addEventListener('load', TreeStyleTabService, false);
|
||||||
window.addEventListener('unload', TreeStyleTabService, false);
|
window.addEventListener('unload', TreeStyleTabService, false);
|
||||||
|
|
||||||
|
function TreeStyleTabBrowserObserver(aTabBrowser)
|
||||||
|
{
|
||||||
|
this.mTabBrowser = aTabBrowser;
|
||||||
|
TreeStyleTabService.ObserverService.addObserver(this, 'TreeStyleTab:levelMarginModified', false);
|
||||||
|
TreeStyleTabService.addPrefListener(this);
|
||||||
|
}
|
||||||
|
TreeStyleTabBrowserObserver.prototype = {
|
||||||
|
domain : 'extensions.treestyletab',
|
||||||
|
mTabBrowser : null,
|
||||||
|
observe : function(aSubject, aTopic, aData)
|
||||||
|
{
|
||||||
|
switch (aTopic)
|
||||||
|
{
|
||||||
|
case 'TreeStyleTab:levelMarginModified':
|
||||||
|
if (this.mTabBrowser.__treestyletab__levelMargin > -1) {
|
||||||
|
TreeStyleTabService.updateAllTabsIndent(this.mTabBrowser);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'nsPref:changed':
|
||||||
|
var value = TreeStyleTabService.getPref(aData);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
destroy : function()
|
||||||
|
{
|
||||||
|
TreeStyleTabService.ObserverService.removeObserver(this, 'TreeStyleTab:levelMarginModified');
|
||||||
|
TreeStyleTabService.removePrefListener(this);
|
||||||
|
delete this.mTabBrowser;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
@ -1,14 +1,18 @@
|
|||||||
pref("extensions.treestyletab.autoCollapseExpandSubTreeOnSelect", true);
|
pref("extensions.treestyletab.autoCollapseExpandSubTreeOnSelect", true);
|
||||||
pref("extensions.treestyletab.collapseExpandSubTree.dblclick", false);
|
pref("extensions.treestyletab.collapseExpandSubTree.dblclick", false);
|
||||||
pref("extensions.treestyletab.autoExpandSubTreeOnAppendChild", true);
|
pref("extensions.treestyletab.autoExpandSubTreeOnAppendChild", true);
|
||||||
pref("extensions.treestyletab.attachChildrenToGrandParentOnRemoveTab", true);
|
pref("extensions.treestyletab.attachChildrenToGrandParentOnRemoveTab", true);
|
||||||
// 0 = default, 1 = only visible tabs
|
// 0 = default, 1 = only visible tabs
|
||||||
pref("extensions.treestyletab.focusMode", 1);
|
pref("extensions.treestyletab.focusMode", 1);
|
||||||
|
pref("extensions.treestyletab.levelMargin", 16);
|
||||||
|
|
||||||
|
|
||||||
pref("browser.link.open_newwindow.restriction", 0);
|
pref("browser.link.open_newwindow.restriction", 0);
|
||||||
|
|
||||||
|
|
||||||
pref("extensions.multipletab.show.multipletab-selection-item-removeTabSubTree", true);
|
pref("extensions.multipletab.show.multipletab-selection-item-removeTabSubTree", true);
|
||||||
pref("extensions.multipletab.show.context-item-removeTabSubTree", true);
|
pref("extensions.multipletab.show.context-item-removeTabSubTree", true);
|
||||||
|
|
||||||
|
|
||||||
pref("extensions.treestyletab@piro.sakura.ne.jp.name", "chrome://treestyletab/locale/treestyletab.properties");
|
pref("extensions.treestyletab@piro.sakura.ne.jp.name", "chrome://treestyletab/locale/treestyletab.properties");
|
||||||
pref("extensions.treestyletab@piro.sakura.ne.jp.description", "chrome://treestyletab/locale/treestyletab.properties");
|
pref("extensions.treestyletab@piro.sakura.ne.jp.description", "chrome://treestyletab/locale/treestyletab.properties");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user