import { css } from '@emotion/css'; import React, { FC, ReactNode, useContext, useEffect } from 'react'; import { connect, ConnectedProps } from 'react-redux'; import { useLocation } from 'react-router-dom'; import { locationUtil, textUtil } from '@grafana/data'; import { selectors as e2eSelectors } from '@grafana/e2e-selectors/src'; import { locationService } from '@grafana/runtime'; import { ButtonGroup, ModalsController, ToolbarButton, PageToolbar, useForceUpdate, Tag, ToolbarButtonRow, ModalsContext, ConfirmModal, } from '@grafana/ui'; import { AppChromeUpdate } from 'app/core/components/AppChrome/AppChromeUpdate'; import { NavToolbarSeparator } from 'app/core/components/AppChrome/NavToolbarSeparator'; import config from 'app/core/config'; import { useGrafana } from 'app/core/context/GrafanaContext'; import { useAppNotification } from 'app/core/copy/appNotification'; import { appEvents } from 'app/core/core'; import { useBusEvent } from 'app/core/hooks/useBusEvent'; import { t, Trans } from 'app/core/internationalization'; import { SaveDashboardDrawer } from 'app/features/dashboard/components/SaveDashboard/SaveDashboardDrawer'; import { ShareModal } from 'app/features/dashboard/components/ShareModal'; import { playlistSrv } from 'app/features/playlist/PlaylistSrv'; import { updateTimeZoneForSession } from 'app/features/profile/state/reducers'; import { KioskMode } from 'app/types'; import { DashboardMetaChangedEvent, ShowModalReactEvent } from 'app/types/events'; import { setStarred } from '../../../../core/reducers/navBarTree'; import { getDashboardSrv } from '../../services/DashboardSrv'; import { DashboardModel } from '../../state'; import { DashNavButton } from './DashNavButton'; import { DashNavTimeControls } from './DashNavTimeControls'; const mapDispatchToProps = { setStarred, updateTimeZoneForSession, }; const connector = connect(null, mapDispatchToProps); const selectors = e2eSelectors.pages.Dashboard.DashNav; export interface OwnProps { dashboard: DashboardModel; isFullscreen: boolean; kioskMode?: KioskMode | null; hideTimePicker: boolean; folderTitle?: string; title: string; shareModalActiveTab?: string; onAddPanel: () => void; } interface DashNavButtonModel { show: (props: Props) => boolean; component: FC>; index?: number | 'end'; } const customLeftActions: DashNavButtonModel[] = []; const customRightActions: DashNavButtonModel[] = []; export function addCustomLeftAction(content: DashNavButtonModel) { customLeftActions.push(content); } export function addCustomRightAction(content: DashNavButtonModel) { customRightActions.push(content); } type Props = OwnProps & ConnectedProps; export const DashNav = React.memo((props) => { const forceUpdate = useForceUpdate(); const { chrome } = useGrafana(); const { showModal, hideModal } = useContext(ModalsContext); // We don't really care about the event payload here only that it triggeres a re-render of this component useBusEvent(props.dashboard.events, DashboardMetaChangedEvent); const originalUrl = props.dashboard.snapshot?.originalUrl ?? ''; const gotoSnapshotOrigin = () => { window.location.href = textUtil.sanitizeUrl(props.dashboard.snapshot.originalUrl); }; const notifyApp = useAppNotification(); const onOpenSnapshotOriginal = () => { try { const sanitizedUrl = new URL(textUtil.sanitizeUrl(originalUrl), config.appUrl); const appUrl = new URL(config.appUrl); if (sanitizedUrl.host !== appUrl.host) { appEvents.publish( new ShowModalReactEvent({ component: ConfirmModal, props: { title: 'Proceed to external site?', modalClass: modalStyles, body: ( <>

{`This link connects to an external website at`} {originalUrl}

{"Are you sure you'd like to proceed?"}

), confirmVariant: 'primary', confirmText: 'Proceed', onConfirm: gotoSnapshotOrigin, }, }) ); } else { gotoSnapshotOrigin(); } } catch (err) { notifyApp.error('Invalid URL', err instanceof Error ? err.message : undefined); } }; const onStarDashboard = () => { const dashboardSrv = getDashboardSrv(); const { dashboard, setStarred } = props; dashboardSrv.starDashboard(dashboard.uid, Boolean(dashboard.meta.isStarred)).then((newState) => { setStarred({ id: dashboard.uid, title: dashboard.title, url: dashboard.meta.url ?? '', isStarred: newState }); dashboard.meta.isStarred = newState; forceUpdate(); }); }; const onClose = () => { locationService.partial({ viewPanel: null }); }; const onToggleTVMode = () => { chrome.onToggleKioskMode(); }; const onOpenSettings = () => { locationService.partial({ editview: 'settings' }); }; const onPlaylistPrev = () => { playlistSrv.prev(); }; const onPlaylistNext = () => { playlistSrv.next(); }; const onPlaylistStop = () => { playlistSrv.stop(); forceUpdate(); }; const addCustomContent = (actions: DashNavButtonModel[], buttons: ReactNode[]) => { actions.map((action, index) => { const Component = action.component; const element = ; typeof action.index === 'number' ? buttons.splice(action.index, 0, element) : buttons.push(element); }); }; const isPlaylistRunning = () => { return playlistSrv.isPlaying; }; // Open/Close useEffect(() => { const dashboard = props.dashboard; const shareModalActiveTab = props.shareModalActiveTab; const { canShare } = dashboard.meta; if (canShare && shareModalActiveTab) { // automagically open modal showModal(ShareModal, { dashboard, onDismiss: hideModal, activeTab: shareModalActiveTab, }); } return () => { hideModal(); }; }, [showModal, hideModal, props.dashboard, props.shareModalActiveTab]); const renderLeftActions = () => { const { dashboard, kioskMode } = props; const { canStar, canShare, isStarred } = dashboard.meta; const buttons: ReactNode[] = []; if (kioskMode || isPlaylistRunning()) { return []; } if (canStar) { let desc = isStarred ? t('dashboard.toolbar.unmark-favorite', 'Unmark as favorite') : t('dashboard.toolbar.mark-favorite', 'Mark as favorite'); buttons.push( ); } if (canShare) { buttons.push( {({ showModal, hideModal }) => ( { showModal(ShareModal, { dashboard, onDismiss: hideModal, }); }} /> )} ); } if (dashboard.meta.publicDashboardEnabled) { buttons.push( ); } addCustomContent(customLeftActions, buttons); return buttons; }; const renderPlaylistControls = () => { return ( Stop playlist ); }; const renderTimeControls = () => { const { dashboard, updateTimeZoneForSession, hideTimePicker } = props; if (hideTimePicker) { return null; } return ( ); }; const renderRightActions = () => { const { dashboard, onAddPanel, isFullscreen, kioskMode } = props; const { canSave, canEdit, showSettings } = dashboard.meta; const { snapshot } = dashboard; const snapshotUrl = snapshot && snapshot.originalUrl; const buttons: ReactNode[] = []; const tvButton = config.featureToggles.topnav ? null : ( ); if (isPlaylistRunning()) { return [renderPlaylistControls(), renderTimeControls()]; } if (kioskMode === KioskMode.TV) { return [renderTimeControls(), tvButton]; } if (canEdit && !isFullscreen) { buttons.push( ); } if (canSave && !isFullscreen) { buttons.push( {({ showModal, hideModal }) => ( { showModal(SaveDashboardDrawer, { dashboard, onDismiss: hideModal, }); }} /> )} ); } if (snapshotUrl) { buttons.push( ); } if (showSettings) { buttons.push( ); } addCustomContent(customRightActions, buttons); buttons.push(renderTimeControls()); buttons.push(tvButton); if (config.featureToggles.scenes) { buttons.push( locationService.push(`/scenes/dashboard/${dashboard.uid}`)} /> ); } return buttons; }; const { isFullscreen, title, folderTitle } = props; // this ensures the component rerenders when the location changes const location = useLocation(); const titleHref = locationUtil.getUrlForPartial(location, { search: 'open' }); const parentHref = locationUtil.getUrlForPartial(location, { search: 'open', query: 'folder:current' }); const onGoBack = isFullscreen ? onClose : undefined; if (config.featureToggles.topnav) { return ( {renderLeftActions()} {renderRightActions()} } /> ); } return ( {renderRightActions()} ); }); DashNav.displayName = 'DashNav'; export default connector(DashNav); const modalStyles = css({ width: 'max-content', maxWidth: '80vw', });