123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336 |
- import Mousetrap from 'mousetrap';
- import 'mousetrap-global-bind';
- import 'mousetrap/plugins/global-bind/mousetrap-global-bind';
- import { LegacyGraphHoverClearEvent, locationUtil } from '@grafana/data';
- import { locationService } from '@grafana/runtime';
- import appEvents from 'app/core/app_events';
- import { getExploreUrl } from 'app/core/utils/explore';
- import { SaveDashboardDrawer } from 'app/features/dashboard/components/SaveDashboard/SaveDashboardDrawer';
- import { ShareModal } from 'app/features/dashboard/components/ShareModal';
- import { DashboardModel } from 'app/features/dashboard/state';
- import { getTimeSrv } from '../../features/dashboard/services/TimeSrv';
- import { getDatasourceSrv } from '../../features/plugins/datasource_srv';
- import {
- RemovePanelEvent,
- ShiftTimeEvent,
- ShiftTimeEventDirection,
- ShowModalReactEvent,
- ZoomOutEvent,
- AbsoluteTimeEvent,
- } from '../../types/events';
- import { HelpModal } from '../components/help/HelpModal';
- import { contextSrv } from '../core';
- import { exitKioskMode, toggleKioskMode } from '../navigation/kiosk';
- import { toggleTheme } from './toggleTheme';
- import { withFocusedPanel } from './withFocusedPanelId';
- export class KeybindingSrv {
- reset() {
- Mousetrap.reset();
- }
- initGlobals() {
- if (locationService.getLocation().pathname !== '/login') {
- this.bind(['?', 'h'], this.showHelpModal);
- this.bind('g h', this.goToHome);
- this.bind('g a', this.openAlerting);
- this.bind('g p', this.goToProfile);
- this.bind('s o', this.openSearch);
- this.bind('t a', this.makeAbsoluteTime);
- this.bind('f', this.openSearch);
- this.bind('esc', this.exit);
- this.bindGlobal('esc', this.globalEsc);
- }
- this.bind('t t', () => toggleTheme(false));
- this.bind('t r', () => toggleTheme(true));
- }
- globalEsc() {
- const anyDoc = document as any;
- const activeElement = anyDoc.activeElement;
- // typehead needs to handle it
- const typeaheads = document.querySelectorAll('.slate-typeahead--open');
- if (typeaheads.length > 0) {
- return;
- }
- // second check if we are in an input we can blur
- if (activeElement && activeElement.blur) {
- if (
- activeElement.nodeName === 'INPUT' ||
- activeElement.nodeName === 'TEXTAREA' ||
- activeElement.hasAttribute('data-slate-editor')
- ) {
- anyDoc.activeElement.blur();
- return;
- }
- }
- // ok no focused input or editor that should block this, let exist!
- this.exit();
- }
- private openSearch() {
- locationService.partial({ search: 'open' });
- }
- private closeSearch() {
- locationService.partial({ search: null });
- }
- private openAlerting() {
- locationService.push('/alerting');
- }
- private goToHome() {
- locationService.push('/');
- }
- private goToProfile() {
- locationService.push('/profile');
- }
- private makeAbsoluteTime() {
- appEvents.publish(new AbsoluteTimeEvent());
- }
- private showHelpModal() {
- appEvents.publish(new ShowModalReactEvent({ component: HelpModal }));
- }
- private exit() {
- const search = locationService.getSearchObject();
- if (search.editview) {
- locationService.partial({ editview: null });
- return;
- }
- if (search.inspect) {
- locationService.partial({ inspect: null, inspectTab: null });
- return;
- }
- if (search.editPanel) {
- locationService.partial({ editPanel: null, tab: null });
- return;
- }
- if (search.viewPanel) {
- locationService.partial({ viewPanel: null, tab: null });
- return;
- }
- if (search.kiosk) {
- exitKioskMode();
- }
- if (search.search) {
- this.closeSearch();
- }
- }
- private showDashEditView() {
- locationService.partial({
- editview: 'settings',
- });
- }
- bind(keyArg: string | string[], fn: () => void) {
- Mousetrap.bind(
- keyArg,
- (evt: any) => {
- evt.preventDefault();
- evt.stopPropagation();
- evt.returnValue = false;
- fn.call(this);
- },
- 'keydown'
- );
- }
- bindGlobal(keyArg: string, fn: () => void) {
- Mousetrap.bindGlobal(
- keyArg,
- (evt: any) => {
- evt.preventDefault();
- evt.stopPropagation();
- evt.returnValue = false;
- fn.call(this);
- },
- 'keydown'
- );
- }
- unbind(keyArg: string, keyType?: string) {
- Mousetrap.unbind(keyArg, keyType);
- }
- bindWithPanelId(keyArg: string, fn: (panelId: number) => void) {
- this.bind(keyArg, withFocusedPanel(fn));
- }
- setupTimeRangeBindings(updateUrl = true) {
- this.bind('t z', () => {
- appEvents.publish(new ZoomOutEvent({ scale: 2, updateUrl }));
- });
- this.bind('ctrl+z', () => {
- appEvents.publish(new ZoomOutEvent({ scale: 2, updateUrl }));
- });
- this.bind('t left', () => {
- appEvents.publish(new ShiftTimeEvent({ direction: ShiftTimeEventDirection.Left, updateUrl }));
- });
- this.bind('t right', () => {
- appEvents.publish(new ShiftTimeEvent({ direction: ShiftTimeEventDirection.Right, updateUrl }));
- });
- }
- setupDashboardBindings(dashboard: DashboardModel) {
- this.bind('mod+o', () => {
- dashboard.graphTooltip = (dashboard.graphTooltip + 1) % 3;
- dashboard.events.publish(new LegacyGraphHoverClearEvent());
- dashboard.startRefresh();
- });
- this.bind('mod+s', () => {
- if (dashboard.meta.canSave) {
- appEvents.publish(
- new ShowModalReactEvent({
- component: SaveDashboardDrawer,
- props: {
- dashboard,
- },
- })
- );
- }
- });
- this.setupTimeRangeBindings();
- // edit panel
- this.bindWithPanelId('e', (panelId) => {
- if (dashboard.canEditPanelById(panelId)) {
- const isEditing = locationService.getSearchObject().editPanel !== undefined;
- locationService.partial({ editPanel: isEditing ? null : panelId });
- }
- });
- // view panel
- this.bindWithPanelId('v', (panelId) => {
- const isViewing = locationService.getSearchObject().viewPanel !== undefined;
- locationService.partial({ viewPanel: isViewing ? null : panelId });
- });
- this.bindWithPanelId('i', (panelId) => {
- locationService.partial({ inspect: panelId });
- });
- // jump to explore if permissions allow
- if (contextSrv.hasAccessToExplore()) {
- this.bindWithPanelId('x', async (panelId) => {
- const panel = dashboard.getPanelById(panelId)!;
- const url = await getExploreUrl({
- panel,
- datasourceSrv: getDatasourceSrv(),
- timeSrv: getTimeSrv(),
- });
- if (url) {
- const urlWithoutBase = locationUtil.stripBaseFromUrl(url);
- if (urlWithoutBase) {
- locationService.push(urlWithoutBase);
- }
- }
- });
- }
- // delete panel
- this.bindWithPanelId('p r', (panelId) => {
- if (dashboard.canEditPanelById(panelId) && !(dashboard.panelInView || dashboard.panelInEdit)) {
- appEvents.publish(new RemovePanelEvent(panelId));
- }
- });
- // duplicate panel
- this.bindWithPanelId('p d', (panelId) => {
- if (dashboard.canEditPanelById(panelId)) {
- const panelIndex = dashboard.getPanelInfoById(panelId)!.index;
- dashboard.duplicatePanel(dashboard.panels[panelIndex]);
- }
- });
- // share panel
- this.bindWithPanelId('p s', (panelId) => {
- const panelInfo = dashboard.getPanelInfoById(panelId);
- appEvents.publish(
- new ShowModalReactEvent({
- component: ShareModal,
- props: {
- dashboard: dashboard,
- panel: panelInfo?.panel,
- },
- })
- );
- });
- // toggle panel legend
- this.bindWithPanelId('p l', (panelId) => {
- const panelInfo = dashboard.getPanelInfoById(panelId)!;
- if (panelInfo.panel.legend) {
- panelInfo.panel.legend.show = !panelInfo.panel.legend.show;
- panelInfo.panel.render();
- }
- });
- // toggle all panel legends
- this.bind('d l', () => {
- dashboard.toggleLegendsForAll();
- });
- // collapse all rows
- this.bind('d shift+c', () => {
- dashboard.collapseRows();
- });
- // expand all rows
- this.bind('d shift+e', () => {
- dashboard.expandRows();
- });
- this.bind('d n', () => {
- locationService.push('/dashboard/new');
- });
- this.bind('d r', () => {
- dashboard.startRefresh();
- });
- this.bind('d s', () => {
- this.showDashEditView();
- });
- this.bind('d k', () => {
- toggleKioskMode();
- });
- //Autofit panels
- this.bind('d a', () => {
- // this has to be a full page reload
- const queryParams = locationService.getSearchObject();
- const newUrlParam = queryParams.autofitpanels ? '' : '&autofitpanels';
- window.location.href = window.location.href + newUrlParam;
- });
- }
- }
- export const keybindingSrv = new KeybindingSrv();
|