Fix 'title' attribute on React Button component

E.g., Actions menu in tag selector

Neither 'title' (HTML) nor 'tooltiptext' (XUL) works on HTML elements in
XUL documents, so add a mechanism to fake tooltip behavior and use it in
our Button component. This can be triggered by other React components if
necessary.

This is (mostly) mirroring Firefox tooltip behavior, which is a bit less
sophisicated than macOS behavior. If we end up using this for
everything, we can improve the behavior (e.g., start the delay when
entering the element, not when the mouse stops).

Fixes #1947
This commit is contained in:
Dan Stillman 2021-02-01 04:10:38 -05:00
parent cf132fdd29
commit ba9c7ae739
3 changed files with 97 additions and 1 deletions

View file

@ -99,8 +99,23 @@ class Button extends PureComponent {
}
if (!this.props.isDisabled) {
attr.onMouseDown = this.handleMouseDown
attr.onMouseDown = (event) => {
// Hide tooltip on mousedown
if (this.title) {
window.Zotero_Tooltip.stop();
}
return this.handleMouseDown(event);
};
attr.onClick = this.handleClick
// Fake tooltip behavior as long as 'title' doesn't work for HTML-in-XUL elements
if (this.title) {
attr.onMouseOver = () => {
window.Zotero_Tooltip.start(this.title);
};
attr.onMouseOut = () => {
window.Zotero_Tooltip.stop();
};
}
}
return attr

View file

@ -0,0 +1,78 @@
/*
***** BEGIN LICENSE BLOCK *****
Copyright © 2020 Corporation for Digital Scholarship
Vienna, Virginia, USA
https://www.zotero.org
This file is part of Zotero.
Zotero is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Zotero is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
***** END LICENSE BLOCK *****
*/
/**
* Fake tooltip implementation for HTML-in-XUL elements where neither 'title' nor 'tooltiptext' works
*/
// eslint-disable-next-line camelcase,no-unused-vars
var Zotero_Tooltip = new function () {
// On macOS, the tooltip appears even if the mouse keeps moving over the element, but Mozilla
// shows it only once the mouse stops, so follow that as long as there are XUL elements.
const MOUSE_STOP_DELAY = 500;
var text;
var timeoutID;
var x;
var y;
/**
* Start tracking the mouse and show a tooltip after it stops
*/
this.start = function (tooltipText) {
window.addEventListener('mousemove', handleMouseMove);
text = tooltipText;
};
/**
* Stop tracking the mouse and hide the tooltip if it's showing
*/
this.stop = function () {
window.removeEventListener('mousemove', handleMouseMove);
clearTimeout(timeoutID);
// Mozilla hides the tooltip as soon as the mouse leaves the element, which is also different
// from macOS behavior
hidePopup();
};
function handleMouseMove(event) {
if (timeoutID) {
clearTimeout(timeoutID);
}
x = event.screenX;
y = event.screenY;
timeoutID = setTimeout(handleMouseStop, MOUSE_STOP_DELAY);
}
function handleMouseStop() {
var tooltipElem = document.getElementById('fake-tooltip');
tooltipElem.setAttribute('label', text);
tooltipElem.openPopupAtScreen(x, y, false, null);
}
function hidePopup() {
var tooltipElem = document.getElementById('fake-tooltip');
tooltipElem.hidePopup();
}
};

View file

@ -49,6 +49,7 @@
<script src="browser.js" type="application/javascript"/>
<script src="lookup.js"/>
<script src="locateMenu.js" type="application/javascript"/>
<script src="tooltip.js"/>
<commandset id="mainCommandSet">
<command id="cmd_zotero_reportErrors" oncommand="ZoteroPane_Local.reportErrors();"/>
@ -289,6 +290,8 @@
<menuitem class="menuitem-iconic zotero-menuitem-rename-from-parent" oncommand="ZoteroPane_Local.renameSelectedAttachmentsFromParents()"/>
<menuitem class="menuitem-iconic zotero-menuitem-reindex" oncommand="ZoteroPane_Local.reindexItem();"/>
</menupopup>
<tooltip id="fake-tooltip"/>
</popupset>
<hbox id="zotero-trees" flex="1">