zotero/chrome/content/zotero/components/tabBar.jsx
Martynas Bagdonas 01645c5e51
Pin library tab (#2589)
Fixes #2575
2022-05-03 01:21:59 -04:00

339 lines
10 KiB
JavaScript

/*
***** 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 *****
*/
'use strict';
import React, { forwardRef, useState, useRef, useImperativeHandle, useEffect, useLayoutEffect } from 'react';
import cx from 'classnames';
const { IconXmark, IconArrowLeft, IconArrowRight } = require('./icons');
const SCROLL_ARROW_SCROLL_BY = 222;
const TabBar = forwardRef(function (props, ref) {
const [tabs, setTabs] = useState([]);
const [dragging, setDragging] = useState(false);
const [dragMouseX, setDragMouseX] = useState(0);
const dragIDRef = useRef(null);
const dragGrabbedDeltaXRef = useRef();
const tabsInnerContainerRef = useRef();
const tabsRef = useRef();
const startArrowRef = useRef();
const endArrowRef = useRef();
// Used to throttle mouse movement
const mouseMoveWaitUntil = useRef(0);
useImperativeHandle(ref, () => ({ setTabs }));
useEffect(() => {
let handleResize = () => updateScrollArrows();
window.addEventListener('resize', handleResize);
return () => {
window.removeEventListener('resize', handleResize);
};
}, []);
useLayoutEffect(() => updateScrollArrows());
// Use offsetLeft and offsetWidth to calculate and translate tab X position
useLayoutEffect(() => {
if (!dragIDRef.current) return;
let tab = Array.from(tabsRef.current.children).find(x => x.dataset.id === dragIDRef.current);
if (tab) {
// While the actual tab node retains its space between other tabs,
// we use CSS translation to move it to the left/right side to
// position it under the mouse
let x = dragMouseX - tab.offsetLeft - dragGrabbedDeltaXRef.current;
let firstTab = tabsRef.current.firstChild;
let lastTab = tabsRef.current.lastChild;
// Don't allow to move tab beyond the second and the last tab
if (Zotero.rtl) {
if (tab.offsetLeft + x < lastTab.offsetLeft
|| tab.offsetLeft + tab.offsetWidth + x > firstTab.offsetLeft) {
x = 0;
}
}
else if (tab.offsetLeft + x > lastTab.offsetLeft
|| tab.offsetLeft + x < firstTab.offsetLeft + firstTab.offsetWidth) {
x = 0;
}
tab.style.transform = dragging ? `translateX(${x}px)` : 'unset';
}
});
function updateScrollArrows() {
let scrollable = tabsRef.current.scrollWidth !== tabsRef.current.clientWidth;
if (scrollable) {
tabsInnerContainerRef.current.classList.add('scrollable');
if (tabsRef.current.scrollLeft !== 0) {
startArrowRef.current.classList.add('active');
}
else {
startArrowRef.current.classList.remove('active');
}
if (tabsRef.current.scrollWidth - tabsRef.current.clientWidth !== Math.abs(tabsRef.current.scrollLeft)) {
endArrowRef.current.classList.add('active');
}
else {
endArrowRef.current.classList.remove('active');
}
}
else {
tabsInnerContainerRef.current.classList.remove('scrollable');
}
}
function handleTabMouseDown(event, id) {
if (event.button === 2) {
let { screenX, screenY } = event;
// Popup gets immediately closed without this
setTimeout(() => {
props.onContextMenu(screenX, screenY, id);
}, 0);
return;
}
if (event.target.closest('.tab-close')) {
return;
}
props.onTabSelect(id);
event.stopPropagation();
}
function handleTabClick(event, id) {
if (event.button === 1) {
props.onTabClose(id);
}
}
function handleDragStart(event, id, index) {
// Library tab is not draggable
if (index === 0) {
return;
}
event.dataTransfer.effectAllowed = 'move';
// We don't want the generated image from the target element,
// therefore setting an empty image
let img = document.createElement('img');
img.src = '';
event.dataTransfer.setDragImage(img, 0, 0);
// Some data needs to be set, although this is not used anywhere
event.dataTransfer.setData('zotero/tab', id);
// Store the relative mouse to tab X position where the tab was grabbed
dragGrabbedDeltaXRef.current = event.clientX - event.target.offsetLeft;
// Enable dragging
setDragging(true);
// Store the current tab id
dragIDRef.current = id;
}
function handleDragEnd() {
setDragging(false);
}
function handleTabBarDragOver(event) {
event.preventDefault();
event.dataTransfer.dropEffect = 'move';
// Throttle
if (!dragIDRef.current || mouseMoveWaitUntil.current > Date.now()) {
return;
}
setDragMouseX(event.clientX);
// Get the current tab DOM node
let tabIndex = Array.from(tabsRef.current.children).findIndex(x => x.dataset.id === dragIDRef.current);
let tab = tabsRef.current.children[tabIndex];
// Calculate the center points of each tab
let points = Array.from(tabsRef.current.children).map((child) => {
return child.offsetLeft + child.offsetWidth / 2;
});
// Calculate where the new tab left and right (x1, x2) side points should
// be relative to the current mouse position, and take into account
// the initial relative mouse to tab position where the tab was grabbed
let x1 = event.clientX - dragGrabbedDeltaXRef.current;
let x2 = event.clientX - dragGrabbedDeltaXRef.current + tab.offsetWidth;
let index = null;
// Try to determine if the new tab left or right side is crossing
// the middle point of the previous or the next tab, and use its index if so
for (let i = 0; i < points.length - 1; i++) {
if (i === tabIndex || i + 1 === tabIndex) {
continue;
}
let p1 = points[i];
let p2 = points[i + 1];
if (
Zotero.rtl && (x2 < p1 && x2 > p2 || x1 < p1 && x1 > p2)
|| !Zotero.rtl && (x2 > p1 && x2 < p2 || x1 > p1 && x1 < p2)
) {
index = i + 1;
break;
}
}
// If the new tab position doesn't fit between the central points
// of other tabs, check if it's moved beyond the last tab
if (index === null) {
let p = points[points.length - 1];
if (Zotero.rtl && x1 < p || !Zotero.rtl && x2 > p) {
index = points.length;
}
}
if (index !== null) {
props.onTabMove(dragIDRef.current, index);
}
mouseMoveWaitUntil.current = Date.now() + 20;
}
function handleTabClose(event, id) {
props.onTabClose(id);
event.stopPropagation();
}
function handleTabMouseMove(title) {
// Fix `title` not working for HTML-in-XUL. Using `mousemove` ensures we restart the tooltip
// after just a small movement even when the active tab has changed under the cursor, which
// matches behavior in Firefox.
window.Zotero_Tooltip.start(title);
}
function handleTabBarMouseOut() {
// Hide any possibly open `title` tooltips when mousing out of any tab or the tab bar as a
// whole. `mouseout` bubbles up from element you moved out of, so it covers both cases.
window.Zotero_Tooltip.stop();
}
function handleWheel(event) {
// Normalize wheel speed
let x = event.deltaX;
if (x && event.deltaMode) {
if (event.deltaMode === 1) {
x *= 20;
}
else {
x *= 400;
}
}
window.requestAnimationFrame(() => {
tabsRef.current.scrollLeft += x;
});
}
function handleClickScrollStart() {
tabsRef.current.scrollTo({
left: tabsRef.current.scrollLeft - (SCROLL_ARROW_SCROLL_BY * (Zotero.rtl ? -1 : 1)),
behavior: 'smooth'
});
}
function handleClickScrollEnd() {
tabsRef.current.scrollTo({
left: tabsRef.current.scrollLeft + (SCROLL_ARROW_SCROLL_BY * (Zotero.rtl ? -1 : 1)),
behavior: 'smooth'
});
}
// Prevent maximizing/minimizing window
function handleScrollArrowDoubleClick(event) {
event.preventDefault();
}
function renderTab({ id, title, selected, iconBackgroundImage }, index) {
return (
<div
key={id}
data-id={id}
className={cx('tab', { selected, dragging: dragging && id === dragIDRef.current })}
draggable={true}
onMouseMove={() => handleTabMouseMove(title)}
onMouseDown={(event) => handleTabMouseDown(event, id)}
onClick={(event) => handleTabClick(event, id)}
onAuxClick={(event) => handleTabClick(event, id)}
onDragStart={(event) => handleDragStart(event, id, index)}
onDragEnd={handleDragEnd}
>
<div className="tab-name">{iconBackgroundImage &&
<span className="icon-bg" style={{ backgroundImage: iconBackgroundImage }}/>}{title}</div>
<div
className="tab-close"
onClick={(event) => handleTabClose(event, id)}
>
<IconXmark/>
</div>
</div>
);
}
return (
<div>
<div
ref={tabsInnerContainerRef}
className="tab-bar-inner-container"
onWheel={handleWheel}
>
<div className="pinned-tabs">
<div className="tabs">
{tabs.length ? renderTab(tabs[0], 0) : null}
</div>
</div>
<div
ref={startArrowRef}
className="scroll-start-arrow"
style={{ transform: Zotero.rtl ? 'scaleX(-1)' : undefined }}
onClick={handleClickScrollStart}
onDoubleClick={handleScrollArrowDoubleClick}
><IconArrowLeft/></div>
<div className="tabs-wrapper">
<div
ref={tabsRef}
className="tabs"
onDragOver={handleTabBarDragOver}
onMouseOut={handleTabBarMouseOut}
onScroll={updateScrollArrows}
>
{tabs.map((tab, index) => renderTab(tab, index))}
</div>
</div>
<div
ref={endArrowRef}
className="scroll-end-arrow"
style={{ transform: Zotero.rtl ? 'scaleX(-1)' : undefined }}
onClick={handleClickScrollEnd}
onDoubleClick={handleScrollArrowDoubleClick}
><IconArrowRight/></div>
</div>
</div>
);
});
export default TabBar;