mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-04 18:59:08 +01:00
save work
This commit is contained in:
parent
76ec792fe5
commit
b1de0e5447
@ -4,7 +4,7 @@
|
|||||||
import { Button } from "@/element/button";
|
import { Button } from "@/element/button";
|
||||||
import { ContextMenuModel } from "@/store/contextmenu";
|
import { ContextMenuModel } from "@/store/contextmenu";
|
||||||
import { clsx } from "clsx";
|
import { clsx } from "clsx";
|
||||||
import { useAtomValue } from "jotai";
|
import { atom, useAtom, useAtomValue } from "jotai";
|
||||||
import { forwardRef, memo, useCallback, useEffect, useImperativeHandle, useRef, useState } from "react";
|
import { forwardRef, memo, useCallback, useEffect, useImperativeHandle, useRef, useState } from "react";
|
||||||
|
|
||||||
import { atoms, globalStore, refocusNode } from "@/app/store/global";
|
import { atoms, globalStore, refocusNode } from "@/app/store/global";
|
||||||
@ -12,8 +12,11 @@ import { RpcApi } from "@/app/store/wshclientapi";
|
|||||||
import { TabRpcClient } from "@/app/store/wshrpcutil";
|
import { TabRpcClient } from "@/app/store/wshrpcutil";
|
||||||
import { ObjectService } from "../store/services";
|
import { ObjectService } from "../store/services";
|
||||||
import { makeORef, useWaveObjectValue } from "../store/wos";
|
import { makeORef, useWaveObjectValue } from "../store/wos";
|
||||||
|
|
||||||
import "./tab.scss";
|
import "./tab.scss";
|
||||||
|
|
||||||
|
const adjacentTabsAtom = atom<Set<string>>(new Set<string>());
|
||||||
|
|
||||||
interface TabProps {
|
interface TabProps {
|
||||||
id: string;
|
id: string;
|
||||||
isActive: boolean;
|
isActive: boolean;
|
||||||
@ -30,8 +33,8 @@ interface TabProps {
|
|||||||
onMouseDown: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
onMouseDown: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
||||||
onLoaded: () => void;
|
onLoaded: () => void;
|
||||||
onPinChange: () => void;
|
onPinChange: () => void;
|
||||||
onMouseEnter: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
// onMouseEnter: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
||||||
onMouseLeave: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
// onMouseLeave: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Tab = memo(
|
const Tab = memo(
|
||||||
@ -52,8 +55,8 @@ const Tab = memo(
|
|||||||
onClick,
|
onClick,
|
||||||
onClose,
|
onClose,
|
||||||
onMouseDown,
|
onMouseDown,
|
||||||
onMouseEnter,
|
// onMouseEnter,
|
||||||
onMouseLeave,
|
// onMouseLeave,
|
||||||
onPinChange,
|
onPinChange,
|
||||||
},
|
},
|
||||||
ref
|
ref
|
||||||
@ -66,8 +69,11 @@ const Tab = memo(
|
|||||||
const editableTimeoutRef = useRef<NodeJS.Timeout>();
|
const editableTimeoutRef = useRef<NodeJS.Timeout>();
|
||||||
const loadedRef = useRef(false);
|
const loadedRef = useRef(false);
|
||||||
const tabRef = useRef<HTMLDivElement>(null);
|
const tabRef = useRef<HTMLDivElement>(null);
|
||||||
|
const adjacentTabsRef = useRef<Set<string>>(new Set());
|
||||||
|
|
||||||
const tabIndicesMoved = useAtomValue<number[]>(atoms.tabIndicesMoved);
|
const tabIndicesMoved = useAtomValue<number[]>(atoms.tabIndicesMoved);
|
||||||
|
const tabs = document.querySelectorAll(".tab");
|
||||||
|
const [adjacentTabs, setAdjacentTabs] = useAtom(adjacentTabsAtom);
|
||||||
|
|
||||||
useImperativeHandle(ref, () => tabRef.current as HTMLDivElement);
|
useImperativeHandle(ref, () => tabRef.current as HTMLDivElement);
|
||||||
|
|
||||||
@ -202,34 +208,113 @@ const Tab = memo(
|
|||||||
[onPinChange, handleRenameTab, id, onClose, isPinned]
|
[onPinChange, handleRenameTab, id, onClose, isPinned]
|
||||||
);
|
);
|
||||||
|
|
||||||
const showSeparator = useCallback(
|
if (isDragging) {
|
||||||
(id) => {
|
console.log("isDragging", isDragging);
|
||||||
const idx = tabIds.indexOf(id);
|
console.log("dragging tab idx>>>>>>", id);
|
||||||
const found = tabIndicesMoved.find((i, ii) => ii !== 0 && i === idx);
|
const draggingTabIdx = tabIds.indexOf(id);
|
||||||
console.log("id**********", id);
|
|
||||||
if (found) {
|
// Add the dragging tab and its right adjacent tab to the Set
|
||||||
console.log("found=====", tabIds[found]);
|
adjacentTabsRef.current.add(id);
|
||||||
console.log("tabIndicesMoved", tabIndicesMoved);
|
if (draggingTabIdx + 1 < tabIds.length) {
|
||||||
return false;
|
adjacentTabsRef.current.add(tabIds[draggingTabIdx + 1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isActive) {
|
||||||
|
const activeTabIdx = tabIds.indexOf(id);
|
||||||
|
|
||||||
|
// Add the active tab and its right adjacent tab to the Set
|
||||||
|
adjacentTabsRef.current.add(id);
|
||||||
|
if (activeTabIdx + 1 < tabIds.length) {
|
||||||
|
adjacentTabsRef.current.add(tabIds[activeTabIdx + 1]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return true;
|
|
||||||
},
|
|
||||||
[isFirst, tabIndicesMoved]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isActive) return;
|
console.log("triggered!!!!");
|
||||||
|
if ((isDragging || isActive) && tabIndicesMoved.length) {
|
||||||
|
// Find the index of the current tab ID
|
||||||
|
const currentIndex = tabIds.indexOf(id);
|
||||||
|
|
||||||
const idx = tabIds.indexOf(id);
|
console.log("tabIds", tabIds);
|
||||||
const targetIds = [tabIds[idx], tabIds[idx + 1]];
|
console.log("id", id);
|
||||||
|
|
||||||
tabRefs.current.forEach((ref) => {
|
// Get the right adjacent ID
|
||||||
const separator = ref.current.querySelector(".separator") as HTMLElement;
|
const rightAdjacentId = tabIds[currentIndex + 1];
|
||||||
if (!separator) return;
|
// Get the left adjacent ID
|
||||||
|
const leftAdjacentId = tabIds[currentIndex - 1];
|
||||||
|
|
||||||
separator.style.opacity = targetIds.includes(ref.current.dataset.tabId) ? "0" : "1";
|
// console.log("rightAdjacentId", rightAdjacentId);
|
||||||
});
|
|
||||||
}, [id, isActive, tabIds, tabIndicesMoved]);
|
// Set the opacity of the separator for the current tab
|
||||||
|
if (currentIndex !== -1) {
|
||||||
|
const currentTabElement = document.querySelector(`[data-tab-id="${id}"]`) as HTMLElement;
|
||||||
|
if (currentTabElement) {
|
||||||
|
const separator = currentTabElement.querySelector(".separator") as HTMLElement;
|
||||||
|
if (separator) {
|
||||||
|
console.log("1");
|
||||||
|
separator.style.opacity = "0"; // Always hide the separator of the current tab
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the opacity of the separator for the right adjacent tab
|
||||||
|
if (rightAdjacentId) {
|
||||||
|
const rightAdjacentTabElement = document.querySelector(
|
||||||
|
`[data-tab-id="${rightAdjacentId}"]`
|
||||||
|
) as HTMLElement;
|
||||||
|
if (rightAdjacentTabElement) {
|
||||||
|
const separator = rightAdjacentTabElement.querySelector(".separator") as HTMLElement;
|
||||||
|
if (separator) {
|
||||||
|
console.log("2");
|
||||||
|
separator.style.opacity = "0"; // Hide the separator of the right adjacent tab
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup function to reset opacity
|
||||||
|
return () => {
|
||||||
|
if (!isActive && currentIndex !== -1) {
|
||||||
|
const currentTabElement = document.querySelector(`[data-tab-id="${id}"]`) as HTMLElement;
|
||||||
|
|
||||||
|
// To check if leftAdjacentId is the active tab
|
||||||
|
const leftAdjacentElement = document.querySelector(
|
||||||
|
`[data-tab-id="${leftAdjacentId}"]`
|
||||||
|
) as HTMLElement;
|
||||||
|
if (
|
||||||
|
currentTabElement &&
|
||||||
|
leftAdjacentElement &&
|
||||||
|
!leftAdjacentElement.classList.contains("active")
|
||||||
|
) {
|
||||||
|
console.log(
|
||||||
|
"currentTabElement>>>>>>",
|
||||||
|
currentTabElement,
|
||||||
|
currentTabElement &&
|
||||||
|
leftAdjacentElement &&
|
||||||
|
!leftAdjacentElement.classList.contains("active")
|
||||||
|
);
|
||||||
|
const separator = currentTabElement.querySelector(".separator") as HTMLElement;
|
||||||
|
if (separator) {
|
||||||
|
separator.style.opacity = "1"; // Reset opacity for the current tab only if not active
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rightAdjacentId) {
|
||||||
|
const rightAdjacentTabElement = document.querySelector(
|
||||||
|
`[data-tab-id="${rightAdjacentId}"]`
|
||||||
|
) as HTMLElement;
|
||||||
|
console.log("rightAdjacentId!!!!!", rightAdjacentId);
|
||||||
|
if (rightAdjacentTabElement) {
|
||||||
|
const separator = rightAdjacentTabElement.querySelector(".separator") as HTMLElement;
|
||||||
|
if (separator) {
|
||||||
|
separator.style.opacity = "1"; // Reset opacity for the right adjacent tab
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [id, tabIds, isFirst, isActive, tabIndicesMoved]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -243,11 +328,11 @@ const Tab = memo(
|
|||||||
onMouseDown={onMouseDown}
|
onMouseDown={onMouseDown}
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
onContextMenu={handleContextMenu}
|
onContextMenu={handleContextMenu}
|
||||||
onMouseEnter={onMouseEnter}
|
// onMouseEnter={onMouseEnter}
|
||||||
onMouseLeave={onMouseLeave}
|
// onMouseLeave={onMouseLeave}
|
||||||
data-tab-id={id}
|
data-tab-id={id}
|
||||||
>
|
>
|
||||||
{showSeparator(id) && <div className="separator"></div>}
|
<div className="separator"></div>
|
||||||
<div className="tab-inner">
|
<div className="tab-inner">
|
||||||
<div
|
<div
|
||||||
ref={editableRef}
|
ref={editableRef}
|
||||||
|
@ -428,9 +428,9 @@ const TabBar = memo(({ workspace }: TabBarProps) => {
|
|||||||
|
|
||||||
// Track indices that have been moved
|
// Track indices that have been moved
|
||||||
if (getDragDirection(currentX) === "+") {
|
if (getDragDirection(currentX) === "+") {
|
||||||
setTabIndicesMoved([tabIndex, newTabIndex, newTabIndex + 1]);
|
setTabIndicesMoved([tabIds[newTabIndex], tabIds[newTabIndex + 1]].filter(Boolean));
|
||||||
} else if (getDragDirection(currentX) === "-") {
|
} else if (getDragDirection(currentX) === "-") {
|
||||||
setTabIndicesMoved([newTabIndex - 1, newTabIndex, newTabIndex + 1]);
|
setTabIndicesMoved([tabIds[newTabIndex - 1], tabIds[newTabIndex]].filter(Boolean));
|
||||||
}
|
}
|
||||||
|
|
||||||
tabIds.splice(newTabIndex, 0, tabId);
|
tabIds.splice(newTabIndex, 0, tabId);
|
||||||
@ -517,6 +517,7 @@ const TabBar = memo(({ workspace }: TabBarProps) => {
|
|||||||
setDraggingTab(null);
|
setDraggingTab(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setTabIndicesMoved([]);
|
||||||
document.removeEventListener("mouseup", handleMouseUp);
|
document.removeEventListener("mouseup", handleMouseUp);
|
||||||
document.removeEventListener("mousemove", handleMouseMove);
|
document.removeEventListener("mousemove", handleMouseMove);
|
||||||
draggingRemovedRef.current = false;
|
draggingRemovedRef.current = false;
|
||||||
@ -666,8 +667,8 @@ const TabBar = memo(({ workspace }: TabBarProps) => {
|
|||||||
tabWidth={tabWidthRef.current}
|
tabWidth={tabWidthRef.current}
|
||||||
isNew={tabId === newTabId}
|
isNew={tabId === newTabId}
|
||||||
tabIds={tabIds}
|
tabIds={tabIds}
|
||||||
onMouseEnter={() => handleMouseEnterTab(index)}
|
// onMouseEnter={() => handleMouseEnterTab(index)}
|
||||||
onMouseLeave={() => handleMouseLeaveTab(index)}
|
// onMouseLeave={() => handleMouseLeaveTab(index)}
|
||||||
tabRefs={tabRefs}
|
tabRefs={tabRefs}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
Loading…
Reference in New Issue
Block a user