import React, {forwardRef, useState} from 'react'; import type {HeaderProps} from '../../typings/hyper'; import {decorate, getTabsProps} from '../utils/plugins'; import Tabs_ from './tabs'; const Tabs = decorate(Tabs_, 'Tabs'); const Header = forwardRef((props, ref) => { const [headerMouseDownWindowX, setHeaderMouseDownWindowX] = useState(0); const [headerMouseDownWindowY, setHeaderMouseDownWindowY] = useState(0); const onChangeIntent = (active: string) => { // we ignore clicks if they're a byproduct of a drag // motion to move the window if (window.screenX !== headerMouseDownWindowX || window.screenY !== headerMouseDownWindowY) { return; } props.onChangeTab(active); }; const handleHeaderMouseDown = () => { // the hack of all hacks, this prevents the term // iframe from losing focus, for example, when // the user drags the nav around // Fixed by calling window.focusActiveTerm(), thus we can support drag tab // ev.preventDefault(); // persist start positions of a potential drag motion // to differentiate dragging from clicking setHeaderMouseDownWindowX(window.screenX); setHeaderMouseDownWindowY(window.screenY); }; const handleHamburgerMenuClick = (event: React.MouseEvent) => { let {right: x, bottom: y} = event.currentTarget.getBoundingClientRect(); x -= 15; // to compensate padding y -= 12; // ^ same props.openHamburgerMenu({x, y}); }; const handleMaximizeClick = () => { if (props.maximized) { props.unmaximize(); } else { props.maximize(); } }; const handleMinimizeClick = () => { props.minimize(); }; const handleCloseClick = () => { props.close(); }; const getWindowHeaderConfig = () => { const {showHamburgerMenu, showWindowControls} = props; const defaults = { hambMenu: !props.isMac, // show by default on windows and linux winCtrls: !props.isMac // show by default on Windows and Linux }; // don't allow the user to change defaults on macOS if (props.isMac) { return defaults; } return { hambMenu: showHamburgerMenu === '' ? defaults.hambMenu : showHamburgerMenu, winCtrls: showWindowControls === '' ? defaults.winCtrls : showWindowControls }; }; const {isMac} = props; const {borderColor} = props; let title = 'Hyper'; if (props.tabs.length === 1 && props.tabs[0].title) { // if there's only one tab we use its title as the window title title = props.tabs[0].title; } const {hambMenu, winCtrls} = getWindowHeaderConfig(); const left = winCtrls === 'left'; const maxButtonHref = props.maximized ? './renderer/assets/icons.svg#restore-window' : './renderer/assets/icons.svg#maximize-window'; return (
window.focusActiveTerm()} onDoubleClick={handleMaximizeClick} ref={ref} > {!isMac && (
1 ? 'header_windowHeaderWithBorder' : ''}`} style={{borderColor}} > {hambMenu && ( )} {title} {winCtrls && (
)}
)} {props.customChildrenBefore} {props.customChildren}
); }); Header.displayName = 'Header'; export default Header;