| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311 |
- import $ from 'jquery';
- import _ from 'lodash';
- import coreModule from 'app/core/core_module';
- import appEvents from 'app/core/app_events';
- import { getExploreUrl } from 'app/core/utils/explore';
- import { store } from 'app/store/store';
- import Mousetrap from 'mousetrap';
- import 'mousetrap-global-bind';
- import { ContextSrv } from './context_srv';
- import { ILocationService, ITimeoutService } from 'angular';
- export class KeybindingSrv {
- helpModal: boolean;
- modalOpen = false;
- timepickerOpen = false;
- /** @ngInject */
- constructor(
- private $rootScope: any,
- private $location: ILocationService,
- private $timeout: ITimeoutService,
- private datasourceSrv: any,
- private timeSrv: any,
- private contextSrv: ContextSrv
- ) {
- // clear out all shortcuts on route change
- $rootScope.$on('$routeChangeSuccess', () => {
- Mousetrap.reset();
- // rebind global shortcuts
- this.setupGlobal();
- });
- this.setupGlobal();
- appEvents.on('show-modal', () => (this.modalOpen = true));
- appEvents.on('timepickerOpen', () => (this.timepickerOpen = true));
- appEvents.on('timepickerClosed', () => (this.timepickerOpen = false));
- }
- setupGlobal() {
- if (!(this.$location.path() === '/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('f', this.openSearch);
- this.bind('esc', this.exit);
- }
- }
- openSearch() {
- appEvents.emit('show-dash-search');
- }
- openAlerting() {
- this.$location.url('/alerting');
- }
- goToHome() {
- this.$location.url('/');
- }
- goToProfile() {
- this.$location.url('/profile');
- }
- showHelpModal() {
- appEvents.emit('show-modal', { templateHtml: '<help-modal></help-modal>' });
- }
- exit() {
- const popups = $('.popover.in, .slate-typeahead');
- if (popups.length > 0) {
- return;
- }
- appEvents.emit('hide-modal');
- if (this.modalOpen) {
- this.modalOpen = false;
- return;
- }
- if (this.timepickerOpen) {
- this.$rootScope.appEvent('closeTimepicker');
- this.timepickerOpen = false;
- return;
- }
- // close settings view
- const search = this.$location.search();
- if (search.editview) {
- delete search.editview;
- this.$location.search(search);
- return;
- }
- if (search.fullscreen) {
- appEvents.emit('panel-change-view', { fullscreen: false, edit: false });
- return;
- }
- if (search.kiosk) {
- this.$rootScope.appEvent('toggle-kiosk-mode', { exit: true });
- }
- }
- bind(keyArg: string | string[], fn: () => void) {
- Mousetrap.bind(
- keyArg,
- (evt: any) => {
- evt.preventDefault();
- evt.stopPropagation();
- evt.returnValue = false;
- return this.$rootScope.$apply(fn.bind(this));
- },
- 'keydown'
- );
- }
- bindGlobal(keyArg: string, fn: () => void) {
- Mousetrap.bindGlobal(
- keyArg,
- (evt: any) => {
- evt.preventDefault();
- evt.stopPropagation();
- evt.returnValue = false;
- return this.$rootScope.$apply(fn.bind(this));
- },
- 'keydown'
- );
- }
- unbind(keyArg: string, keyType?: string) {
- Mousetrap.unbind(keyArg, keyType);
- }
- showDashEditView() {
- const search = _.extend(this.$location.search(), { editview: 'settings' });
- this.$location.search(search);
- }
- setupDashboardBindings(scope: any, dashboard: any) {
- this.bind('mod+o', () => {
- dashboard.graphTooltip = (dashboard.graphTooltip + 1) % 3;
- appEvents.emit('graph-hover-clear');
- dashboard.startRefresh();
- });
- this.bind('mod+s', () => {
- scope.appEvent('save-dashboard');
- });
- this.bind('t z', () => {
- scope.appEvent('zoom-out', 2);
- });
- this.bind('ctrl+z', () => {
- scope.appEvent('zoom-out', 2);
- });
- this.bind('t left', () => {
- scope.appEvent('shift-time', -1);
- });
- this.bind('t right', () => {
- scope.appEvent('shift-time', 1);
- });
- // edit panel
- this.bind('e', () => {
- if (dashboard.meta.focusPanelId && dashboard.meta.canEdit) {
- appEvents.emit('panel-change-view', {
- fullscreen: true,
- edit: true,
- panelId: dashboard.meta.focusPanelId,
- toggle: true,
- });
- }
- });
- // view panel
- this.bind('v', () => {
- if (dashboard.meta.focusPanelId) {
- appEvents.emit('panel-change-view', {
- fullscreen: true,
- panelId: dashboard.meta.focusPanelId,
- toggle: true,
- });
- }
- });
- // jump to explore if permissions allow
- if (this.contextSrv.hasAccessToExplore()) {
- this.bind('x', async () => {
- if (dashboard.meta.focusPanelId) {
- const panel = dashboard.getPanelById(dashboard.meta.focusPanelId);
- const datasource = await this.datasourceSrv.get(panel.datasource);
- const url = await getExploreUrl(panel, panel.targets, datasource, this.datasourceSrv, this.timeSrv);
- if (url) {
- this.$timeout(() => this.$location.url(url));
- }
- }
- });
- }
- // delete panel
- this.bind('p r', () => {
- if (dashboard.meta.focusPanelId && dashboard.meta.canEdit) {
- appEvents.emit('remove-panel', dashboard.meta.focusPanelId);
- dashboard.meta.focusPanelId = 0;
- }
- });
- // duplicate panel
- this.bind('p d', () => {
- if (dashboard.meta.focusPanelId && dashboard.meta.canEdit) {
- const panelIndex = dashboard.getPanelInfoById(dashboard.meta.focusPanelId).index;
- dashboard.duplicatePanel(dashboard.panels[panelIndex]);
- }
- });
- // share panel
- this.bind('p s', () => {
- if (dashboard.meta.focusPanelId) {
- const shareScope = scope.$new();
- const panelInfo = dashboard.getPanelInfoById(dashboard.meta.focusPanelId);
- shareScope.panel = panelInfo.panel;
- shareScope.dashboard = dashboard;
- appEvents.emit('show-modal', {
- src: 'public/app/features/dashboard/components/ShareModal/template.html',
- scope: shareScope,
- });
- }
- });
- // toggle panel legend
- this.bind('p l', () => {
- if (dashboard.meta.focusPanelId) {
- const panelInfo = dashboard.getPanelInfoById(dashboard.meta.focusPanelId);
- if (panelInfo.panel.legend) {
- const panelRef = dashboard.getPanelById(dashboard.meta.focusPanelId);
- panelRef.legend.show = !panelRef.legend.show;
- panelRef.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', () => {
- this.$location.url('/dashboard/new');
- });
- this.bind('d r', () => {
- dashboard.startRefresh();
- });
- this.bind('d s', () => {
- this.showDashEditView();
- });
- this.bind('d k', () => {
- appEvents.emit('toggle-kiosk-mode');
- });
- this.bind('d v', () => {
- appEvents.emit('toggle-view-mode');
- });
- //Autofit panels
- this.bind('d a', () => {
- // this has to be a full page reload
- const queryParams = store.getState().location.query;
- const newUrlParam = queryParams.autofitpanels ? '' : '&autofitpanels';
- window.location.href = window.location.href + newUrlParam;
- });
- }
- }
- coreModule.service('keybindingSrv', KeybindingSrv);
- /**
- * Code below exports the service to react components
- */
- let singletonInstance: KeybindingSrv;
- export function setKeybindingSrv(instance: KeybindingSrv) {
- singletonInstance = instance;
- }
- export function getKeybindingSrv(): KeybindingSrv {
- return singletonInstance;
- }
|