Guidance panel for new save icon

For new installations, a panel will show for the Z icon with a ">"
button to go to a second panel for the save icon. For existing
installations, a panel will show just for the save icon. In both cases,
the panels are non-auto-hiding but are dismissed if clicked on or when a
new content page is loaded.

There's also a new close button, but it's not enabled currently. Might
be useful for panels with embedded links where an accidental click on
the panel alone shouldn't close the panel (like the Firefox customize
mode wizard).
This commit is contained in:
Dan Stillman 2015-06-24 20:01:01 -04:00
parent a909951106
commit 4868095820
10 changed files with 215 additions and 65 deletions

View file

@ -29,71 +29,130 @@
xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"> xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">
<binding id="guidancepanel"> <binding id="guidancepanel">
<resources>
<stylesheet src="chrome://zotero/skin/bindings/guidancepanel.css"/>
</resources>
<implementation> <implementation>
<!--
@param {Object} [options]
@param {String} [options.text] - Text to use in place of firstRunGuidance.<about>
@param {DOMElement} [options.forEl] - Anchor node
@param {Boolean} [options.force] - Show even if already shown, and don't update
firstRunGuidanceShown.<about> pref
-->
<method name="show"> <method name="show">
<parameter name="forEl"/> <parameter name="options"/>
<parameter name="text"/>
<body> <body>
<![CDATA[ <![CDATA[
Components.utils.import("resource://gre/modules/Services.jsm"); Components.utils.import("resource://gre/modules/Services.jsm");
if(!Zotero.Prefs.get("firstRunGuidance")) return; if(!Zotero.Prefs.get("firstRunGuidance")) return;
var about = this.getAttribute("about"), options = options || {};
pref = "firstRunGuidanceShown."+about, let text = options.text;
shown = false; let useLastText = options.useLastText || false;
try { let forEl = options.forEl || document.getElementById(this.getAttribute("for"));
shown = Zotero.Prefs.get(pref); let force = options.force || false;
} catch(e) {};
if(shown) return; if (!forEl) return;
// Don't show two panels at once
if (Zotero.guidanceBeingShown) {
return;
}
var about = this.getAttribute("about");
var pref = false;
if (about) {
pref = "firstRunGuidanceShown." + about;
let shown = false;
try {
shown = Zotero.Prefs.get(pref);
} catch(e) {};
if (shown && !force) {
return;
}
}
Zotero.guidanceBeingShown = true;
var x = this.getAttribute("x"), var x = this.getAttribute("x"),
y = this.getAttribute("y"), y = this.getAttribute("y"),
position = this.getAttribute("position"), position = this.getAttribute("position"),
panel = document.getAnonymousNodes(this)[0]; panel = document.getAnonymousNodes(this)[0];
if(!forEl) forEl = document.getElementById(this.getAttribute("for")); if (!useLastText) {
if(!text) text = Zotero.getString("firstRunGuidance."+about); if (!text) {
text = text.split("\n"); text = Zotero.getString("firstRunGuidance." + about);
var descriptionNode = panel.lastChild; }
text = text.split("\n");
var descriptionNode = this.id('panel-description');
while (descriptionNode.hasChildNodes()) { while (descriptionNode.hasChildNodes()) {
descriptionNode.removeChild(descriptionNode.firstChild); descriptionNode.removeChild(descriptionNode.firstChild);
}
while(text.length) {
var textLine = text.shift();
descriptionNode.appendChild(document.createTextNode(textLine));
if(text.length) descriptionNode.appendChild(document.createElementNS(
"http://www.w3.org/1999/xhtml", "br"));
}
} }
while(text.length) { this.setAttribute(
var textLine = text.shift(); "onpopuphidden",
descriptionNode.appendChild(document.createTextNode(textLine)); "this.hidden = true; "
if(text.length) descriptionNode.appendChild(document.createElementNS( + "Zotero.guidanceBeingShown = false; "
"http://www.w3.org/1999/xhtml", "br")); + (this.getAttribute("onpopuphidden") || "")
} );
this.setAttribute('onpopuphidden', 'this.hidden = true'); this._initNavButton('back', options.back);
this._initNavButton('forward', options.forward);
var me = this; var self = this;
var f = function() { var f = function() {
if(me.hasAttribute("foregroundonly") && Services.ww.activeWindow != window) return; if (self.hasAttribute("foregroundonly") && Services.ww.activeWindow != window) return;
me.hidden = false;
// Hide panel if content page changes
if (self.getAttribute('hideonpagechange') == "true") {
let popupShownListener = function () {
self.removeEventListener("popupshown", popupShownListener);
let appcontent = document.getElementById('appcontent');
let pageHideListener = function () {
appcontent.removeEventListener("pagehide", pageHideListener);
self.hide();
};
appcontent.addEventListener("pagehide", pageHideListener);
};
self.addEventListener("popupshown", popupShownListener);
}
self.hidden = false;
panel.openPopup(forEl, position ? position : "after_start", panel.openPopup(forEl, position ? position : "after_start",
x ? parseInt(x, 10) : 0, y ? parseInt(y, 10) : 0, false, false, null); x ? parseInt(x, 10) : 0, y ? parseInt(y, 10) : 0, false, false, null);
Zotero.Prefs.set(pref, true); if (pref) {
Zotero.Prefs.set(pref, true);
}
}; };
if(this.hasAttribute("delay")) { if(this.hasAttribute("delay") && !force) {
window.setTimeout(f, this.getAttribute("delay")); window.setTimeout(f, this.getAttribute("delay"));
} else { } else {
f(); f();
} }
if(this.hasAttribute("noautohide")) { if (this.hasAttribute("noautohide")) {
var listener = function() { let listener = function () {
panel.removeEventListener("click", listener);
panel.hidePopup(); panel.hidePopup();
panel.removeEventListener("click", listener, false); };
} panel.addEventListener("click", listener);
panel.addEventListener("click", listener, false);
} }
]]> ]]>
</body> </body>
</method> </method>
<method name="hide"> <method name="hide">
<body> <body>
<![CDATA[ <![CDATA[
@ -101,12 +160,62 @@
]]> ]]>
</body> </body>
</method> </method>
<method name="_initNavButton">
<parameter name="dir"/>
<parameter name="nextID"/>
<body><![CDATA[
if (!nextID) {
nextID = this.getAttribute(dir);
}
if (!nextID) {
return;
}
var nextElem = document.getElementById(nextID);
button = this.id(dir + '-button');
button.hidden = false;
var listener = function (event) {
button.removeEventListener("click", listener);
this.hide();
var data = {
force: true
};
// Point the next panel back to this one
data[dir == 'back' ? 'forward' : 'back'] = this.getAttribute('id');
// When going backwards, don't regenerate text
if (dir == 'back') {
data.useLastText = true;
}
nextElem.show(data);
event.stopPropagation();
}.bind(this);
button.addEventListener("click", listener);
]]></body>
</method>
<method name="id">
<parameter name="id"/>
<body><![CDATA[
return document.getAnonymousNodes(this)[0].getElementsByAttribute('anonid', id)[0];
]]></body>
</method>
</implementation> </implementation>
<content> <content>
<xul:panel orient="horizontal" style="max-width: 400px" type="arrow" align="top" xbl:inherits="noautohide"> <xul:panel type="arrow" align="top" xbl:inherits="noautohide">
<xul:image src="chrome://zotero/skin/zotero-new-z-48px.png" style="margin-right: 10px; width: 48px; height: 48px;"/> <xul:stack>
<xul:description flex="1"></xul:description> <xul:hbox align="center">
<xul:image src="chrome://zotero/skin/zotero-new-z-48px.png" style="margin-right: 10px; width: 48px; height: 48px;"/>
<xul:description anonid="panel-description" flex="1"></xul:description>
</xul:hbox>
<xul:hbox anonid="close-button-box">
<xul:toolbarbutton anonid="close-button" class="close-icon" hidden="true"/>
</xul:hbox>
<xul:hbox anonid="nav-buttons">
<xul:toolbarbutton anonid="back-button" oncommand="hide()" hidden="true"/>
<xul:toolbarbutton anonid="forward-button" oncommand="hide()" hidden="true"/>
</xul:hbox>
</xul:stack>
</xul:panel> </xul:panel>
</content> </content>
</binding> </binding>

View file

@ -1874,10 +1874,14 @@
if(field === 'creator') { if(field === 'creator') {
// Reset creator mode settings here so that flex attribute gets reset // Reset creator mode settings here so that flex attribute gets reset
this.switchCreatorMode(row, (otherFields.fieldMode ? 1 : 0), true); this.switchCreatorMode(row, (otherFields.fieldMode ? 1 : 0), true);
Zotero.debug("HERE");
if(Zotero.ItemTypes.getName(this.item.itemTypeID) === "bookSection") { if(Zotero.ItemTypes.getName(this.item.itemTypeID) === "bookSection") {
Zotero.debug("YES");
var creatorTypeLabels = document.getAnonymousNodes(this)[0].getElementsByClassName("creator-type-label"); var creatorTypeLabels = document.getAnonymousNodes(this)[0].getElementsByClassName("creator-type-label");
document.getElementById("zotero-author-guidance").show(creatorTypeLabels[creatorTypeLabels.length-1]); Zotero.debug(creatorTypeLabels[creatorTypeLabels.length-1] + "");
document.getElementById("zotero-author-guidance").show({
forEl: creatorTypeLabels[creatorTypeLabels.length-1]
});
} }
} }

View file

@ -441,14 +441,6 @@ var Zotero_Browser = new function() {
button.tooltipText = tooltiptext; button.tooltipText = tooltiptext;
if (state == tab.CAPTURE_STATE_TRANSLATABLE) { if (state == tab.CAPTURE_STATE_TRANSLATABLE) {
button.classList.add('translate'); button.classList.add('translate');
// Show guidance panel if necessary
if (inToolbar) {
button.addEventListener("load", function() {
document.getElementById("zotero-status-image-guidance").show();
});
}
// TODO: Different guidance for web pages?
} }
else { else {
button.classList.remove('translate'); button.classList.remove('translate');

View file

@ -61,9 +61,10 @@ CustomizableUI.addListener({
var shortcut = Zotero.getString( var shortcut = Zotero.getString(
Zotero.isMac ? "general.keys.cmdShift" : "general.keys.ctrlShift" Zotero.isMac ? "general.keys.cmdShift" : "general.keys.ctrlShift"
) + Zotero.Prefs.get("keys.openZotero"); ) + Zotero.Prefs.get("keys.openZotero");
document.getElementById("zotero-toolbar-button-guidance").show( document.getElementById("zotero-main-button-guidance").show({
null, Zotero.getString(property, shortcut) text: Zotero.getString(property, shortcut)
); });
document.getElementById("zotero-save-button-guidance").show();
} }
else if (id == getSingleID('save')) { else if (id == getSingleID('save')) {
Zotero_Browser.updateStatus(); Zotero_Browser.updateStatus();

View file

@ -50,9 +50,12 @@
<stack id="zotero-pane-stack" persist="savedHeight" savedHeight="300" hidden="true"/> <stack id="zotero-pane-stack" persist="savedHeight" savedHeight="300" hidden="true"/>
<zoteroguidancepanel id="zotero-toolbar-button-guidance" about="toolbarButton" for="zotero-toolbar-main-button" <zoteroguidancepanel id="zotero-main-button-guidance" about="toolbarButton" for="zotero-toolbar-main-button"
position="bottomcenter topleft" delay="2000" foregroundonly="true"/> position="bottomcenter topleft" delay="2000" foregroundonly="true" noautohide="true"
<zoteroguidancepanel id="zotero-status-image-guidance" about="saveIcon" for="zotero-toolbar-save-button" x="17"/> hideonpagechange="true" forward="zotero-save-button-guidance"/>
<zoteroguidancepanel id="zotero-save-button-guidance" about="saveButton" for="zotero-toolbar-save-button"
position="bottomcenter topleft" x="-8" delay="2000" foregroundonly="true" noautohide="true"
hideonpagechange="true"/>
<!-- Annotation Toolbar --> <!-- Annotation Toolbar -->
<toolbar id="zotero-annotate-tb" crop="end" insertbefore="content" hidden="true"> <toolbar id="zotero-annotate-tb" crop="end" insertbefore="content" hidden="true">

View file

@ -999,12 +999,12 @@ connector.error.title = Zotero Connector Error
connector.standaloneOpen = Your database cannot be accessed because Zotero Standalone is currently open. Please view your items in Zotero Standalone. connector.standaloneOpen = Your database cannot be accessed because Zotero Standalone is currently open. Please view your items in Zotero Standalone.
connector.loadInProgress = Zotero Standalone was launched but is not accessible. If you experienced an error opening Zotero Standalone, restart Firefox. connector.loadInProgress = Zotero Standalone was launched but is not accessible. If you experienced an error opening Zotero Standalone, restart Firefox.
firstRunGuidance.saveIcon = Zotero has found a reference on this page. Click this icon in the address bar to save the reference to your Zotero library.
firstRunGuidance.authorMenu = Zotero lets you specify editors and translators, too. You can turn an author into an editor or translator by selecting from this menu. firstRunGuidance.authorMenu = Zotero lets you specify editors and translators, too. You can turn an author into an editor or translator by selecting from this menu.
firstRunGuidance.quickFormat = Type a title or author to search for a reference.\n\nAfter you've made your selection, click the bubble or press Ctrl-\u2193 to add page numbers, prefixes, or suffixes. You can also include a page number along with your search terms to add it directly.\n\nYou can edit citations directly in the word processor document. firstRunGuidance.quickFormat = Type a title or author to search for a reference.\n\nAfter you've made your selection, click the bubble or press Ctrl-\u2193 to add page numbers, prefixes, or suffixes. You can also include a page number along with your search terms to add it directly.\n\nYou can edit citations directly in the word processor document.
firstRunGuidance.quickFormatMac = Type a title or author to search for a reference.\n\nAfter you've made your selection, click the bubble or press Cmd-\u2193 to add page numbers, prefixes, or suffixes. You can also include a page number along with your search terms to add it directly.\n\nYou can edit citations directly in the word processor document. firstRunGuidance.quickFormatMac = Type a title or author to search for a reference.\n\nAfter you've made your selection, click the bubble or press Cmd-\u2193 to add page numbers, prefixes, or suffixes. You can also include a page number along with your search terms to add it directly.\n\nYou can edit citations directly in the word processor document.
firstRunGuidance.toolbarButton.new = Click here to open Zotero, or use the %S keyboard shortcut. firstRunGuidance.toolbarButton.new = Click here to open Zotero, or use the %S keyboard shortcut.
firstRunGuidance.toolbarButton.upgrade = The Zotero icon can now be found in the Firefox toolbar. Click the icon to open Zotero, or use the %S keyboard shortcut. firstRunGuidance.toolbarButton.upgrade = The Zotero icon can now be found in the Firefox toolbar. Click the icon to open Zotero, or use the %S keyboard shortcut.
firstRunGuidance.saveButton = Click this button to save any web page to your Zotero library. On some pages, Zotero will be able to save full details, including author and date.
styles.bibliography = Bibliography styles.bibliography = Bibliography
styles.editor.save = Save Citation Style styles.editor.save = Save Citation Style

View file

@ -0,0 +1,51 @@
stack {
max-width: 400px;
margin: 0;
padding: 0;
}
*[anonid=nav-buttons] {
-moz-box-align: end;
-moz-box-pack: end;
}
*[anonid=nav-buttons] > toolbarbutton {
width: 22px;
height: 22px;
border: 1px solid lightgray;
border-radius: 3px;
background-position: 5px 5px;
background-size: 10px;
background-repeat: no-repeat;
margin: 0;
margin-bottom: -7px;
}
*[anonid=nav-buttons] > toolbarbutton:hover {
border-color: var(--toolbarbutton-hover-bordercolor);
box-shadow: var(--toolbarbutton-hover-boxshadow);
}
*[anonid=nav-buttons] > toolbarbutton:active:hover {
border-color: var(--toolbarbutton-active-bordercolor);
box-shadow: var(--toolbarbutton-active-boxshadow);
transition-duration: 10ms;
}
*[anonid=back-button] {
background-image: url("chrome://zotero/skin/chevron-left_808080_32.png");
}
*[anonid=forward-button] {
margin-right: -16px;
background-image: url("chrome://zotero/skin/chevron-right_808080_32.png");
}
*[anonid=close-button-box] {
-moz-box-align: start;
-moz-box-pack: end;
}
*[anonid=close-button] {
margin: -16px -16px;
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 347 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 353 B

View file

@ -1,13 +1,3 @@
#zotero-status-image {
width: 16px;
height: 16px;
margin-right: 3px;
}
#zotero-status-image:not(.translate):not(:hover) {
filter: grayscale(100%);
}
#zotero-pane #zotero-pane
{ {
min-height: 32px; /* must match value in ZoteroPane.updateTagSelectorSize() */ min-height: 32px; /* must match value in ZoteroPane.updateTagSelectorSize() */