From 0e1e21dcadba9a1411dc8a5abacce9099a587162 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Wed, 3 May 2017 14:55:22 -0400 Subject: [PATCH 01/21] Move dashboard panel rendering logic to each registered type. --- src/core_plugins/kibana/index.js | 3 +- .../public/dashboard/__tests__/panel.js | 4 +- .../kibana/public/dashboard/dashboard.js | 1 - .../panel/get_object_loaders_for_dashboard.js | 11 -- .../dashboard/panel/load_saved_object.js | 16 -- .../kibana/public/dashboard/panel/panel.html | 39 +---- .../kibana/public/dashboard/panel/panel.js | 137 ++++++------------ .../embeddable/search_embeddable_handler.js | 75 ++++++++++ .../search_embeddable_handler_provider.js | 12 ++ .../discover/embeddable/search_template.html | 16 ++ .../visualize_embeddable_handler.js | 49 +++++++ .../visualize_embeddable_handler_provider.js | 11 ++ .../embeddable/visualize_template.html | 11 ++ src/ui/public/registry/embeddable_handlers.js | 7 + src/ui/ui_exports.js | 5 + 15 files changed, 240 insertions(+), 157 deletions(-) delete mode 100644 src/core_plugins/kibana/public/dashboard/panel/get_object_loaders_for_dashboard.js delete mode 100644 src/core_plugins/kibana/public/dashboard/panel/load_saved_object.js create mode 100644 src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js create mode 100644 src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js create mode 100644 src/core_plugins/kibana/public/discover/embeddable/search_template.html create mode 100644 src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js create mode 100644 src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js create mode 100644 src/core_plugins/kibana/public/visualize/embeddable/visualize_template.html create mode 100644 src/ui/public/registry/embeddable_handlers.js diff --git a/src/core_plugins/kibana/index.js b/src/core_plugins/kibana/index.js index d2a15195e29b2..f7124b0201aa5 100644 --- a/src/core_plugins/kibana/index.js +++ b/src/core_plugins/kibana/index.js @@ -58,7 +58,8 @@ export default function (kibana) { 'navbarExtensions', 'managementSections', 'devTools', - 'docViews' + 'docViews', + 'embeddableHandlers', ], injectVars, }, diff --git a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js index 12ecb45c3e986..047ee0cb40938 100644 --- a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js +++ b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js @@ -71,7 +71,7 @@ describe('dashboard panel', function () { expect($scope.error).to.be('Could not locate that visualization (id: foo1)'); parentScope.$digest(); const content = $el.find('.panel-content'); - expect(content).to.have.length(0); + expect(content.children().length).to.be(0); }); }); @@ -81,7 +81,7 @@ describe('dashboard panel', function () { expect($scope.error).not.to.be.ok(); parentScope.$digest(); const content = $el.find('.panel-content'); - expect(content).to.have.length(1); + expect(content.children().length).to.be.greaterThan(0); }); }); }); diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.js b/src/core_plugins/kibana/public/dashboard/dashboard.js index 0ac9fca905886..1911707510883 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard.js @@ -22,7 +22,6 @@ import { UtilsBrushEventProvider } from 'ui/utils/brush_event'; import { FilterBarClickHandlerProvider } from 'ui/filter_bar/filter_bar_click_handler'; import { DashboardState } from './dashboard_state'; import { notify } from 'ui/notify'; -import './panel/get_object_loaders_for_dashboard'; import { documentationLinks } from 'ui/documentation_links/documentation_links'; import { showCloneModal } from './top_nav/show_clone_modal'; import { ESC_KEY_CODE } from 'ui_framework/services'; diff --git a/src/core_plugins/kibana/public/dashboard/panel/get_object_loaders_for_dashboard.js b/src/core_plugins/kibana/public/dashboard/panel/get_object_loaders_for_dashboard.js deleted file mode 100644 index 40c99b3c29372..0000000000000 --- a/src/core_plugins/kibana/public/dashboard/panel/get_object_loaders_for_dashboard.js +++ /dev/null @@ -1,11 +0,0 @@ -import { uiModules } from 'ui/modules'; -const module = uiModules.get('app/dashboard'); - -/** - * We have more types available than just 'search' and 'visualization' but as of now, they - * can't be added to a dashboard. - */ -module.factory('getObjectLoadersForDashboard', function (savedSearches, savedVisualizations) { - return () => [savedSearches, savedVisualizations]; -}); - diff --git a/src/core_plugins/kibana/public/dashboard/panel/load_saved_object.js b/src/core_plugins/kibana/public/dashboard/panel/load_saved_object.js deleted file mode 100644 index b7f3bed8364d3..0000000000000 --- a/src/core_plugins/kibana/public/dashboard/panel/load_saved_object.js +++ /dev/null @@ -1,16 +0,0 @@ -/** - * Retrieves the saved object represented by the panel and returns it, along with the appropriate - * edit Url. - * @param {Array.} loaders - The available loaders for different panel types. - * @param {PanelState} panel - * @returns {Promise.<{savedObj: SavedObject, editUrl: String}>} - */ -export function loadSavedObject(loaders, panel) { - const loader = loaders.find((loader) => loader.type === panel.type); - if (!loader) { - throw new Error(`No loader for object of type ${panel.type}`); - } - return loader.get(panel.id).then(savedObj => { - return { savedObj, editUrl: loader.urlFor(panel.id) }; - }); -} diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.html b/src/core_plugins/kibana/public/dashboard/panel/panel.html index 38d56f67777ea..d20d0fdde1bee 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.html +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.html @@ -1,11 +1,11 @@ -
+
- {{::savedObj.title}} + {{::title}} - - - - - +
diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index e11a67f280e5a..afe6c190efde8 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -1,23 +1,17 @@ import _ from 'lodash'; import 'ui/visualize'; import 'ui/doc_table'; -import * as columnActions from 'ui/doc_table/actions/columns'; -import 'plugins/kibana/dashboard/panel/get_object_loaders_for_dashboard'; import 'plugins/kibana/visualize/saved_visualizations'; import 'plugins/kibana/discover/saved_searches'; -import { FilterManagerProvider } from 'ui/filter_manager'; import { uiModules } from 'ui/modules'; import panelTemplate from 'plugins/kibana/dashboard/panel/panel.html'; import { savedObjectManagementRegistry } from 'plugins/kibana/management/saved_object_registry'; -import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; -import { loadSavedObject } from 'plugins/kibana/dashboard/panel/load_saved_object'; import { DashboardViewMode } from '../dashboard_view_mode'; +import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; uiModules .get('app/dashboard') -.directive('dashboardPanel', function (savedVisualizations, savedSearches, Notifier, Private, $injector, getObjectLoadersForDashboard) { - const filterManager = Private(FilterManagerProvider); - +.directive('dashboardPanel', function (Notifier, Private, $injector) { const services = savedObjectManagementRegistry.all().map(function (serviceObj) { const service = $injector.get(serviceObj.service); return { @@ -91,98 +85,59 @@ uiModules link: function ($scope, element) { if (!$scope.panel.id || !$scope.panel.type) return; - /** - * Initializes the panel for the saved object. - * @param {{savedObj: SavedObject, editUrl: String}} savedObjectInfo - */ - function initializePanel(savedObjectInfo) { - $scope.savedObj = savedObjectInfo.savedObj; - $scope.editUrl = savedObjectInfo.editUrl; - - element.on('$destroy', function () { - $scope.savedObj.destroy(); - $scope.$destroy(); - }); + const savePanelState = (panel) => { + $scope.panel = Object.assign($scope.panel, panel); + $scope.saveState(); + }; - // create child ui state from the savedObj - const uiState = $scope.savedObj.uiStateJSON ? JSON.parse($scope.savedObj.uiStateJSON) : {}; - $scope.uiState = $scope.createChildUiState(getPersistedStateId($scope.panel), uiState); + $scope.isViewOnlyMode = () => { + return $scope.dashboardViewMode === DashboardViewMode.VIEW || $scope.isFullScreenMode; + }; - if ($scope.panel.type === savedVisualizations.type && $scope.savedObj.vis) { - $scope.savedObj.vis.setUiState($scope.uiState); - $scope.savedObj.vis.listeners.click = $scope.getVisClickHandler(); - $scope.savedObj.vis.listeners.brush = $scope.getVisBrushHandler(); - $scope.registerPanelIndexPattern($scope.panel.panelIndex, $scope.savedObj.vis.indexPattern); - } else if ($scope.panel.type === savedSearches.type) { - if ($scope.savedObj.searchSource) { - $scope.registerPanelIndexPattern($scope.panel.panelIndex, $scope.savedObj.searchSource.get('index')); - } - // This causes changes to a saved search to be hidden, but also allows - // the user to locally modify and save changes to a saved search only in a dashboard. - // See https://github.com/elastic/kibana/issues/9523 for more details. - $scope.panel.columns = $scope.panel.columns || $scope.savedObj.columns; - $scope.panel.sort = $scope.panel.sort || $scope.savedObj.sort; + // TODO: vis actions should be generalized for use by all panel renderers, e.g. updateFilters, updateTimeRange. + const containerAPI = { + getVisClickHandler: $scope.getVisClickHandler, + getVisBrushHandler: $scope.getVisBrushHandler, + savePanelState, + getIsViewOnlyMode: $scope.isViewOnlyMode, + createChildUiState: $scope.createChildUiState, + registerPanelIndexPattern: $scope.registerPanelIndexPattern, + }; - $scope.setSortOrder = function setSortOrder(columnName, direction) { - $scope.panel.sort = [columnName, direction]; - $scope.saveState(); - }; + const handleError = (error) => { + $scope.error = error.message; - $scope.addColumn = function addColumn(columnName) { - $scope.savedObj.searchSource.get('index').popularizeField(columnName, 1); - columnActions.addColumn($scope.panel.columns, columnName); - $scope.saveState(); // sync to sharing url - }; + // Dashboard listens for this broadcast, once for every visualization (pendingVisCount). + // We need to broadcast even in the event of an error or it'll never fetch the data for + // other visualizations. + $scope.$root.$broadcast('ready:vis'); - $scope.removeColumn = function removeColumn(columnName) { - $scope.savedObj.searchSource.get('index').popularizeField(columnName, 1); - columnActions.removeColumn($scope.panel.columns, columnName); - $scope.saveState(); // sync to sharing url - }; + // If the savedObjectType matches the panel type, this means the object itself has been deleted, + // so we shouldn't even have an edit link. If they don't match, it means something else is wrong + // with the object (but the object still exists), so we link to the object editor instead. + const objectItselfDeleted = error.savedObjectType === $scope.panel.type; + if (objectItselfDeleted) return; - $scope.moveColumn = function moveColumn(columnName, newIndex) { - columnActions.moveColumn($scope.panel.columns, columnName, newIndex); - $scope.saveState(); // sync to sharing url - }; - } + const type = $scope.panel.type; + const id = $scope.panel.id; + const service = _.find(services, { type: type }); + if (!service) return; - $scope.filter = function (field, value, operator) { - const index = $scope.savedObj.searchSource.get('index').id; - filterManager.add(field, value, operator, index); - }; + $scope.editUrl = '#management/kibana/objects/' + service.name + '/' + id + '?notFound=' + error.savedObjectType; + }; + const embeddableHandlers = Private(EmbeddableHandlersRegistryProvider); + const embeddableHandler = embeddableHandlers.byName[$scope.panel.type]; + if (!embeddableHandler) { + handleError(`No embeddable handler for panel type ${$scope.panel.type} was found.`); + return; } - - $scope.loadedPanel = loadSavedObject(getObjectLoadersForDashboard(), $scope.panel) - .then(initializePanel) - .catch(function (e) { - $scope.error = e.message; - - // Dashboard listens for this broadcast, once for every visualization (pendingVisCount). - // We need to broadcast even in the event of an error or it'll never fetch the data for - // other visualizations. - $scope.$root.$broadcast('ready:vis'); - - // If the savedObjectType matches the panel type, this means the object itself has been deleted, - // so we shouldn't even have an edit link. If they don't match, it means something else is wrong - // with the object (but the object still exists), so we link to the object editor instead. - const objectItselfDeleted = e.savedObjectType === $scope.panel.type; - if (objectItselfDeleted) return; - - const type = $scope.panel.type; - const id = $scope.panel.id; - const service = _.find(services, { type: type }); - if (!service) return; - - $scope.editUrl = '#management/kibana/objects/' + service.name + '/' + id + '?notFound=' + e.savedObjectType; - }); - - /** - * @returns {boolean} True if the user can only view, not edit. - */ - $scope.isViewOnlyMode = () => { - return $scope.dashboardViewMode === DashboardViewMode.VIEW || $scope.isFullScreenMode; - }; + $scope.editUrl = embeddableHandler.getEditPath($scope.panel); + embeddableHandler.getTitleFor($scope.panel).then(title => { + $scope.title = title; + }); + $scope.loadedPanel = + embeddableHandler.render(element.find('#embeddedPanel').get(0), $scope.panel, containerAPI).catch(handleError); } }; }); diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js new file mode 100644 index 0000000000000..65db615b4c908 --- /dev/null +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -0,0 +1,75 @@ +import searchTemplate from './search_template.html'; +import angular from 'angular'; +import * as columnActions from 'ui/doc_table/actions/columns'; +import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; +import { FilterManagerProvider } from 'ui/filter_manager'; + +export class SearchEmbeddableHandler { + constructor($compile, $rootScope, searchLoader, Private) { + this.$compile = $compile; + this.searchLoader = searchLoader; + this.filterManager = Private(FilterManagerProvider); + this.$rootScope = $rootScope; + this.name = 'search'; + this.title = 'Saved Searches'; + } + + getEditPath(panel) { + return this.searchLoader.urlFor(panel.id); + } + + getTitleFor(panel) { + return this.searchLoader.get(panel.id).then(savedObject => savedObject.title); + } + + render(domNode, panel, actions) { + return this.searchLoader.get(panel.id).then((savedObject) => { + const editUrl = this.searchLoader.urlFor(panel.id); + const searchScope = this.$rootScope.$new(); + searchScope.editUrl = editUrl; + searchScope.savedObj = savedObject; + searchScope.panel = panel; + + // This causes changes to a saved search to be hidden, but also allows + // the user to locally modify and save changes to a saved search only in a dashboard. + // See https://github.com/elastic/kibana/issues/9523 for more details. + actions.savePanelState({ + columns: searchScope.panel.columns || searchScope.savedObj.columns, + sort: searchScope.panel.sort || searchScope.savedObj.sort + }); + + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + searchScope.uiState = actions.createChildUiState(getPersistedStateId(panel), uiState); + + searchScope.setSortOrder = function setSortOrder(columnName, direction) { + actions.savePanelState({ sort: [columnName, direction] }); + }; + + searchScope.addColumn = function addColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.addColumn(searchScope.panel.columns, columnName); + actions.savePanelState({}); // sync to sharing url + }; + + searchScope.removeColumn = function removeColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.removeColumn(searchScope.panel.columns, columnName); + actions.savePanelState({}); // sync to sharing url + }; + + searchScope.moveColumn = function moveColumn(columnName, newIndex) { + columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); + actions.savePanelState({}); // sync to sharing url + }; + + searchScope.filter = function (field, value, operator) { + const index = savedObject.searchSource.get('index').id; + this.filterManager.add(field, value, operator, index); + }; + + const searchInstance = this.$compile(searchTemplate)(searchScope); + const rootNode = angular.element(domNode); + rootNode.append(searchInstance); + }); + } +} diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js new file mode 100644 index 0000000000000..f11c3283604d1 --- /dev/null +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js @@ -0,0 +1,12 @@ +import { SearchEmbeddableHandler } from './search_embeddable_handler'; +import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; + +export function searchEmbeddableHandlerProvider(Private) { + const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches, Private) => { + return new SearchEmbeddableHandler($compile, $rootScope, savedSearches, Private); + }; + return Private(SearchEmbeddableHandlerProvider); +} + + +EmbeddableHandlersRegistryProvider.register(searchEmbeddableHandlerProvider); diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_template.html b/src/core_plugins/kibana/public/discover/embeddable/search_template.html new file mode 100644 index 0000000000000..0bb48561b27df --- /dev/null +++ b/src/core_plugins/kibana/public/discover/embeddable/search_template.html @@ -0,0 +1,16 @@ + + diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js new file mode 100644 index 0000000000000..9e841834b9679 --- /dev/null +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -0,0 +1,49 @@ +import angular from 'angular'; + +import visualizationTemplate from './visualize_template.html'; +import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; + +export class VisualizeEmbeddableHandler { + constructor($compile, $rootScope, visualizeLoader) { + this.$compile = $compile; + this.visualizeLoader = visualizeLoader; + this.$rootScope = $rootScope; + this.name = 'visualization'; + this.title = 'Visualizations'; + } + + getEditPath(panel) { + return this.visualizeLoader.urlFor(panel.id); + } + + getTitleFor(panel) { + return this.visualizeLoader.get(panel.id).then(savedObject => savedObject.title); + } + + render(domNode, panel, actions) { + return this.visualizeLoader.get(panel.id).then((savedObject) => { + const visualizeScope = this.$rootScope.$new(); + visualizeScope.editUrl = this.getEditPath(panel); + visualizeScope.savedObj = savedObject; + visualizeScope.panel = panel; + + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + visualizeScope.uiState = actions.createChildUiState(getPersistedStateId(panel), uiState); + + visualizeScope.savedObj.vis.setUiState(uiState); + visualizeScope.savedObj.vis.listeners.click = actions.getVisClickHandler(); + visualizeScope.savedObj.vis.listeners.brush = actions.getVisBrushHandler(); + visualizeScope.isFullScreenMode = actions.getIsViewOnlyMode(); + + const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); + const rootNode = angular.element(domNode); + rootNode.append(visualizationInstance); + + visualizationInstance.on('$destroy', function () { + visualizeScope.savedObj.destroy(); + visualizeScope.$destroy(); + }); + }); + } +} + diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js new file mode 100644 index 0000000000000..6abab61d6c053 --- /dev/null +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js @@ -0,0 +1,11 @@ +import { VisualizeEmbeddableHandler } from './visualize_embeddable_handler'; +import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; + +export function visualizeEmbeddableHandlerProvider(Private) { + const VisualizeEmbeddableHandlerProvider = ($compile, $rootScope, savedVisualizations) => { + return new VisualizeEmbeddableHandler($compile, $rootScope, savedVisualizations); + }; + return Private(VisualizeEmbeddableHandlerProvider); +} + +EmbeddableHandlersRegistryProvider.register(visualizeEmbeddableHandlerProvider); diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_template.html b/src/core_plugins/kibana/public/visualize/embeddable/visualize_template.html new file mode 100644 index 0000000000000..4a7d8fd6481a5 --- /dev/null +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_template.html @@ -0,0 +1,11 @@ + + diff --git a/src/ui/public/registry/embeddable_handlers.js b/src/ui/public/registry/embeddable_handlers.js new file mode 100644 index 0000000000000..0e093928c35e9 --- /dev/null +++ b/src/ui/public/registry/embeddable_handlers.js @@ -0,0 +1,7 @@ +import { uiRegistry } from 'ui/registry/_registry'; + +export const EmbeddableHandlersRegistryProvider = uiRegistry({ + name: 'embeddableHandlers', + index: ['name'], + order: ['title'] +}); diff --git a/src/ui/ui_exports.js b/src/ui/ui_exports.js index 02007b74d0b8d..39db7fce0edfc 100644 --- a/src/ui/ui_exports.js +++ b/src/ui/ui_exports.js @@ -22,6 +22,10 @@ export default class UiExports { visEditorTypes: [ 'ui/vis/editors/default/default', ], + embeddableHandlers: [ + 'plugins/kibana/visualize/embeddable/visualize_embeddable_handler_provider', + 'plugins/kibana/discover/embeddable/search_embeddable_handler_provider', + ], }; this.urlBasePath = urlBasePath; this.exportConsumer = _.memoize(this.exportConsumer); @@ -106,6 +110,7 @@ export default class UiExports { case 'visRequestHandlers': case 'visEditorTypes': case 'savedObjectTypes': + case 'embeddableHandlers': case 'fieldFormats': case 'fieldFormatEditors': case 'spyModes': From 2ae2d8eb70050eaca3c6240f7910dd9cac55e73c Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Fri, 2 Jun 2017 16:23:11 -0400 Subject: [PATCH 02/21] Remove dashboard knowledge of click and brush handlers for visualizations Move it to the VisualizeEmbeddableHandler. --- .../kibana/public/dashboard/dashboard.html | 6 ++---- .../kibana/public/dashboard/dashboard.js | 7 +------ .../kibana/public/dashboard/grid.js | 17 +++++------------ .../kibana/public/dashboard/panel/panel.js | 16 ++++------------ .../embeddable/visualize_embeddable_handler.js | 18 ++++++++++++------ .../visualize_embeddable_handler_provider.js | 4 ++-- 6 files changed, 26 insertions(+), 42 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.html b/src/core_plugins/kibana/public/dashboard/dashboard.html index b98003c727704..853ae8a463997 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.html +++ b/src/core_plugins/kibana/public/dashboard/dashboard.html @@ -117,8 +117,7 @@

on-panel-removed="onPanelRemoved" dashboard-view-mode="dashboardViewMode" panels="panels" - get-vis-click-handler="getFilterBarClickHandler" - get-vis-brush-handler="getBrushEvent" + app-state="appState" save-state="saveState" app-state="appState" toggle-expand="toggleExpandPanel" @@ -133,9 +132,8 @@

panel="expandedPanel" is-full-screen-mode="!chrome.getVisible()" is-expanded="true" + app-state="appState" dashboard-view-mode="dashboardViewMode" - get-vis-click-handler="getFilterBarClickHandler" - get-vis-brush-handler="getBrushEvent" save-state="saveState" app-state="appState" register-panel-index-pattern="registerPanelIndexPattern" diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.js b/src/core_plugins/kibana/public/dashboard/dashboard.js index 1911707510883..bcc65c4f4a6e0 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard.js @@ -18,8 +18,6 @@ import { DocTitleProvider } from 'ui/doc_title'; import { getTopNavConfig } from './top_nav/get_top_nav_config'; import { DashboardConstants, createDashboardEditUrl } from './dashboard_constants'; import { VisualizeConstants } from 'plugins/kibana/visualize/visualize_constants'; -import { UtilsBrushEventProvider } from 'ui/utils/brush_event'; -import { FilterBarClickHandlerProvider } from 'ui/filter_bar/filter_bar_click_handler'; import { DashboardState } from './dashboard_state'; import { notify } from 'ui/notify'; import { documentationLinks } from 'ui/documentation_links/documentation_links'; @@ -81,8 +79,6 @@ app.directive('dashboardApp', function ($injector) { const kbnUrl = $injector.get('kbnUrl'); const confirmModal = $injector.get('confirmModal'); const Private = $injector.get('Private'); - const brushEvent = Private(UtilsBrushEventProvider); - const filterBarClickHandler = Private(FilterBarClickHandlerProvider); return { restrict: 'E', @@ -99,6 +95,7 @@ app.directive('dashboardApp', function ($injector) { } const dashboardState = new DashboardState(dash, AppState, dashboardConfig); + $scope.appState = dashboardState.getAppState(); // The 'previouslyStored' check is so we only update the time filter on dashboard open, not during // normal cross app navigation. @@ -149,8 +146,6 @@ app.directive('dashboardApp', function ($injector) { $scope.appState = dashboardState.getAppState(); $scope.landingPageUrl = () => `#${DashboardConstants.LANDING_PAGE_PATH}`; - $scope.getBrushEvent = () => brushEvent(dashboardState.getAppState()); - $scope.getFilterBarClickHandler = () => filterBarClickHandler(dashboardState.getAppState()); $scope.hasExpandedPanel = () => $scope.expandedPanel !== null; $scope.getDashTitle = () => getDashboardTitle( dashboardState.getTitle(), diff --git a/src/core_plugins/kibana/public/dashboard/grid.js b/src/core_plugins/kibana/public/dashboard/grid.js index cb2b47a2d058c..f00122b62ba39 100644 --- a/src/core_plugins/kibana/public/dashboard/grid.js +++ b/src/core_plugins/kibana/public/dashboard/grid.js @@ -38,16 +38,6 @@ app.directive('dashboardGrid', function ($compile, Notifier) { * @type {Array} */ panels: '=', - /** - * Returns a click handler for a visualization. - * @type {function} - */ - getVisClickHandler: '=', - /** - * Returns a brush event handler for a visualization. - * @type {function} - */ - getVisBrushHandler: '=', /** * Call when changes should be propagated to the url and thus saved in state. * @type {function} @@ -60,6 +50,10 @@ app.directive('dashboardGrid', function ($compile, Notifier) { * @type {function} */ toggleExpand: '=', + /** + * @type {Object} + */ + appState: '=', }, link: function ($scope, $el) { const notify = new Notifier(); @@ -217,11 +211,10 @@ app.directive('dashboardGrid', function ($compile, Notifier) { is-full-screen-mode="isFullScreenMode" is-expanded="false" dashboard-view-mode="dashboardViewMode" - get-vis-click-handler="getVisClickHandler" - get-vis-brush-handler="getVisBrushHandler" save-state="saveState" app-state="appState" register-panel-index-pattern="registerPanelIndexPattern" + app-state="appState" toggle-expand="toggleExpand(${panel.panelIndex})" create-child-ui-state="createChildUiState"> `; diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index afe6c190efde8..3959673fba10d 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -66,18 +66,12 @@ uiModules */ isExpanded: '=', /** - * Returns a click handler for a visualization. - * @type {function} - */ - getVisClickHandler: '=', - /** - * Returns a brush event handler for a visualization. + * Call when changes should be propagated to the url and thus saved in state. * @type {function} */ - getVisBrushHandler: '=', + saveState: '=', /** - * Call when changes should be propagated to the url and thus saved in state. - * @type {function} + * @type {Object} */ saveState: '=', appState: '=', @@ -94,11 +88,9 @@ uiModules return $scope.dashboardViewMode === DashboardViewMode.VIEW || $scope.isFullScreenMode; }; - // TODO: vis actions should be generalized for use by all panel renderers, e.g. updateFilters, updateTimeRange. const containerAPI = { - getVisClickHandler: $scope.getVisClickHandler, - getVisBrushHandler: $scope.getVisBrushHandler, savePanelState, + getAppState: () => $scope.appState, getIsViewOnlyMode: $scope.isViewOnlyMode, createChildUiState: $scope.createChildUiState, registerPanelIndexPattern: $scope.registerPanelIndexPattern, diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index 9e841834b9679..443fecaba1d8d 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -2,14 +2,19 @@ import angular from 'angular'; import visualizationTemplate from './visualize_template.html'; import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; +import { UtilsBrushEventProvider as utilsBrushEventProvider } from 'ui/utils/brush_event'; +import { FilterBarClickHandlerProvider as filterBarClickHandlerProvider } from 'ui/filter_bar/filter_bar_click_handler'; export class VisualizeEmbeddableHandler { - constructor($compile, $rootScope, visualizeLoader) { + constructor($compile, $rootScope, visualizeLoader, timefilter, Notifier) { this.$compile = $compile; this.visualizeLoader = visualizeLoader; this.$rootScope = $rootScope; this.name = 'visualization'; this.title = 'Visualizations'; + + this.brushEvent = utilsBrushEventProvider(timefilter); + this.filterBarClickHandler = filterBarClickHandlerProvider(Notifier); } getEditPath(panel) { @@ -20,7 +25,7 @@ export class VisualizeEmbeddableHandler { return this.visualizeLoader.get(panel.id).then(savedObject => savedObject.title); } - render(domNode, panel, actions) { + render(domNode, panel, container) { return this.visualizeLoader.get(panel.id).then((savedObject) => { const visualizeScope = this.$rootScope.$new(); visualizeScope.editUrl = this.getEditPath(panel); @@ -28,12 +33,13 @@ export class VisualizeEmbeddableHandler { visualizeScope.panel = panel; const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - visualizeScope.uiState = actions.createChildUiState(getPersistedStateId(panel), uiState); + visualizeScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); visualizeScope.savedObj.vis.setUiState(uiState); - visualizeScope.savedObj.vis.listeners.click = actions.getVisClickHandler(); - visualizeScope.savedObj.vis.listeners.brush = actions.getVisBrushHandler(); - visualizeScope.isFullScreenMode = actions.getIsViewOnlyMode(); + + visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); + visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); + visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); const rootNode = angular.element(domNode); diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js index 6abab61d6c053..8cf235fbcac2d 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js @@ -2,8 +2,8 @@ import { VisualizeEmbeddableHandler } from './visualize_embeddable_handler'; import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; export function visualizeEmbeddableHandlerProvider(Private) { - const VisualizeEmbeddableHandlerProvider = ($compile, $rootScope, savedVisualizations) => { - return new VisualizeEmbeddableHandler($compile, $rootScope, savedVisualizations); + const VisualizeEmbeddableHandlerProvider = ($compile, $rootScope, savedVisualizations, timefilter, Notifier) => { + return new VisualizeEmbeddableHandler($compile, $rootScope, savedVisualizations, timefilter, Notifier); }; return Private(VisualizeEmbeddableHandlerProvider); } From 01ba52509d05b0531990fe6b0f63e10589dd3698 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Mon, 5 Jun 2017 11:33:56 -0400 Subject: [PATCH 03/21] merge master with manual changes --- .../embeddable/search_embeddable_handler.js | 15 ++++++++------- .../embeddable/visualize_embeddable_handler.js | 2 ++ 2 files changed, 10 insertions(+), 7 deletions(-) diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index 65db615b4c908..200b5592b5f00 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -22,44 +22,45 @@ export class SearchEmbeddableHandler { return this.searchLoader.get(panel.id).then(savedObject => savedObject.title); } - render(domNode, panel, actions) { + render(domNode, panel, container) { return this.searchLoader.get(panel.id).then((savedObject) => { const editUrl = this.searchLoader.urlFor(panel.id); const searchScope = this.$rootScope.$new(); searchScope.editUrl = editUrl; searchScope.savedObj = savedObject; searchScope.panel = panel; + container.registerPanelIndexPattern(panel.panelIndex, savedObject.searchSource.get('index')); // This causes changes to a saved search to be hidden, but also allows // the user to locally modify and save changes to a saved search only in a dashboard. // See https://github.com/elastic/kibana/issues/9523 for more details. - actions.savePanelState({ + container.savePanelState({ columns: searchScope.panel.columns || searchScope.savedObj.columns, sort: searchScope.panel.sort || searchScope.savedObj.sort }); const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - searchScope.uiState = actions.createChildUiState(getPersistedStateId(panel), uiState); + searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); searchScope.setSortOrder = function setSortOrder(columnName, direction) { - actions.savePanelState({ sort: [columnName, direction] }); + container.savePanelState({ sort: [columnName, direction] }); }; searchScope.addColumn = function addColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.addColumn(searchScope.panel.columns, columnName); - actions.savePanelState({}); // sync to sharing url + container.savePanelState({}); // sync to sharing url }; searchScope.removeColumn = function removeColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.removeColumn(searchScope.panel.columns, columnName); - actions.savePanelState({}); // sync to sharing url + container.savePanelState({}); // sync to sharing url }; searchScope.moveColumn = function moveColumn(columnName, newIndex) { columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); - actions.savePanelState({}); // sync to sharing url + container.savePanelState({}); // sync to sharing url }; searchScope.filter = function (field, value, operator) { diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index 443fecaba1d8d..b967992b07dc6 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -41,6 +41,8 @@ export class VisualizeEmbeddableHandler { visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); + container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); + const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); const rootNode = angular.element(domNode); rootNode.append(visualizationInstance); From 3de0f34fc0c9a43e07666a50791fd5ed30d507c6 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 15 Jun 2017 12:43:35 -0400 Subject: [PATCH 04/21] No need to use lodash --- src/core_plugins/kibana/public/dashboard/panel/panel.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 3959673fba10d..225481efedc00 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -1,4 +1,3 @@ -import _ from 'lodash'; import 'ui/visualize'; import 'ui/doc_table'; import 'plugins/kibana/visualize/saved_visualizations'; @@ -112,7 +111,7 @@ uiModules const type = $scope.panel.type; const id = $scope.panel.id; - const service = _.find(services, { type: type }); + const service = services.find(service => service.type === type); if (!service) return; $scope.editUrl = '#management/kibana/objects/' + service.name + '/' + id + '?notFound=' + error.savedObjectType; From d815bfd06f700bdbd9b64a06cefabcf25227894f Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 27 Jun 2017 14:42:11 -0400 Subject: [PATCH 05/21] Add EmbeddableHandler base class --- .../kibana/public/dashboard/panel/panel.js | 10 +++--- .../embeddable/search_embeddable_handler.js | 15 ++++---- .../kibana/public/discover/index.js | 1 + .../visualize_embeddable_handler.js | 13 +++---- .../public/embeddable/embeddable_handler.js | 35 +++++++++++++++++++ .../embeddable_handlers_registry.js} | 6 ++-- src/ui/public/embeddable/index.js | 2 ++ 7 files changed, 63 insertions(+), 19 deletions(-) create mode 100644 src/ui/public/embeddable/embeddable_handler.js rename src/ui/public/{registry/embeddable_handlers.js => embeddable/embeddable_handlers_registry.js} (56%) create mode 100644 src/ui/public/embeddable/index.js diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 225481efedc00..6ae82ee2d4fbc 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -6,7 +6,7 @@ import { uiModules } from 'ui/modules'; import panelTemplate from 'plugins/kibana/dashboard/panel/panel.html'; import { savedObjectManagementRegistry } from 'plugins/kibana/management/saved_object_registry'; import { DashboardViewMode } from '../dashboard_view_mode'; -import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; +import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; uiModules .get('app/dashboard') @@ -95,6 +95,7 @@ uiModules registerPanelIndexPattern: $scope.registerPanelIndexPattern, }; + const panelId = $scope.panel.id; const handleError = (error) => { $scope.error = error.message; @@ -110,11 +111,10 @@ uiModules if (objectItselfDeleted) return; const type = $scope.panel.type; - const id = $scope.panel.id; const service = services.find(service => service.type === type); if (!service) return; - $scope.editUrl = '#management/kibana/objects/' + service.name + '/' + id + '?notFound=' + error.savedObjectType; + $scope.editUrl = '#management/kibana/objects/' + service.name + '/' + panelId + '?notFound=' + error.savedObjectType; }; const embeddableHandlers = Private(EmbeddableHandlersRegistryProvider); @@ -123,8 +123,8 @@ uiModules handleError(`No embeddable handler for panel type ${$scope.panel.type} was found.`); return; } - $scope.editUrl = embeddableHandler.getEditPath($scope.panel); - embeddableHandler.getTitleFor($scope.panel).then(title => { + $scope.editUrl = embeddableHandler.getEditPath(panelId); + embeddableHandler.getTitleFor(panelId).then(title => { $scope.title = title; }); $scope.loadedPanel = diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index 200b5592b5f00..61ad9e85f0495 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -3,23 +3,26 @@ import angular from 'angular'; import * as columnActions from 'ui/doc_table/actions/columns'; import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; import { FilterManagerProvider } from 'ui/filter_manager'; +import { EmbeddableHandler } from 'ui/embeddable'; + + +export class SearchEmbeddableHandler extends EmbeddableHandler { -export class SearchEmbeddableHandler { constructor($compile, $rootScope, searchLoader, Private) { + super(); this.$compile = $compile; this.searchLoader = searchLoader; this.filterManager = Private(FilterManagerProvider); this.$rootScope = $rootScope; this.name = 'search'; - this.title = 'Saved Searches'; } - getEditPath(panel) { - return this.searchLoader.urlFor(panel.id); + getEditPath(panelId) { + return this.searchLoader.urlFor(panelId); } - getTitleFor(panel) { - return this.searchLoader.get(panel.id).then(savedObject => savedObject.title); + getTitleFor(panelId) { + return this.searchLoader.get(panelId).then(savedObject => savedObject.title); } render(domNode, panel, container) { diff --git a/src/core_plugins/kibana/public/discover/index.js b/src/core_plugins/kibana/public/discover/index.js index 29d7bbd5c311f..bdab201dedefa 100644 --- a/src/core_plugins/kibana/public/discover/index.js +++ b/src/core_plugins/kibana/public/discover/index.js @@ -6,3 +6,4 @@ import 'plugins/kibana/discover/components/field_chooser/field_chooser'; import 'plugins/kibana/discover/controllers/discover'; import 'plugins/kibana/discover/styles/main.less'; import 'ui/doc_table/components/table_row'; + diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index b967992b07dc6..d6f3a594ead6f 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -4,25 +4,26 @@ import visualizationTemplate from './visualize_template.html'; import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state'; import { UtilsBrushEventProvider as utilsBrushEventProvider } from 'ui/utils/brush_event'; import { FilterBarClickHandlerProvider as filterBarClickHandlerProvider } from 'ui/filter_bar/filter_bar_click_handler'; +import { EmbeddableHandler } from 'ui/embeddable'; -export class VisualizeEmbeddableHandler { +export class VisualizeEmbeddableHandler extends EmbeddableHandler { constructor($compile, $rootScope, visualizeLoader, timefilter, Notifier) { + super(); this.$compile = $compile; this.visualizeLoader = visualizeLoader; this.$rootScope = $rootScope; this.name = 'visualization'; - this.title = 'Visualizations'; this.brushEvent = utilsBrushEventProvider(timefilter); this.filterBarClickHandler = filterBarClickHandlerProvider(Notifier); } - getEditPath(panel) { - return this.visualizeLoader.urlFor(panel.id); + getEditPath(panelId) { + return this.visualizeLoader.urlFor(panelId); } - getTitleFor(panel) { - return this.visualizeLoader.get(panel.id).then(savedObject => savedObject.title); + getTitleFor(panelId) { + return this.visualizeLoader.get(panelId).then(savedObject => savedObject.title); } render(domNode, panel, container) { diff --git a/src/ui/public/embeddable/embeddable_handler.js b/src/ui/public/embeddable/embeddable_handler.js new file mode 100644 index 0000000000000..abb75cea21af1 --- /dev/null +++ b/src/ui/public/embeddable/embeddable_handler.js @@ -0,0 +1,35 @@ +/** + * The EmbeddableHandler defines how to render and embed any object into the Dashboard, or some other + * container that supports EmbeddableHandlers. + */ +export class EmbeddableHandler { + /** + * @param {string} panelId - the id of the panel to grad the title for. + * @return {string} a path that dictates where the user will be navigated to when they click the edit icon. + */ + getEditPath(/* panelId */) { + throw new Error('Must implement getEditPath.'); + } + + /** + * @param {string} panelId - the id of the panel to grad the title for. + * @return {string} - the title to display for this particular panel. + */ + getTitleFor(/* panelId */) { + throw new Error('Must implement getTitleFor.'); + } + + /** + * @param {Element} domNode - the dom node to mount the rendered embeddable on + * @param {Object} panel - a panel object which container information about the panel. Can also be modified to + * store per panel information. + * @property {string} panel.id - an id to specify the object that this panel contains. + * @property {string} panel.type - specifies which EmbeddableHandler should render this panel. + * @property {number} panel.panelIndex - a unique identifier for this panel in a specific container. Different + * from panel.id because you can have the same object rendered multiple times in two different panels. + * @param {Object} container - Contains functions to communicate with the container. + */ + render(/* domNode, panel, container */) { + throw new Error('Must implement render.'); + } +} diff --git a/src/ui/public/registry/embeddable_handlers.js b/src/ui/public/embeddable/embeddable_handlers_registry.js similarity index 56% rename from src/ui/public/registry/embeddable_handlers.js rename to src/ui/public/embeddable/embeddable_handlers_registry.js index 0e093928c35e9..408288c842173 100644 --- a/src/ui/public/registry/embeddable_handlers.js +++ b/src/ui/public/embeddable/embeddable_handlers_registry.js @@ -1,7 +1,9 @@ import { uiRegistry } from 'ui/registry/_registry'; +/** + * Registry of functions (EmbeddableHandlerProviders) which return an EmbeddableHandler. + */ export const EmbeddableHandlersRegistryProvider = uiRegistry({ name: 'embeddableHandlers', - index: ['name'], - order: ['title'] + index: ['name'] }); diff --git a/src/ui/public/embeddable/index.js b/src/ui/public/embeddable/index.js new file mode 100644 index 0000000000000..ac54aadb57218 --- /dev/null +++ b/src/ui/public/embeddable/index.js @@ -0,0 +1,2 @@ +export { EmbeddableHandler } from './embeddable_handler'; +export { EmbeddableHandlersRegistryProvider } from './embeddable_handlers_registry'; From 505365c0793ed67fb60f579b71a5c3eadef22e33 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 18 Jul 2017 13:31:33 -0400 Subject: [PATCH 06/21] Use correct path to embeddable_handlers_registry --- .../discover/embeddable/search_embeddable_handler_provider.js | 2 +- .../embeddable/visualize_embeddable_handler_provider.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js index f11c3283604d1..44ea067cad44f 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js @@ -1,5 +1,5 @@ import { SearchEmbeddableHandler } from './search_embeddable_handler'; -import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; +import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; export function searchEmbeddableHandlerProvider(Private) { const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches, Private) => { diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js index 8cf235fbcac2d..babdb3ea23000 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js @@ -1,5 +1,5 @@ import { VisualizeEmbeddableHandler } from './visualize_embeddable_handler'; -import { EmbeddableHandlersRegistryProvider } from 'ui/registry/embeddable_handlers'; +import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; export function visualizeEmbeddableHandlerProvider(Private) { const VisualizeEmbeddableHandlerProvider = ($compile, $rootScope, savedVisualizations, timefilter, Notifier) => { From f2c75f578ed0a1584d319ed3a7f965a590f8a687 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 18 Jul 2017 14:29:21 -0400 Subject: [PATCH 07/21] clean up --- src/core_plugins/kibana/public/dashboard/panel/panel.js | 3 ++- src/core_plugins/kibana/public/discover/index.js | 1 - 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 6ae82ee2d4fbc..94111829cd4f8 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -72,7 +72,6 @@ uiModules /** * @type {Object} */ - saveState: '=', appState: '=', }, link: function ($scope, element) { @@ -96,6 +95,8 @@ uiModules }; const panelId = $scope.panel.id; + + // TODO: This function contains too much internal panel knowledge. Logic should be pushed to embeddable handlers. const handleError = (error) => { $scope.error = error.message; diff --git a/src/core_plugins/kibana/public/discover/index.js b/src/core_plugins/kibana/public/discover/index.js index bdab201dedefa..29d7bbd5c311f 100644 --- a/src/core_plugins/kibana/public/discover/index.js +++ b/src/core_plugins/kibana/public/discover/index.js @@ -6,4 +6,3 @@ import 'plugins/kibana/discover/components/field_chooser/field_chooser'; import 'plugins/kibana/discover/controllers/discover'; import 'plugins/kibana/discover/styles/main.less'; import 'ui/doc_table/components/table_row'; - From 1acbffe880b681ef908447a332ac0d2d9f9de008 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Fri, 21 Jul 2017 09:49:09 -0400 Subject: [PATCH 08/21] Set visualize scope uiState that is of type PersistedState, otherwise it won't actually be set. --- .../public/visualize/embeddable/visualize_embeddable_handler.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index d6f3a594ead6f..17f5f409b39d8 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -36,7 +36,7 @@ export class VisualizeEmbeddableHandler extends EmbeddableHandler { const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; visualizeScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); - visualizeScope.savedObj.vis.setUiState(uiState); + visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); From 7f775b43ee3c0ee848685c7243c609b47b61571f Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Mon, 24 Jul 2017 11:41:12 -0400 Subject: [PATCH 09/21] add retry to loading saved search data --- test/functional/apps/discover/_discover.js | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/test/functional/apps/discover/_discover.js b/test/functional/apps/discover/_discover.js index aa70c7fe2f246..39cc89deae503 100644 --- a/test/functional/apps/discover/_discover.js +++ b/test/functional/apps/discover/_discover.js @@ -229,17 +229,19 @@ export default function ({ getService, getPageObjects }) { }); }); - describe('data-shared-item', function () { + describe('data-shared-item', async function () { it('should have correct data-shared-item title and description', async () => { const expected = { title: 'A Saved Search', description: 'A Saved Search Description' }; - await PageObjects.discover.loadSavedSearch(expected.title); - const { title, description } = await PageObjects.common.getSharedItemTitleAndDescription(); - expect(title).to.eql(expected.title); - expect(description).to.eql(expected.description); + await retry.try(async () => { + await PageObjects.discover.loadSavedSearch(expected.title); + const { title, description } = await PageObjects.common.getSharedItemTitleAndDescription(); + expect(title).to.eql(expected.title); + expect(description).to.eql(expected.description); + }); }); }); }); From 0f57ba5dee7d290cff3d4d865c33df002cd6f36d Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 27 Jul 2017 10:09:53 -0400 Subject: [PATCH 10/21] Fix handleError param and remove unnecessary private param --- src/core_plugins/kibana/public/dashboard/panel/panel.js | 2 +- .../discover/embeddable/search_embeddable_handler_provider.js | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 23a68722a1ee9..e2bd417f584d9 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -127,7 +127,7 @@ uiModules const embeddableHandlers = Private(EmbeddableHandlersRegistryProvider); const embeddableHandler = embeddableHandlers.byName[$scope.panel.type]; if (!embeddableHandler) { - handleError(`No embeddable handler for panel type ${$scope.panel.type} was found.`); + handleError(new Error(`No embeddable handler for panel type ${$scope.panel.type} was found.`)); return; } $scope.editUrl = embeddableHandler.getEditPath(panelId); diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js index 44ea067cad44f..9298f6527374e 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js @@ -2,8 +2,8 @@ import { SearchEmbeddableHandler } from './search_embeddable_handler'; import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; export function searchEmbeddableHandlerProvider(Private) { - const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches, Private) => { - return new SearchEmbeddableHandler($compile, $rootScope, savedSearches, Private); + const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches) => { + return new SearchEmbeddableHandler($compile, $rootScope, savedSearches); }; return Private(SearchEmbeddableHandlerProvider); } From cf541fb8fe6634a9c2f5ee84cbde2f878e57d6a8 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 27 Jul 2017 10:38:02 -0400 Subject: [PATCH 11/21] Rename savePanelState updatePanel and return the new object rather than mutating the original --- .../kibana/public/dashboard/panel/panel.js | 10 +++++++--- .../discover/embeddable/search_embeddable_handler.js | 10 +++++----- 2 files changed, 12 insertions(+), 8 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index e2bd417f584d9..1773e24d64b54 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -82,9 +82,13 @@ uiModules link: function ($scope, element) { if (!$scope.panel.id || !$scope.panel.type) return; - const savePanelState = (panel) => { - $scope.panel = Object.assign($scope.panel, panel); + const updatePanel = (panelAttributes) => { + $scope.panel = { + ...$scope.panel, + ...panelAttributes + }; $scope.saveState(); + return $scope.panel; }; $scope.isViewOnlyMode = () => { @@ -93,7 +97,7 @@ uiModules const containerAPI = { onFilter: $scope.onFilter, - savePanelState, + updatePanel, getAppState: () => $scope.appState, getIsViewOnlyMode: $scope.isViewOnlyMode, createChildUiState: $scope.createChildUiState, diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index 5f9cf2c2c8d9d..36d2d9f831642 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -35,7 +35,7 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { // This causes changes to a saved search to be hidden, but also allows // the user to locally modify and save changes to a saved search only in a dashboard. // See https://github.com/elastic/kibana/issues/9523 for more details. - container.savePanelState({ + searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns || searchScope.savedObj.columns, sort: searchScope.panel.sort || searchScope.savedObj.sort }); @@ -44,24 +44,24 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); searchScope.setSortOrder = function setSortOrder(columnName, direction) { - container.savePanelState({ sort: [columnName, direction] }); + searchScope.panel = container.updatePanel({ sort: [columnName, direction] }); }; searchScope.addColumn = function addColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.addColumn(searchScope.panel.columns, columnName); - container.savePanelState({}); // sync to sharing url + searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); }; searchScope.removeColumn = function removeColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.removeColumn(searchScope.panel.columns, columnName); - container.savePanelState({}); // sync to sharing url + searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); }; searchScope.moveColumn = function moveColumn(columnName, newIndex) { columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); - container.savePanelState({}); // sync to sharing url + searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); }; searchScope.filter = function (field, value, operator) { From 0a2545af81d70253e61854ce98f99a634cbaebf6 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 1 Aug 2017 12:39:36 -0400 Subject: [PATCH 12/21] Make ContainerAPI a base class and move the dashboard specific functionality into a new class --- .../kibana/public/dashboard/dashboard.html | 11 +--- .../kibana/public/dashboard/dashboard.js | 21 +------ .../dashboard/dashboard_container_api.js | 38 ++++++++++++ .../kibana/public/dashboard/grid.js | 28 ++------- .../kibana/public/dashboard/panel/panel.js | 45 +------------- .../embeddable/search_embeddable_handler.js | 10 ++-- src/ui/public/embeddable/container_api.js | 58 +++++++++++++++++++ src/ui/public/embeddable/index.js | 1 + 8 files changed, 113 insertions(+), 99 deletions(-) create mode 100644 src/core_plugins/kibana/public/dashboard/dashboard_container_api.js create mode 100644 src/ui/public/embeddable/container_api.js diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.html b/src/core_plugins/kibana/public/dashboard/dashboard.html index 542433ad6b143..d80074278d2c5 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.html +++ b/src/core_plugins/kibana/public/dashboard/dashboard.html @@ -80,15 +80,11 @@

on-panel-removed="onPanelRemoved" dashboard-view-mode="dashboardViewMode" panels="panels" - app-state="appState" save-state="saveState" - app-state="appState" - toggle-expand="toggleExpandPanel" - create-child-ui-state="createChildUiState" toggle-expand="toggleExpandPanel" register-panel-index-pattern="registerPanelIndexPattern" data-shared-items-count="{{panels.length}}" - on-filter="filter" + container-api="containerApi" > panel="expandedPanel" is-full-screen-mode="!chrome.getVisible()" is-expanded="true" - app-state="appState" dashboard-view-mode="dashboardViewMode" - save-state="saveState" - app-state="appState" - register-panel-index-pattern="registerPanelIndexPattern" - create-child-ui-state="createChildUiState" toggle-expand="toggleExpandPanel(expandedPanel.panelIndex)" > diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.js b/src/core_plugins/kibana/public/dashboard/dashboard.js index c0c0cbcea5535..585421ba18dfb 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard.js @@ -26,6 +26,7 @@ import { showCloneModal } from './top_nav/show_clone_modal'; import { migrateLegacyQuery } from 'ui/utils/migrateLegacyQuery'; import { QueryManagerProvider } from 'ui/query_manager'; import { ESC_KEY_CODE } from 'ui_framework/services'; +import { DashboardContainerAPI } from './dashboard_container_api'; const app = uiModules.get('app/dashboard', [ 'elasticsearch', @@ -101,6 +102,7 @@ app.directive('dashboardApp', function ($injector) { const dashboardState = new DashboardState(dash, AppState, dashboardConfig); $scope.appState = dashboardState.getAppState(); const queryManager = Private(QueryManagerProvider)(dashboardState.getAppState()); + $scope.containerApi = new DashboardContainerAPI(dashboardState, queryManager); // The 'previouslyStored' check is so we only update the time filter on dashboard open, not during // normal cross app navigation. @@ -120,6 +122,7 @@ app.directive('dashboardApp', function ($injector) { }; $scope.panels = dashboardState.getPanels(); $scope.fullScreenMode = dashboardState.getFullScreenMode(); + $scope.indexPatterns = dashboardState.getPanelIndexPatterns(); }; // Part of the exposed plugin API - do not remove without careful consideration. @@ -151,7 +154,6 @@ app.directive('dashboardApp', function ($injector) { $scope.timefilter = timefilter; $scope.expandedPanel = null; $scope.dashboardViewMode = dashboardState.getViewMode(); - $scope.appState = dashboardState.getAppState(); $scope.landingPageUrl = () => `#${DashboardConstants.LANDING_PAGE_PATH}`; $scope.hasExpandedPanel = () => $scope.expandedPanel !== null; @@ -206,17 +208,6 @@ app.directive('dashboardApp', function ($injector) { notify.info(`Search successfully added to your dashboard`); }; - /** - * Creates a child ui state for the panel. It's passed the ui state to use, but needs to - * be generated from the parent (why, I don't know yet). - * @param path {String} - the unique path for this ui state. - * @param uiState {Object} - the uiState for the child. - * @returns {Object} - */ - $scope.createChildUiState = function createChildUiState(path, uiState) { - return dashboardState.uiState.createChild(path, uiState, true); - }; - $scope.$watch('model.darkTheme', () => { dashboardState.setDarkTheme($scope.model.darkTheme); updateTheme(); @@ -236,12 +227,6 @@ app.directive('dashboardApp', function ($injector) { $scope.indexPatterns = dashboardState.getPanelIndexPatterns(); }; - $scope.filter = function (field, value, operator, index) { - queryManager.add(field, value, operator, index); - updateState(); - }; - - $scope.$watch('model.query', (newQuery) => { $scope.model.query = migrateLegacyQuery(newQuery); dashboardState.applyFilters($scope.model.query, filterBar.getFilters()); diff --git a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js new file mode 100644 index 0000000000000..f52cba0bb41c8 --- /dev/null +++ b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js @@ -0,0 +1,38 @@ +import { ContainerAPI } from 'ui/embeddable'; + +export class DashboardContainerAPI extends ContainerAPI { + constructor(dashboardState, queryManager) { + super(); + this.dashboardState = dashboardState; + this.queryManager = queryManager; + } + + onFilter(field, value, operator, index) { + this.queryManager.add(field, value, operator, index); + } + + updatePanel(panelIndex, panelAttributes) { + const panelToUpdate = this.dashboardState.getPanels().find((panel) => panel.panelIndex === panelIndex); + Object.assign(panelToUpdate, panelAttributes); + this.dashboardState.saveState(); + return panelToUpdate; + } + + getAppState() { + return this.dashboardState.appState; + } + + getIsViewOnlyMode() { + return this.dashboardState.getIsViewMode(); + } + + createChildUiState(path, uiState) { + return this.dashboardState.uiState.createChild(path, uiState, true); + } + + registerPanelIndexPattern(panelIndex, pattern) { + this.dashboardState.registerPanelIndexPatternMap(panelIndex, pattern); + this.dashboardState.saveState(); + } + +} diff --git a/src/core_plugins/kibana/public/dashboard/grid.js b/src/core_plugins/kibana/public/dashboard/grid.js index ab9e12d2ba668..54fd3fab1ab79 100644 --- a/src/core_plugins/kibana/public/dashboard/grid.js +++ b/src/core_plugins/kibana/public/dashboard/grid.js @@ -18,17 +18,6 @@ app.directive('dashboardGrid', function ($compile, Notifier) { * @type {DashboardViewMode} */ dashboardViewMode: '=', - /** - * Used to create a child persisted state for the panel from parent state. - * @type {function} - Returns a {PersistedState} child uiState for this scope. - */ - createChildUiState: '=', - /** - * Registers an index pattern with the dashboard app used by each panel. The index patterns are used by the - * filter bar for generating field suggestions. - * @type {function(IndexPattern)} - */ - registerPanelIndexPattern: '=', /** * Trigger after a panel has been removed from the grid. */ @@ -50,14 +39,9 @@ app.directive('dashboardGrid', function ($compile, Notifier) { */ toggleExpand: '=', /** - * @type {Object} - */ - appState: '=', - /** - * Called when a filter action has been triggered by a panel - * @type {function} + * @type {DashboardContainerApi} */ - onFilter: '=', + containerApi: '=', }, link: function ($scope, $el) { const notify = new Notifier(); @@ -215,13 +199,9 @@ app.directive('dashboardGrid', function ($compile, Notifier) { is-full-screen-mode="isFullScreenMode" is-expanded="false" dashboard-view-mode="dashboardViewMode" - save-state="saveState" - app-state="appState" - register-panel-index-pattern="registerPanelIndexPattern" - app-state="appState" + container-api="containerApi" toggle-expand="toggleExpand(${panel.panelIndex})" - create-child-ui-state="createChildUiState" - on-filter="onFilter"> + > `; const panelElement = $compile(panelHtml)($scope); panelElementMapping[panel.panelIndex] = panelElement; diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 1773e24d64b54..b7c234cc22ca6 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -33,17 +33,6 @@ uiModules * @type {boolean} */ isFullScreenMode: '=', - /** - * Used to create a child persisted state for the panel from parent state. - * @type {function} - Returns a {PersistedState} child uiState for this scope. - */ - createChildUiState: '=', - /** - * Registers an index pattern with the dashboard app used by this panel. Used by the filter bar for - * generating field suggestions. - * @type {function(IndexPattern)} - */ - registerPanelIndexPattern: '=', /** * Contains information about this panel. * @type {PanelState} @@ -65,45 +54,17 @@ uiModules */ isExpanded: '=', /** - * Call when changes should be propagated to the url and thus saved in state. - * @type {function} - */ - saveState: '=', - /** - * Called when a filter action has been triggered - * @type {function} - */ - onFilter: '=', - /** - * @type {Object} + * @type {DashboardContainerApi} */ - appState: '=', + containerApi: '=' }, link: function ($scope, element) { if (!$scope.panel.id || !$scope.panel.type) return; - const updatePanel = (panelAttributes) => { - $scope.panel = { - ...$scope.panel, - ...panelAttributes - }; - $scope.saveState(); - return $scope.panel; - }; - $scope.isViewOnlyMode = () => { return $scope.dashboardViewMode === DashboardViewMode.VIEW || $scope.isFullScreenMode; }; - const containerAPI = { - onFilter: $scope.onFilter, - updatePanel, - getAppState: () => $scope.appState, - getIsViewOnlyMode: $scope.isViewOnlyMode, - createChildUiState: $scope.createChildUiState, - registerPanelIndexPattern: $scope.registerPanelIndexPattern, - }; - const panelId = $scope.panel.id; // TODO: This function contains too much internal panel knowledge. Logic should be pushed to embeddable handlers. @@ -139,7 +100,7 @@ uiModules $scope.title = title; }); $scope.loadedPanel = - embeddableHandler.render(element.find('#embeddedPanel').get(0), $scope.panel, containerAPI).catch(handleError); + embeddableHandler.render(element.find('#embeddedPanel').get(0), $scope.panel, $scope.containerApi).catch(handleError); } }; }); diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index 36d2d9f831642..b0ad8bee0857a 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -35,7 +35,7 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { // This causes changes to a saved search to be hidden, but also allows // the user to locally modify and save changes to a saved search only in a dashboard. // See https://github.com/elastic/kibana/issues/9523 for more details. - searchScope.panel = container.updatePanel({ + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns || searchScope.savedObj.columns, sort: searchScope.panel.sort || searchScope.savedObj.sort }); @@ -44,24 +44,24 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); searchScope.setSortOrder = function setSortOrder(columnName, direction) { - searchScope.panel = container.updatePanel({ sort: [columnName, direction] }); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); }; searchScope.addColumn = function addColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.addColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); }; searchScope.removeColumn = function removeColumn(columnName) { savedObject.searchSource.get('index').popularizeField(columnName, 1); columnActions.removeColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); }; searchScope.moveColumn = function moveColumn(columnName, newIndex) { columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); - searchScope.panel = container.updatePanel({ columns: searchScope.panel.columns }); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); }; searchScope.filter = function (field, value, operator) { diff --git a/src/ui/public/embeddable/container_api.js b/src/ui/public/embeddable/container_api.js new file mode 100644 index 0000000000000..522fd15a5c583 --- /dev/null +++ b/src/ui/public/embeddable/container_api.js @@ -0,0 +1,58 @@ +/** + * The ContainerAPI is an interface for embeddable objects to interact with the container they are embedded within. + */ +export class ContainerAPI { + /** + * Available so the embeddable object can trigger a filter action. + * @param field + * @param value + * @param operator + * @param index + */ + onFilter(/*field, value, operator, index */) { + throw new Error('Must implement onFilter.'); + } + + /** + * @return {AppState} + */ + getAppState() { + throw new Error('Must implement getAppState.'); + } + + /** + * @return {boolean} If the container is marked as being in view only mode or not. + */ + getIsViewOnlyMode() { + throw new Error('Must implement getIsViewOnlyMode.'); + } + + /** + * Creates a child ui state for the panel. It's passed the ui state to use, but needs to + * be generated from the parent. + * @param path {String} - the unique path for this ui state. + * @param uiState {Object} - the uiState for the child. + * @returns {Object} + */ + createChildUiState(/* path, uiState */) { + throw new Error('Must implement createChildUiState.'); + } + + /** + * Call this to tell the container that this panel uses a particular index pattern. + * @param {string} panelIndex - a unique id that identifies the panel to update. + * @param {string} indexPattern - an index pattern the panel uses + */ + registerPanelIndexPattern(/* panelIndex, indexPattern */) { + throw new Error('Must implement registerPanelIndexPattern.'); + } + + /** + * @param {string} panelIndex - a unique id that identifies the panel to update. + * @param {Object} panelAttributes - the new panel attributes that will be applied to the panel. + * @return {Object} - the updated panel. + */ + updatePanel(/*paneIndex, panelAttributes */) { + throw new Error('Must implement updatePanel.'); + } +} diff --git a/src/ui/public/embeddable/index.js b/src/ui/public/embeddable/index.js index ac54aadb57218..bdd73ce99a110 100644 --- a/src/ui/public/embeddable/index.js +++ b/src/ui/public/embeddable/index.js @@ -1,2 +1,3 @@ export { EmbeddableHandler } from './embeddable_handler'; export { EmbeddableHandlersRegistryProvider } from './embeddable_handlers_registry'; +export { ContainerAPI } from './container_api'; From 3c9fe4e22b41e5ebc8486bbbce4984afa9bd4565 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 1 Aug 2017 12:54:29 -0400 Subject: [PATCH 13/21] Make api's async and clean up documentation --- .../public/dashboard/__tests__/panel.js | 4 +- .../kibana/public/dashboard/panel/panel.js | 6 +- .../embeddable/search_embeddable_handler.js | 92 +++++++++---------- .../visualize_embeddable_handler.js | 46 +++++----- .../public/embeddable/embeddable_handler.js | 21 +++-- 5 files changed, 86 insertions(+), 83 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js index c25b8a4de0d2b..da69aec871b8e 100644 --- a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js +++ b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js @@ -59,7 +59,7 @@ describe('dashboard panel', function () { it('should not visualize the visualization if it does not exist', function () { init({ found: false }); - return $scope.loadedPanel.then(() => { + return $scope.renderPromise.then(() => { expect($scope.error).to.be('Could not locate that visualization (id: foo1)'); parentScope.$digest(); const content = $el.find('.panel-content'); @@ -69,7 +69,7 @@ describe('dashboard panel', function () { it('should try to visualize the visualization if found', function () { init({ id: 'foo1', type: 'visualization', _version: 2, attributes: {} }); - return $scope.loadedPanel.then(() => { + return $scope.renderPromise.then(() => { expect($scope.error).not.to.be.ok(); parentScope.$digest(); const content = $el.find('.panel-content'); diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index b7c234cc22ca6..5a296003a57ee 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -95,11 +95,13 @@ uiModules handleError(new Error(`No embeddable handler for panel type ${$scope.panel.type} was found.`)); return; } - $scope.editUrl = embeddableHandler.getEditPath(panelId); + embeddableHandler.getEditPath(panelId).then(path => { + $scope.editUrl = path; + }); embeddableHandler.getTitleFor(panelId).then(title => { $scope.title = title; }); - $scope.loadedPanel = + $scope.renderPromise = embeddableHandler.render(element.find('#embeddedPanel').get(0), $scope.panel, $scope.containerApi).catch(handleError); } }; diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index b0ad8bee0857a..5f9efad3f3ddb 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -15,63 +15,63 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { this.name = 'search'; } - getEditPath(panelId) { - return this.searchLoader.urlFor(panelId); + async getEditPath(panelId) { + return await this.searchLoader.urlFor(panelId); } - getTitleFor(panelId) { - return this.searchLoader.get(panelId).then(savedObject => savedObject.title); + async getTitleFor(panelId) { + const savedObject = await this.searchLoader.get(panelId); + return savedObject.title; } - render(domNode, panel, container) { - return this.searchLoader.get(panel.id).then((savedObject) => { - const editUrl = this.searchLoader.urlFor(panel.id); - const searchScope = this.$rootScope.$new(); - searchScope.editUrl = editUrl; - searchScope.savedObj = savedObject; - searchScope.panel = panel; - container.registerPanelIndexPattern(panel.panelIndex, savedObject.searchSource.get('index')); + async render(domNode, panel, container) { + const savedObject = await this.searchLoader.get(panel.id); + const editUrl = await this.getEditPath(panel.id); + const searchScope = this.$rootScope.$new(); + searchScope.editUrl = editUrl; + searchScope.savedObj = savedObject; + searchScope.panel = panel; + container.registerPanelIndexPattern(panel.panelIndex, savedObject.searchSource.get('index')); - // This causes changes to a saved search to be hidden, but also allows - // the user to locally modify and save changes to a saved search only in a dashboard. - // See https://github.com/elastic/kibana/issues/9523 for more details. - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { - columns: searchScope.panel.columns || searchScope.savedObj.columns, - sort: searchScope.panel.sort || searchScope.savedObj.sort - }); + // This causes changes to a saved search to be hidden, but also allows + // the user to locally modify and save changes to a saved search only in a dashboard. + // See https://github.com/elastic/kibana/issues/9523 for more details. + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { + columns: searchScope.panel.columns || searchScope.savedObj.columns, + sort: searchScope.panel.sort || searchScope.savedObj.sort + }); - const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); - searchScope.setSortOrder = function setSortOrder(columnName, direction) { - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); - }; + searchScope.setSortOrder = function setSortOrder(columnName, direction) { + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); + }; - searchScope.addColumn = function addColumn(columnName) { - savedObject.searchSource.get('index').popularizeField(columnName, 1); - columnActions.addColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.addColumn = function addColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.addColumn(searchScope.panel.columns, columnName); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.removeColumn = function removeColumn(columnName) { - savedObject.searchSource.get('index').popularizeField(columnName, 1); - columnActions.removeColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.removeColumn = function removeColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.removeColumn(searchScope.panel.columns, columnName); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.moveColumn = function moveColumn(columnName, newIndex) { - columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.moveColumn = function moveColumn(columnName, newIndex) { + columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.filter = function (field, value, operator) { - const index = savedObject.searchSource.get('index').id; - container.onFilter(field, value, operator, index); - }; + searchScope.filter = function (field, value, operator) { + const index = savedObject.searchSource.get('index').id; + container.onFilter(field, value, operator, index); + }; - const searchInstance = this.$compile(searchTemplate)(searchScope); - const rootNode = angular.element(domNode); - rootNode.append(searchInstance); - }); + const searchInstance = this.$compile(searchTemplate)(searchScope); + const rootNode = angular.element(domNode); + rootNode.append(searchInstance); } } diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index 17f5f409b39d8..a5570b8c55edd 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -18,40 +18,40 @@ export class VisualizeEmbeddableHandler extends EmbeddableHandler { this.filterBarClickHandler = filterBarClickHandlerProvider(Notifier); } - getEditPath(panelId) { + async getEditPath(panelId) { return this.visualizeLoader.urlFor(panelId); } - getTitleFor(panelId) { - return this.visualizeLoader.get(panelId).then(savedObject => savedObject.title); + async getTitleFor(panelId) { + const savedObject = await this.visualizeLoader.get(panelId); + return savedObject.title; } - render(domNode, panel, container) { - return this.visualizeLoader.get(panel.id).then((savedObject) => { - const visualizeScope = this.$rootScope.$new(); - visualizeScope.editUrl = this.getEditPath(panel); - visualizeScope.savedObj = savedObject; - visualizeScope.panel = panel; + async render(domNode, panel, container) { + const savedObject = await this.visualizeLoader.get(panel.id); + const visualizeScope = this.$rootScope.$new(); + visualizeScope.editUrl = await this.getEditPath(panel.id); + visualizeScope.savedObj = savedObject; + visualizeScope.panel = panel; - const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - visualizeScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + visualizeScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); - visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); + visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); - visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); - visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); - visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); + visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); + visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); + visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); - container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); + container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); - const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); - const rootNode = angular.element(domNode); - rootNode.append(visualizationInstance); + const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); + const rootNode = angular.element(domNode); + rootNode.append(visualizationInstance); - visualizationInstance.on('$destroy', function () { - visualizeScope.savedObj.destroy(); - visualizeScope.$destroy(); - }); + visualizationInstance.on('$destroy', function () { + visualizeScope.savedObj.destroy(); + visualizeScope.$destroy(); }); } } diff --git a/src/ui/public/embeddable/embeddable_handler.js b/src/ui/public/embeddable/embeddable_handler.js index abb75cea21af1..97ddf5ffc3e26 100644 --- a/src/ui/public/embeddable/embeddable_handler.js +++ b/src/ui/public/embeddable/embeddable_handler.js @@ -4,32 +4,33 @@ */ export class EmbeddableHandler { /** - * @param {string} panelId - the id of the panel to grad the title for. - * @return {string} a path that dictates where the user will be navigated to when they click the edit icon. + * @param {string} panelId - the id of the panel to grab the title for. + * @return {Promise.} a promise that resolves with the path that dictates where the user will be navigated to + * when they click the edit icon. */ - getEditPath(/* panelId */) { + async getEditPath(/* panelId */) { throw new Error('Must implement getEditPath.'); } /** - * @param {string} panelId - the id of the panel to grad the title for. - * @return {string} - the title to display for this particular panel. + * @param {string} panelId - the id of the panel to grab the title for. + * @return {Promise.} - Promise that resolves with the title to display for the particular panel. */ - getTitleFor(/* panelId */) { + async getTitleFor(/* panelId */) { throw new Error('Must implement getTitleFor.'); } /** * @param {Element} domNode - the dom node to mount the rendered embeddable on - * @param {Object} panel - a panel object which container information about the panel. Can also be modified to + * @param {PanelState} panel - a panel object which container information about the panel. Can also be modified to * store per panel information. * @property {string} panel.id - an id to specify the object that this panel contains. * @property {string} panel.type - specifies which EmbeddableHandler should render this panel. * @property {number} panel.panelIndex - a unique identifier for this panel in a specific container. Different * from panel.id because you can have the same object rendered multiple times in two different panels. - * @param {Object} container - Contains functions to communicate with the container. + * @param {Promise.} A promise that resolves when the object is finished rendering. */ - render(/* domNode, panel, container */) { - throw new Error('Must implement render.'); + async render(/* domNode, panel, container */) { + Promise.reject(new Error('Must implement render.')); } } From be3afde4031321ac918f185fba8bd01b0b623e9f Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Wed, 2 Aug 2017 10:00:31 -0400 Subject: [PATCH 14/21] Fix panel tests --- .../public/dashboard/__tests__/panel.js | 30 +++++++++---------- .../kibana/public/dashboard/dashboard.js | 2 +- .../public/dashboard/dashboard_state.js | 12 ++++---- .../kibana/public/dashboard/panel/panel.js | 7 +++-- 4 files changed, 27 insertions(+), 24 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js index da69aec871b8e..35b807ff0cf54 100644 --- a/src/core_plugins/kibana/public/dashboard/__tests__/panel.js +++ b/src/core_plugins/kibana/public/dashboard/__tests__/panel.js @@ -3,28 +3,30 @@ import ngMock from 'ng_mock'; import Promise from 'bluebird'; import sinon from 'sinon'; import noDigestPromise from 'test_utils/no_digest_promises'; -import mockUiState from 'fixtures/mock_ui_state'; -import { SavedObjectsClientProvider } from 'ui/saved_objects'; +import { DashboardContainerAPI } from '../dashboard_container_api'; +import { DashboardState } from '../dashboard_state'; +import { SavedObjectsClient } from 'ui/saved_objects'; describe('dashboard panel', function () { let $scope; let $el; let parentScope; + let savedDashboard; + let AppState; noDigestPromise.activateForSuite(); function init(mockDocResponse) { ngMock.module('kibana'); - ngMock.inject(($rootScope, $compile, Private) => { - Private.swap(SavedObjectsClientProvider, () => { - return { - get: sinon.stub().returns(Promise.resolve(mockDocResponse)) - }; - }); - + ngMock.inject(($rootScope, $compile, Private, $injector) => { + const SavedDashboard = $injector.get('SavedDashboard'); + AppState = $injector.get('AppState'); + savedDashboard = new SavedDashboard(); + sinon.stub(SavedObjectsClient.prototype, 'get').returns(Promise.resolve(mockDocResponse)); parentScope = $rootScope.$new(); parentScope.saveState = sinon.stub(); - parentScope.createChildUiState = sinon.stub().returns(mockUiState); + const dashboardState = new DashboardState(savedDashboard, AppState, false); + parentScope.containerApi = new DashboardContainerAPI(dashboardState); parentScope.getVisClickHandler = sinon.stub(); parentScope.getVisBrushHandler = sinon.stub(); parentScope.registerPanelIndexPattern = sinon.stub(); @@ -41,11 +43,8 @@ describe('dashboard panel', function () { panel="panel" is-full-screen-mode="false" is-expanded="false" - get-vis-click-handler="getVisClickHandler" - get-vis-brush-handler="getVisBrushHandler" - save-state="saveState" - register-panel-index-pattern="registerPanelIndexPattern" - create-child-ui-state="createChildUiState"> + container-api="containerApi" + > `)(parentScope); $scope = $el.isolateScope(); parentScope.$digest(); @@ -53,6 +52,7 @@ describe('dashboard panel', function () { } afterEach(() => { + SavedObjectsClient.prototype.get.restore(); $scope.$destroy(); $el.remove(); }); diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.js b/src/core_plugins/kibana/public/dashboard/dashboard.js index 585421ba18dfb..c2c4318db4435 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard.js @@ -99,7 +99,7 @@ app.directive('dashboardApp', function ($injector) { docTitle.change(dash.title); } - const dashboardState = new DashboardState(dash, AppState, dashboardConfig); + const dashboardState = new DashboardState(dash, AppState, dashboardConfig.getHideWriteControls()); $scope.appState = dashboardState.getAppState(); const queryManager = Private(QueryManagerProvider)(dashboardState.getAppState()); $scope.containerApi = new DashboardContainerAPI(dashboardState, queryManager); diff --git a/src/core_plugins/kibana/public/dashboard/dashboard_state.js b/src/core_plugins/kibana/public/dashboard/dashboard_state.js index f355482d2b8f1..e8e90a6f71011 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard_state.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard_state.js @@ -59,13 +59,13 @@ export class DashboardState { * * @param savedDashboard {SavedDashboard} * @param AppState {AppState} - * @param dashboardConfig {DashboardConfigProvider} + * @param hideWriteControls {boolean} true if write controls should be hidden. */ - constructor(savedDashboard, AppState, dashboardConfig) { + constructor(savedDashboard, AppState, hideWriteControls) { this.savedDashboard = savedDashboard; - this.dashboardConfig = dashboardConfig; + this.hideWriteControls = hideWriteControls; - this.stateDefaults = getStateDefaults(this.savedDashboard, this.dashboardConfig.getHideWriteControls()); + this.stateDefaults = getStateDefaults(this.savedDashboard, this.hideWriteControls); this.appState = new AppState(this.stateDefaults); this.uiState = this.appState.makeStateful('uiState'); @@ -117,7 +117,7 @@ export class DashboardState { // The right way to fix this might be to ensure the defaults object stored on state is a deep // clone, but given how much code uses the state object, I determined that to be too risky of a change for // now. TODO: revisit this! - this.stateDefaults = getStateDefaults(this.savedDashboard, this.dashboardConfig.getHideWriteControls()); + this.stateDefaults = getStateDefaults(this.savedDashboard, this.hideWriteControls); // The original query won't be restored by the above because the query on this.savedDashboard is applied // in place in order for it to affect the visualizations. this.stateDefaults.query = this.lastSavedDashboardFilters.query; @@ -259,7 +259,7 @@ export class DashboardState { * @returns {DashboardViewMode} */ getViewMode() { - return this.dashboardConfig.getHideWriteControls() ? DashboardViewMode.VIEW : this.appState.viewMode; + return this.hideWriteControls ? DashboardViewMode.VIEW : this.appState.viewMode; } /** diff --git a/src/core_plugins/kibana/public/dashboard/panel/panel.js b/src/core_plugins/kibana/public/dashboard/panel/panel.js index 5a296003a57ee..2c2a53e1b643a 100644 --- a/src/core_plugins/kibana/public/dashboard/panel/panel.js +++ b/src/core_plugins/kibana/public/dashboard/panel/panel.js @@ -101,8 +101,11 @@ uiModules embeddableHandler.getTitleFor(panelId).then(title => { $scope.title = title; }); - $scope.renderPromise = - embeddableHandler.render(element.find('#embeddedPanel').get(0), $scope.panel, $scope.containerApi).catch(handleError); + $scope.renderPromise = embeddableHandler.render( + element.find('#embeddedPanel').get(0), + $scope.panel, + $scope.containerApi) + .catch(handleError); } }; }); From 6ccedd3c1f78d95f57c3d1434d61172a0bc969af Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Wed, 2 Aug 2017 13:03:08 -0400 Subject: [PATCH 15/21] Fix bug which broke tests - need to pass container-api to dashboard-panel --- src/core_plugins/kibana/public/dashboard/dashboard.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/core_plugins/kibana/public/dashboard/dashboard.html b/src/core_plugins/kibana/public/dashboard/dashboard.html index d80074278d2c5..89ffa8f483943 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard.html +++ b/src/core_plugins/kibana/public/dashboard/dashboard.html @@ -82,7 +82,6 @@

panels="panels" save-state="saveState" toggle-expand="toggleExpandPanel" - register-panel-index-pattern="registerPanelIndexPattern" data-shared-items-count="{{panels.length}}" container-api="containerApi" > @@ -93,6 +92,7 @@

is-full-screen-mode="!chrome.getVisible()" is-expanded="true" dashboard-view-mode="dashboardViewMode" + container-api="containerApi" toggle-expand="toggleExpandPanel(expandedPanel.panelIndex)" > From 78cce39fe24fc39535e0234673566b2c5ed5064d Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 3 Aug 2017 16:47:53 -0400 Subject: [PATCH 16/21] Address code comments - Rename onFilter to addFilter - Use angular promises instead of async/await - fix jsdoc - rename createChildUiState to getInitialState --- .../dashboard/__tests__/dashboard_panels.js | 13 ++- .../dashboard/dashboard_container_api.js | 4 +- .../embeddable/search_embeddable_handler.js | 96 ++++++++++--------- .../search_embeddable_handler_provider.js | 4 +- .../visualize_embeddable_handler.js | 68 ++++++------- .../visualize_embeddable_handler_provider.js | 10 +- src/ui/public/embeddable/container_api.js | 10 +- .../public/embeddable/embeddable_handler.js | 5 +- 8 files changed, 112 insertions(+), 98 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/__tests__/dashboard_panels.js b/src/core_plugins/kibana/public/dashboard/__tests__/dashboard_panels.js index 87903c241bdd5..109e9c34e9624 100644 --- a/src/core_plugins/kibana/public/dashboard/__tests__/dashboard_panels.js +++ b/src/core_plugins/kibana/public/dashboard/__tests__/dashboard_panels.js @@ -2,14 +2,18 @@ import angular from 'angular'; import expect from 'expect.js'; import ngMock from 'ng_mock'; import 'plugins/kibana/dashboard/saved_dashboard/saved_dashboard'; +import { DashboardContainerAPI } from '../dashboard_container_api'; +import { DashboardState } from '../dashboard_state'; import { DEFAULT_PANEL_WIDTH, DEFAULT_PANEL_HEIGHT } from 'plugins/kibana/dashboard/panel/panel_state'; describe('dashboard panels', function () { let $scope; let $el; + let AppState; function compile(dashboard) { - ngMock.inject(($rootScope, $controller, $compile, $route) => { + ngMock.inject(($injector, $rootScope, $controller, $compile, $route) => { + AppState = $injector.get('AppState'); $scope = $rootScope.$new(); $route.current = { locals: { @@ -18,6 +22,8 @@ describe('dashboard panels', function () { params: {} }; + const dashboardState = new DashboardState(dashboard, AppState, false); + $scope.containerApi = new DashboardContainerAPI(dashboardState); $el = angular.element(` `); $compile($el)($scope); diff --git a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js index f52cba0bb41c8..33a19d0fc0b69 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js @@ -7,7 +7,7 @@ export class DashboardContainerAPI extends ContainerAPI { this.queryManager = queryManager; } - onFilter(field, value, operator, index) { + addFilter(field, value, operator, index) { this.queryManager.add(field, value, operator, index); } @@ -26,7 +26,7 @@ export class DashboardContainerAPI extends ContainerAPI { return this.dashboardState.getIsViewMode(); } - createChildUiState(path, uiState) { + getInitialState(path, uiState) { return this.dashboardState.uiState.createChild(path, uiState, true); } diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index 5f9efad3f3ddb..d4ef352d1e3fa 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -7,71 +7,75 @@ import { EmbeddableHandler } from 'ui/embeddable'; export class SearchEmbeddableHandler extends EmbeddableHandler { - constructor($compile, $rootScope, searchLoader) { + constructor($compile, $rootScope, searchLoader, Promise) { super(); this.$compile = $compile; this.searchLoader = searchLoader; this.$rootScope = $rootScope; this.name = 'search'; + this.Promise = Promise; } - async getEditPath(panelId) { - return await this.searchLoader.urlFor(panelId); + getEditPath(panelId) { + return this.Promise.resolve(this.searchLoader.urlFor(panelId)); } - async getTitleFor(panelId) { - const savedObject = await this.searchLoader.get(panelId); - return savedObject.title; + getTitleFor(panelId) { + return this.searchLoader.get(panelId).then(savedObject => savedObject.title); } - async render(domNode, panel, container) { - const savedObject = await this.searchLoader.get(panel.id); - const editUrl = await this.getEditPath(panel.id); + render(domNode, panel, container) { const searchScope = this.$rootScope.$new(); - searchScope.editUrl = editUrl; - searchScope.savedObj = savedObject; - searchScope.panel = panel; - container.registerPanelIndexPattern(panel.panelIndex, savedObject.searchSource.get('index')); + return this.getEditPath(panel.id) + .then(editPath => { + searchScope.editPath = editPath; + return this.searchLoader.get(panel.id); + }) + .then(savedObject => { + searchScope.savedObj = savedObject; + searchScope.panel = panel; + container.registerPanelIndexPattern(panel.panelIndex, savedObject.searchSource.get('index')); - // This causes changes to a saved search to be hidden, but also allows - // the user to locally modify and save changes to a saved search only in a dashboard. - // See https://github.com/elastic/kibana/issues/9523 for more details. - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { - columns: searchScope.panel.columns || searchScope.savedObj.columns, - sort: searchScope.panel.sort || searchScope.savedObj.sort - }); + // This causes changes to a saved search to be hidden, but also allows + // the user to locally modify and save changes to a saved search only in a dashboard. + // See https://github.com/elastic/kibana/issues/9523 for more details. + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { + columns: searchScope.panel.columns || searchScope.savedObj.columns, + sort: searchScope.panel.sort || searchScope.savedObj.sort + }); - const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - searchScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + searchScope.uiState = container.getInitialState(getPersistedStateId(panel), uiState); - searchScope.setSortOrder = function setSortOrder(columnName, direction) { - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); - }; + searchScope.setSortOrder = function setSortOrder(columnName, direction) { + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); + }; - searchScope.addColumn = function addColumn(columnName) { - savedObject.searchSource.get('index').popularizeField(columnName, 1); - columnActions.addColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.addColumn = function addColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.addColumn(searchScope.panel.columns, columnName); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.removeColumn = function removeColumn(columnName) { - savedObject.searchSource.get('index').popularizeField(columnName, 1); - columnActions.removeColumn(searchScope.panel.columns, columnName); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.removeColumn = function removeColumn(columnName) { + savedObject.searchSource.get('index').popularizeField(columnName, 1); + columnActions.removeColumn(searchScope.panel.columns, columnName); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.moveColumn = function moveColumn(columnName, newIndex) { - columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); - searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); - }; + searchScope.moveColumn = function moveColumn(columnName, newIndex) { + columnActions.moveColumn(searchScope.panel.columns, columnName, newIndex); + searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { columns: searchScope.panel.columns }); + }; - searchScope.filter = function (field, value, operator) { - const index = savedObject.searchSource.get('index').id; - container.onFilter(field, value, operator, index); - }; + searchScope.filter = function (field, value, operator) { + const index = savedObject.searchSource.get('index').id; + container.addFilter(field, value, operator, index); + }; - const searchInstance = this.$compile(searchTemplate)(searchScope); - const rootNode = angular.element(domNode); - rootNode.append(searchInstance); + const searchInstance = this.$compile(searchTemplate)(searchScope); + const rootNode = angular.element(domNode); + rootNode.append(searchInstance); + }); } } diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js index 9298f6527374e..d7fa5dc8b1f00 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler_provider.js @@ -2,8 +2,8 @@ import { SearchEmbeddableHandler } from './search_embeddable_handler'; import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; export function searchEmbeddableHandlerProvider(Private) { - const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches) => { - return new SearchEmbeddableHandler($compile, $rootScope, savedSearches); + const SearchEmbeddableHandlerProvider = ($compile, $rootScope, savedSearches, Promise) => { + return new SearchEmbeddableHandler($compile, $rootScope, savedSearches, Promise); }; return Private(SearchEmbeddableHandlerProvider); } diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index a5570b8c55edd..f3a4db166007a 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -7,52 +7,56 @@ import { FilterBarClickHandlerProvider as filterBarClickHandlerProvider } from ' import { EmbeddableHandler } from 'ui/embeddable'; export class VisualizeEmbeddableHandler extends EmbeddableHandler { - constructor($compile, $rootScope, visualizeLoader, timefilter, Notifier) { + constructor($compile, $rootScope, visualizeLoader, timefilter, Notifier, Promise) { super(); this.$compile = $compile; this.visualizeLoader = visualizeLoader; this.$rootScope = $rootScope; this.name = 'visualization'; - + this.Promise = Promise; this.brushEvent = utilsBrushEventProvider(timefilter); this.filterBarClickHandler = filterBarClickHandlerProvider(Notifier); } - async getEditPath(panelId) { - return this.visualizeLoader.urlFor(panelId); + getEditPath(panelId) { + return this.Promise.resolve(this.visualizeLoader.urlFor(panelId)); } - async getTitleFor(panelId) { - const savedObject = await this.visualizeLoader.get(panelId); - return savedObject.title; + getTitleFor(panelId) { + return this.visualizeLoader.get(panelId).then(savedObject => savedObject.title); } - async render(domNode, panel, container) { - const savedObject = await this.visualizeLoader.get(panel.id); + render(domNode, panel, container) { const visualizeScope = this.$rootScope.$new(); - visualizeScope.editUrl = await this.getEditPath(panel.id); - visualizeScope.savedObj = savedObject; - visualizeScope.panel = panel; - - const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - visualizeScope.uiState = container.createChildUiState(getPersistedStateId(panel), uiState); - - visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); - - visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); - visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); - visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); - - container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); - - const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); - const rootNode = angular.element(domNode); - rootNode.append(visualizationInstance); - - visualizationInstance.on('$destroy', function () { - visualizeScope.savedObj.destroy(); - visualizeScope.$destroy(); - }); + return this.getEditPath(panel.id) + .then(editPath => { + visualizeScope.editUrl = editPath; + return this.visualizeLoader.get(panel.id); + }) + .then(savedObject => { + visualizeScope.savedObj = savedObject; + visualizeScope.panel = panel; + + const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; + visualizeScope.uiState = container.getInitialState(getPersistedStateId(panel), uiState); + + visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); + + visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); + visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); + visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); + + container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); + + const visualizationInstance = this.$compile(visualizationTemplate)(visualizeScope); + const rootNode = angular.element(domNode); + rootNode.append(visualizationInstance); + + visualizationInstance.on('$destroy', function () { + visualizeScope.savedObj.destroy(); + visualizeScope.$destroy(); + }); + }); } } diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js index babdb3ea23000..1f6704f1f921f 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler_provider.js @@ -2,8 +2,14 @@ import { VisualizeEmbeddableHandler } from './visualize_embeddable_handler'; import { EmbeddableHandlersRegistryProvider } from 'ui/embeddable/embeddable_handlers_registry'; export function visualizeEmbeddableHandlerProvider(Private) { - const VisualizeEmbeddableHandlerProvider = ($compile, $rootScope, savedVisualizations, timefilter, Notifier) => { - return new VisualizeEmbeddableHandler($compile, $rootScope, savedVisualizations, timefilter, Notifier); + const VisualizeEmbeddableHandlerProvider = ( + $compile, + $rootScope, + savedVisualizations, + timefilter, + Notifier, + Promise) => { + return new VisualizeEmbeddableHandler($compile, $rootScope, savedVisualizations, timefilter, Notifier, Promise); }; return Private(VisualizeEmbeddableHandlerProvider); } diff --git a/src/ui/public/embeddable/container_api.js b/src/ui/public/embeddable/container_api.js index 522fd15a5c583..d32b715f45ea4 100644 --- a/src/ui/public/embeddable/container_api.js +++ b/src/ui/public/embeddable/container_api.js @@ -9,8 +9,8 @@ export class ContainerAPI { * @param operator * @param index */ - onFilter(/*field, value, operator, index */) { - throw new Error('Must implement onFilter.'); + addFilter(/*field, value, operator, index */) { + throw new Error('Must implement addFilter.'); } /** @@ -28,14 +28,14 @@ export class ContainerAPI { } /** - * Creates a child ui state for the panel. It's passed the ui state to use, but needs to + * Creates a new state for the panel. It's passed the ui state to use, but needs to * be generated from the parent. * @param path {String} - the unique path for this ui state. * @param uiState {Object} - the uiState for the child. * @returns {Object} */ - createChildUiState(/* path, uiState */) { - throw new Error('Must implement createChildUiState.'); + getInitialState(/* path, uiState */) { + throw new Error('Must implement getInitalState.'); } /** diff --git a/src/ui/public/embeddable/embeddable_handler.js b/src/ui/public/embeddable/embeddable_handler.js index 97ddf5ffc3e26..8c5872888fa1d 100644 --- a/src/ui/public/embeddable/embeddable_handler.js +++ b/src/ui/public/embeddable/embeddable_handler.js @@ -24,10 +24,7 @@ export class EmbeddableHandler { * @param {Element} domNode - the dom node to mount the rendered embeddable on * @param {PanelState} panel - a panel object which container information about the panel. Can also be modified to * store per panel information. - * @property {string} panel.id - an id to specify the object that this panel contains. - * @property {string} panel.type - specifies which EmbeddableHandler should render this panel. - * @property {number} panel.panelIndex - a unique identifier for this panel in a specific container. Different - * from panel.id because you can have the same object rendered multiple times in two different panels. + * @property {ContainerApi} containerApi - an id to specify the object that this panel contains. * @param {Promise.} A promise that resolves when the object is finished rendering. */ async render(/* domNode, panel, container */) { From 033d37688eb420da5a0f291d48935bb4464fa79e Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 3 Aug 2017 17:34:40 -0400 Subject: [PATCH 17/21] changed the wrong variable name --- .../kibana/public/dashboard/dashboard_container_api.js | 4 ++-- .../discover/embeddable/search_embeddable_handler.js | 2 +- .../embeddable/visualize_embeddable_handler.js | 2 +- src/ui/public/embeddable/container_api.js | 10 +++++----- 4 files changed, 9 insertions(+), 9 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js index 33a19d0fc0b69..e74cbc0579a36 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js @@ -26,8 +26,8 @@ export class DashboardContainerAPI extends ContainerAPI { return this.dashboardState.getIsViewMode(); } - getInitialState(path, uiState) { - return this.dashboardState.uiState.createChild(path, uiState, true); + createChildUistate(path, initialState) { + return this.dashboardState.uiState.createChild(path, initialState, true); } registerPanelIndexPattern(panelIndex, pattern) { diff --git a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js index d4ef352d1e3fa..1c12be3e8cab4 100644 --- a/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js +++ b/src/core_plugins/kibana/public/discover/embeddable/search_embeddable_handler.js @@ -45,7 +45,7 @@ export class SearchEmbeddableHandler extends EmbeddableHandler { }); const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - searchScope.uiState = container.getInitialState(getPersistedStateId(panel), uiState); + searchScope.uiState = container.createChildUistate(getPersistedStateId(panel), uiState); searchScope.setSortOrder = function setSortOrder(columnName, direction) { searchScope.panel = container.updatePanel(searchScope.panel.panelIndex, { sort: [columnName, direction] }); diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index f3a4db166007a..5145c7e3b0a2c 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -38,7 +38,7 @@ export class VisualizeEmbeddableHandler extends EmbeddableHandler { visualizeScope.panel = panel; const uiState = savedObject.uiStateJSON ? JSON.parse(savedObject.uiStateJSON) : {}; - visualizeScope.uiState = container.getInitialState(getPersistedStateId(panel), uiState); + visualizeScope.uiState = container.createChildUistate(getPersistedStateId(panel), uiState); visualizeScope.savedObj.vis.setUiState(visualizeScope.uiState); diff --git a/src/ui/public/embeddable/container_api.js b/src/ui/public/embeddable/container_api.js index d32b715f45ea4..932055efcabad 100644 --- a/src/ui/public/embeddable/container_api.js +++ b/src/ui/public/embeddable/container_api.js @@ -28,13 +28,13 @@ export class ContainerAPI { } /** - * Creates a new state for the panel. It's passed the ui state to use, but needs to - * be generated from the parent. + * Creates a new state for the panel. It's passed the ui state object to use, and is returned + * a PersistedState. * @param path {String} - the unique path for this ui state. - * @param uiState {Object} - the uiState for the child. - * @returns {Object} + * @param initialState {Object} - the initial state to use for the child. + * @returns {PersistedState} */ - getInitialState(/* path, uiState */) { + createChildUistate(/* path, initialState */) { throw new Error('Must implement getInitalState.'); } From 4ab70722dd8d90a2a0ebd1b14e4dcefb5d28b4c8 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Thu, 3 Aug 2017 17:37:24 -0400 Subject: [PATCH 18/21] no need for async or Promise.reject on interface --- src/ui/public/embeddable/embeddable_handler.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/ui/public/embeddable/embeddable_handler.js b/src/ui/public/embeddable/embeddable_handler.js index 8c5872888fa1d..aee16ca8e6049 100644 --- a/src/ui/public/embeddable/embeddable_handler.js +++ b/src/ui/public/embeddable/embeddable_handler.js @@ -8,7 +8,7 @@ export class EmbeddableHandler { * @return {Promise.} a promise that resolves with the path that dictates where the user will be navigated to * when they click the edit icon. */ - async getEditPath(/* panelId */) { + getEditPath(/* panelId */) { throw new Error('Must implement getEditPath.'); } @@ -16,7 +16,7 @@ export class EmbeddableHandler { * @param {string} panelId - the id of the panel to grab the title for. * @return {Promise.} - Promise that resolves with the title to display for the particular panel. */ - async getTitleFor(/* panelId */) { + getTitleFor(/* panelId */) { throw new Error('Must implement getTitleFor.'); } @@ -27,7 +27,7 @@ export class EmbeddableHandler { * @property {ContainerApi} containerApi - an id to specify the object that this panel contains. * @param {Promise.} A promise that resolves when the object is finished rendering. */ - async render(/* domNode, panel, container */) { - Promise.reject(new Error('Must implement render.')); + render(/* domNode, panel, container */) { + throw new Error('Must implement render.'); } } From a685b471c97b64de124f501cdf8c452806ad071e Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Fri, 4 Aug 2017 09:14:30 -0400 Subject: [PATCH 19/21] add tests that will fail due to spy pane issue in this PR --- test/functional/apps/dashboard/_dashboard.js | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/test/functional/apps/dashboard/_dashboard.js b/test/functional/apps/dashboard/_dashboard.js index 3ae4ea1476175..9cc9538060c1e 100644 --- a/test/functional/apps/dashboard/_dashboard.js +++ b/test/functional/apps/dashboard/_dashboard.js @@ -182,6 +182,19 @@ export default function ({ getService, getPageObjects }) { expect(spyToggleExists).to.be(true); }); + // This was an actual bug that appeared, where the spy pane appeared on panels after adding them, but + // disappeared when a new dashboard was opened up. + it('shows the spy pane toggle directly after opening a dashboard', async () => { + await PageObjects.dashboard.saveDashboard('spy pane test'); + await PageObjects.dashboard.gotoDashboardLandingPage(); + await PageObjects.dashboard.loadSavedDashboard('spy pane test'); + const panels = await PageObjects.dashboard.getDashboardPanels(); + // Simulate hover + await remote.moveMouseTo(panels[0]); + const spyToggleExists = await PageObjects.visualize.getSpyToggleExists(); + expect(spyToggleExists).to.be(true); + }); + it('shows other panels after being minimized', async () => { await PageObjects.dashboard.toggleExpandPanel(); const panels = await PageObjects.dashboard.getDashboardPanels(); @@ -222,6 +235,7 @@ export default function ({ getService, getPageObjects }) { describe('add new visualization link', () => { it('adds a new visualization', async () => { + await PageObjects.dashboard.clickEdit(); await PageObjects.dashboard.clickAddVisualization(); await PageObjects.dashboard.clickAddNewVisualizationLink(); await PageObjects.visualize.clickAreaChart(); From f74cc0ed9a04763738aee78923dbb54c0527ff82 Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Fri, 4 Aug 2017 09:20:51 -0400 Subject: [PATCH 20/21] Fix logic with spy pane toggle MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit There is still a bit of a bug here as mentioned in https://github.com/elastic/kibana/issues/13340 but it will be fixed separately as it’s also an issue in master --- .../kibana/public/dashboard/dashboard_container_api.js | 4 ---- .../visualize/embeddable/visualize_embeddable_handler.js | 3 ++- src/ui/public/embeddable/container_api.js | 7 ------- test/functional/apps/dashboard/_dashboard.js | 2 ++ 4 files changed, 4 insertions(+), 12 deletions(-) diff --git a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js index e74cbc0579a36..f8ca5a98f5826 100644 --- a/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js +++ b/src/core_plugins/kibana/public/dashboard/dashboard_container_api.js @@ -22,10 +22,6 @@ export class DashboardContainerAPI extends ContainerAPI { return this.dashboardState.appState; } - getIsViewOnlyMode() { - return this.dashboardState.getIsViewMode(); - } - createChildUistate(path, initialState) { return this.dashboardState.uiState.createChild(path, initialState, true); } diff --git a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js index 5145c7e3b0a2c..8bb5c6e9ccd8c 100644 --- a/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js +++ b/src/core_plugins/kibana/public/visualize/embeddable/visualize_embeddable_handler.js @@ -5,6 +5,7 @@ import { getPersistedStateId } from 'plugins/kibana/dashboard/panel/panel_state' import { UtilsBrushEventProvider as utilsBrushEventProvider } from 'ui/utils/brush_event'; import { FilterBarClickHandlerProvider as filterBarClickHandlerProvider } from 'ui/filter_bar/filter_bar_click_handler'; import { EmbeddableHandler } from 'ui/embeddable'; +import chrome from 'ui/chrome'; export class VisualizeEmbeddableHandler extends EmbeddableHandler { constructor($compile, $rootScope, visualizeLoader, timefilter, Notifier, Promise) { @@ -44,7 +45,7 @@ export class VisualizeEmbeddableHandler extends EmbeddableHandler { visualizeScope.savedObj.vis.listeners.click = this.filterBarClickHandler(container.getAppState()); visualizeScope.savedObj.vis.listeners.brush = this.brushEvent(container.getAppState()); - visualizeScope.isFullScreenMode = container.getIsViewOnlyMode(); + visualizeScope.isFullScreenMode = !chrome.getVisible(); container.registerPanelIndexPattern(panel.panelIndex, visualizeScope.savedObj.vis.indexPattern); diff --git a/src/ui/public/embeddable/container_api.js b/src/ui/public/embeddable/container_api.js index 932055efcabad..e1471a88d7a70 100644 --- a/src/ui/public/embeddable/container_api.js +++ b/src/ui/public/embeddable/container_api.js @@ -20,13 +20,6 @@ export class ContainerAPI { throw new Error('Must implement getAppState.'); } - /** - * @return {boolean} If the container is marked as being in view only mode or not. - */ - getIsViewOnlyMode() { - throw new Error('Must implement getIsViewOnlyMode.'); - } - /** * Creates a new state for the panel. It's passed the ui state object to use, and is returned * a PersistedState. diff --git a/test/functional/apps/dashboard/_dashboard.js b/test/functional/apps/dashboard/_dashboard.js index 9cc9538060c1e..42b081b7bdd10 100644 --- a/test/functional/apps/dashboard/_dashboard.js +++ b/test/functional/apps/dashboard/_dashboard.js @@ -196,6 +196,8 @@ export default function ({ getService, getPageObjects }) { }); it('shows other panels after being minimized', async () => { + // Panels are all minimized on a fresh open of a dashboard, so we need to re-expand in order to then minimize. + await PageObjects.dashboard.toggleExpandPanel(); await PageObjects.dashboard.toggleExpandPanel(); const panels = await PageObjects.dashboard.getDashboardPanels(); const visualizations = PageObjects.dashboard.getTestVisualizations(); From ccfaadb6834f1cad088ebce1f10bd3f26e4b9c4c Mon Sep 17 00:00:00 2001 From: Stacey Gammon Date: Tue, 8 Aug 2017 09:52:10 -0400 Subject: [PATCH 21/21] Fix failing test --- test/functional/apps/dashboard/_dashboard.js | 1 + 1 file changed, 1 insertion(+) diff --git a/test/functional/apps/dashboard/_dashboard.js b/test/functional/apps/dashboard/_dashboard.js index 1b75c63f72815..48f32a81e2231 100644 --- a/test/functional/apps/dashboard/_dashboard.js +++ b/test/functional/apps/dashboard/_dashboard.js @@ -237,6 +237,7 @@ export default function ({ getService, getPageObjects }) { describe('full screen mode', () => { it('option not available in edit mode', async () => { + await PageObjects.dashboard.clickEdit(); const exists = await PageObjects.dashboard.fullScreenModeMenuItemExists(); expect(exists).to.be(false); });