・ツリーのインデントを動的に変更するようにした

・ツリーの開閉に失敗することがあったのを修正

git-svn-id: http://www.cozmixng.org/repos/piro/treestyletab/trunk@1241 599a83e7-65a4-db11-8015-0010dcdd6dc2
This commit is contained in:
piro 2007-10-20 08:22:42 +00:00
parent d9d02b036c
commit c1dbf1696d
2 changed files with 175 additions and 29 deletions

View File

@ -33,8 +33,10 @@ var TreeStyleTabService = {
kACTION_PART : 4, kACTION_PART : 4,
levelMargin : 12, levelMargin : 12,
levelMarginProp : 'margin-left',
positionProp : 'screenY', positionProp : 'screenY',
sizeProp : 'height', 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)
@ -1523,6 +1626,8 @@ 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;
}
};

View File

@ -4,11 +4,15 @@ 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");