diff --git a/src/core_plugins/timelion/public/app.less b/src/core_plugins/timelion/public/app.less index 0095eaa710480..97428574bf3de 100644 --- a/src/core_plugins/timelion/public/app.less +++ b/src/core_plugins/timelion/public/app.less @@ -24,6 +24,18 @@ } } +/** + * 1. Needs to be relative to contain absolutely-positioned typeahead suggestions. + */ +.timelionLocalSearch { + position: relative; /* 1 */ +} + +.timelionSearchInputContainer { + flex: 1 1 auto; + display: flex; +} + .timelion-container { margin: 20px; } @@ -38,10 +50,6 @@ background-color: @gray-lighter; } -.timelion-expression { - position: relative !important; -} - .timelion-subnav { background-color: @gray-lighter; margin: 0px @@ -88,33 +96,17 @@ opacity: 0.50; } -.timelion-interval { - width: 70px !important; - height: auto; - -webkit-appearance: none; - -moz-appearance: none; - padding: .5em; - padding-right: 1.5em; - line-height: 1; - border-top: 0; - border-right: 0; - border-bottom: 0; - border-radius: 0; - text-align: center; - - &:focus { - border-color: #ecf0f1; - } +.timelion-interval-custom { + width: 60px; } -.timelion-interval--select { - background-position: right 50%; - background-repeat: no-repeat; - background-image: url(); +.timelion-interval-presets { + width: 90px; } -.timelion-interval-other { - width: 10px !important; +.timelion-interval-presets-compact { + width: 10px; + padding-left: 0; } timelion-interval { @@ -214,10 +206,3 @@ timelion-interval { margin-right: 5px; font-weight: bold; } - -/** - * 1. Needs to be relative to contain absolutely-positioned typeahead suggestions. - */ -.timelionLocalSearch { - position: relative; /* 1 */ -} diff --git a/src/core_plugins/timelion/public/directives/expression_directive.html b/src/core_plugins/timelion/public/directives/expression_directive.html new file mode 100644 index 0000000000000..314528c634042 --- /dev/null +++ b/src/core_plugins/timelion/public/directives/expression_directive.html @@ -0,0 +1,23 @@ +
+ + + +
diff --git a/src/core_plugins/timelion/public/directives/expression_directive.js b/src/core_plugins/timelion/public/directives/expression_directive.js index 9d90b36b5cc6b..3f33918d7a3d1 100644 --- a/src/core_plugins/timelion/public/directives/expression_directive.js +++ b/src/core_plugins/timelion/public/directives/expression_directive.js @@ -2,10 +2,14 @@ import _ from 'lodash'; import $ from 'jquery'; import grammar from 'raw!../chain.peg'; import PEG from 'pegjs'; -const Parser = PEG.buildParser(grammar); -import template from './partials/suggestion.html'; -const app = require('ui/modules').get('apps/timelion', []); +import './partials/suggestion'; +import timelionExpressionInputTemplate from './expression_directive.html'; +import { + FunctionSuggestions, + suggest, + insertAtLocation, +} from './expression_directive_helpers'; /* Autocomplete proposal, this file doesn't actually work like this @@ -30,17 +34,22 @@ Only named arguments, necessarily provided optional by a plugin. Must be inside a function, and start must be adjacent to the argument name .function(arg=b|) +*/ +const Parser = PEG.buildParser(grammar); -*/ +const app = require('ui/modules').get('apps/timelion', []); -app.directive('timelionExpression', function ($compile, $http, $timeout, $rootScope) { +app.directive('timelionExpressionInput', function ($compile, $http, $timeout) { return { - restrict: 'A', - require: 'ngModel', - link: function ($scope, $elem, $attrs, ngModelCtrl) { - - const keys = { + restrict: 'E', + scope: { + sheet: '=', + }, + replace: true, + template: timelionExpressionInputTemplate, + link: function ($scope, $elem) { + const navigationalKeys = { ESC: 27, UP: 38, DOWN: 40, @@ -49,184 +58,144 @@ app.directive('timelionExpression', function ($compile, $http, $timeout, $rootSc }; const functionReference = {}; + const input = $elem.find('#timelionSearchInput'); + const caretLocation = {}; - function init() { - resetSuggestions(); - $elem.on('mouseup', function () { - suggest($attrs.timelionExpression); - digest(); - }); - $elem.on('keydown', keyDownHandler); - $elem.on('keyup', keyUpHandler); - $elem.on('blur', function () { - $timeout(function () { - $scope.suggestions.show = false; - }, 100); - }); + $scope.functionSuggestions = new FunctionSuggestions(); - $elem.after($compile(template)($scope)); + function init() { $http.get('../api/timelion/functions').then(function (resp) { - functionReference.byName = _.indexBy(resp.data, 'name'); - functionReference.list = resp.data; + Object.assign(functionReference, { + byName: _.indexBy(resp.data, 'name'), + list: resp.data, + }); }); } - $scope.$on('$destroy', function () { - $elem.off('mouseup'); - $elem.off('keydown'); - $elem.off('keyup'); - $elem.off('blur'); - }); - - function suggest(val) { - try { - // Inside an existing function providing suggestion only as a reference. Maybe suggest an argument? - const possible = findFunction(getCaretPos(), Parser.parse(val).functions); - // TODO: Reference suggestors. Only supporting completion right now; - resetSuggestions(); + function completeExpression(suggestionIndex) { + if ($scope.functionSuggestions.isEmpty()) { return; - - - if (functionReference.byName) { - if (functionReference.byName[possible.function]) { - $scope.suggestions.list = [functionReference.byName[possible.function]]; - $scope.suggestions.show = true; - } else { - resetSuggestions(); - } - } - } catch (e) { - try { // Is this a structured exception? - e = JSON.parse(e.message); - if (e.location.min > getCaretPos() || e.location.max <= getCaretPos()) { - resetSuggestions(); - return; - } - // TODO: Abstract structured exception handling; - if (e.type === 'incompleteFunction') { - if (e.function == null) { - $scope.suggestions.list = functionReference.list; - } else { - $scope.suggestions.list = _.compact(_.map(functionReference.list, function (func) { - if (_.startsWith(func.name, e.function)) { - return func; - } - })); - } - $scope.suggestions.show = true; - } - $scope.suggestions.location = e.location; - } catch (e) { - resetSuggestions(); - } } - digest(); - } - - function validateSelection() { - const maxSelection = $scope.suggestions.list.length - 1; - if ($scope.suggestions.selected > maxSelection) $scope.suggestions.selected = maxSelection; - else if ($scope.suggestions.selected < 0) $scope.suggestions.selected = 0; - } - - $scope.completeExpression = function (selected) { - if (!$scope.suggestions.list.length) return; - const expression = $attrs.timelionExpression; - const startOf = expression.slice(0, $scope.suggestions.location.min); - const endOf = expression.slice($scope.suggestions.location.max, expression.length); - const completeFunction = $scope.suggestions.list[selected].name + '()'; + const functionName = `${$scope.functionSuggestions.list[suggestionIndex].name}()`; + const expression = $scope.sheet; + const { min, max } = caretLocation; - const newVal = startOf + completeFunction + endOf; + const newExpression = insertAtLocation(functionName, expression, min, max); + input.val(newExpression); - $elem.val(newVal); - $elem[0].selectionStart = $elem[0].selectionEnd = - (startOf + completeFunction).length - 1; - ngModelCtrl.$setViewValue(newVal); + const newCaretPosition = min + functionName.length - 1; + input[0].selectionStart = input[0].selectionEnd = newCaretPosition; - resetSuggestions(); - }; + $scope.functionSuggestions.reset(); + } + function scrollTo(selected) { + const suggestionsListElem = $('[data-suggestions-list]'); + const suggestedElem = $($('[data-suggestion-list-item]')[selected]); - function keyDownHandler(e) { - if (_.contains(_.values(keys), e.keyCode)) e.preventDefault(); - switch (e.keyCode) { - case keys.UP: - if ($scope.suggestions.selected > 0) $scope.suggestions.selected--; - break; - case keys.DOWN: - $scope.suggestions.selected++; - break; - case keys.TAB: - $scope.completeExpression($scope.suggestions.selected); - break; - case keys.ENTER: - if ($scope.suggestions.list.length) { - $scope.completeExpression($scope.suggestions.selected); - } else { - $elem.submit(); - } - break; - case keys.ESC: - $scope.suggestions.show = false; - break; + if (!suggestedElem.position() || !suggestedElem.position().top) { + return; } - scrollTo($scope.suggestions); - digest(); - } - function keyUpHandler(e) { - if (_.contains(_.values(keys), e.keyCode)) return; + suggestionsListElem.scrollTop(suggestionsListElem.scrollTop() + suggestedElem.position().top); + } - suggest($attrs.timelionExpression); - validateSelection(); - digest(); + function getSuggestions() { + const caretPosition = input[0].selectionStart; + + suggest( + $scope.sheet, + caretPosition, + functionReference.list, + Parser + ).then(({ list, location }) => { + // We're using ES6 Promises, not $q, so we have to wrap this in $apply. + $scope.$apply(() => { + $scope.functionSuggestions.setList(list); + $scope.functionSuggestions.show(); + Object.assign(caretLocation, location); + }); + }, ({ location } = {}) => { + $scope.$apply(() => { + Object.assign(caretLocation, location); + $scope.functionSuggestions.reset(); + }); + }); } - function resetSuggestions() { - $scope.suggestions = { - selected: 0, - list: [], - position: {}, - show: false - }; - return $scope.suggestions; + function isNavigationalKey(keyCode) { + const keyCodes = _.values(navigationalKeys); + return keyCodes.includes(keyCode); } - function scrollTo(suggestions) { - validateSelection(); - const suggestionsListElem = $('.suggestions'); - const suggestedElem = $($('.suggestion')[suggestions.selected]); + $scope.mouseUpHandler = () => { + getSuggestions(); + }; - if (!suggestedElem.position() || !suggestedElem.position().top) return; + $scope.blurHandler = () => { + $timeout(() => { + $scope.functionSuggestions.hide(); + }, 100); + }; - suggestionsListElem.scrollTop(suggestionsListElem.scrollTop() + suggestedElem.position().top); - } + $scope.keyDownHandler = e => { + // If we've pressed any non-navigational keys, then the user has typed something and we + // can exit early without doing any navigation. + if (!isNavigationalKey(e.keyCode)) { + return; + } - function findFunction(position, functionList) { - let bestFunction; + switch (e.keyCode) { + case navigationalKeys.UP: + // Up and down keys navigate through suggestions. + e.preventDefault(); + $scope.functionSuggestions.stepForward(); + scrollTo($scope.functionSuggestions.index); + break; + + case navigationalKeys.DOWN: + // Up and down keys navigate through suggestions. + e.preventDefault(); + $scope.functionSuggestions.stepBackward(); + scrollTo($scope.functionSuggestions.index); + break; - _.each(functionList, function (func) { - if ((func.location.min) < position && position < (func.location.max)) { - if (!bestFunction || func.text.length < bestFunction.text.length) { - bestFunction = func; + case navigationalKeys.TAB: + // If there are no suggestions, the user tabs to the next input. + if ($scope.functionSuggestions.isEmpty()) { + return; } - } - }); - return bestFunction; - } + // If we have suggestions, complete the selected one. + e.preventDefault(); + completeExpression($scope.functionSuggestions.index); + break; - function getCaretPos() { - return $elem[0].selectionStart; - } + case navigationalKeys.ENTER: + // If the suggestions are open, complete the expression with the suggestion. + // Otherwise, the default action of submitting the input value will occur. + if (!$scope.functionSuggestions.isEmpty()) { + e.preventDefault(); + completeExpression($scope.functionSuggestions.index); + } + break; - function digest() { - $rootScope.$$phase || $scope.$digest(); - } + case navigationalKeys.ESC: + e.preventDefault(); + $scope.functionSuggestions.hide(); + break; + } + }; - init(); + $scope.keyUpHandler = e => { + // If the user isn't navigating, then we should update the suggestions based on their input. + if (!isNavigationalKey(e.keyCode)) { + getSuggestions(); + } + }; + init(); } }; }); diff --git a/src/core_plugins/timelion/public/directives/expression_directive_helpers.js b/src/core_plugins/timelion/public/directives/expression_directive_helpers.js new file mode 100644 index 0000000000000..7537f060ab666 --- /dev/null +++ b/src/core_plugins/timelion/public/directives/expression_directive_helpers.js @@ -0,0 +1,113 @@ +import _ from 'lodash'; + +export class FunctionSuggestions { + constructor() { + this.reset(); + } + + reset() { + this.index = 0; + this.list = []; + this.isVisible = false; + } + + setList(list) { + this.list = list; + + // We may get a shorter list than the one we have now, so we need to make sure our index doesn't + // fall outside of the new list's range. + this.index = Math.max(0, Math.min(this.index, this.list.length - 1)); + } + + getCount() { + return this.list.length; + } + + isEmpty() { + return this.list.length === 0; + } + + show() { + this.isVisible = true; + } + + hide() { + this.isVisible = false; + } + + stepForward() { + if (this.index > 0) { + this.index -= 1; + } + } + + stepBackward() { + if (this.index < this.list.length - 1) { + this.index += 1; + } + } +} + +export function findFunction(position, functionList) { + let matchingFunction; + + functionList.forEach(func => { + if ((func.location.min) < position && position < (func.location.max)) { + if (!matchingFunction || func.text.length < matchingFunction.text.length) { + matchingFunction = func; + } + } + }); + + return matchingFunction; +} + +export function suggest(val, caretPosition, functionList, Parser) { + return new Promise((resolve, reject) => { + try { + // Inside an existing function providing suggestion only as a reference. Maybe suggest an argument? + findFunction(caretPosition, Parser.parse(val).functions); + + // TODO: Reference suggestors. Only supporting completion right now; + + // We rely on peg to throw an error in order to suggest function(s). If peg doesn't + // throw an error, then we have no suggestions to offer. + return reject(); + } catch (e) { + // NOTE: This is a peg SyntaxError. + try { // Is this a structured exception? + const error = JSON.parse(e.message); + const location = error.location; + + if (location.min > caretPosition || location.max <= caretPosition) { + return reject({ location }); + } + // TODO: Abstract structured exception handling; + if (error.type === 'incompleteFunction') { + if (error.function == null) { + return resolve({ + list: functionList, + location, + }); + } else { + const list = _.compact(_.map(functionList, func => { + if (_.startsWith(func.name, error.function)) { + return func; + } + })); + return resolve({ list, location }); + } + } + } catch (e) { + return reject(); + } + } + }); +} + +export function insertAtLocation(value, destination, min, max) { + // Insert the value at a location caret within the destination. + const startOf = destination.slice(0, min); + const endOf = destination.slice(max, destination.length); + return `${startOf}${value}${endOf}`; +} diff --git a/src/core_plugins/timelion/public/directives/interval/interval.html b/src/core_plugins/timelion/public/directives/interval/interval.html index 557f84dd41417..4e7014ed0addb 100644 --- a/src/core_plugins/timelion/public/directives/interval/interval.html +++ b/src/core_plugins/timelion/public/directives/interval/interval.html @@ -1,8 +1,13 @@ - + - + diff --git a/src/core_plugins/timelion/public/directives/interval/interval.js b/src/core_plugins/timelion/public/directives/interval/interval.js index 8787192a61aee..d7fc360913285 100644 --- a/src/core_plugins/timelion/public/directives/interval/interval.js +++ b/src/core_plugins/timelion/public/directives/interval/interval.js @@ -8,7 +8,11 @@ app.directive('timelionInterval', function ($compile, $timeout) { return { restrict: 'E', scope: { - model: '=', // The interval model + // The interval model + model: '=', + // Differentiates between contexts, e.g. when this directive is used in the header in the + // Timelion app or in the sidebar in the Visualize app. + inHeader: '=', }, template: html, link: function ($scope, $elem) { diff --git a/src/core_plugins/timelion/public/directives/partials/suggestion.html b/src/core_plugins/timelion/public/directives/partials/suggestion.html index 35bf083587a16..054bfc6cfacce 100644 --- a/src/core_plugins/timelion/public/directives/partials/suggestion.html +++ b/src/core_plugins/timelion/public/directives/partials/suggestion.html @@ -1,8 +1,14 @@ -
-
+
+ ng-class="{active: $index === selectedIndex}" + ng-repeat="suggestion in suggestions track by $index | orderBy:'name'" + >

.{{suggestion.name}}() @@ -12,7 +18,7 @@

-
+
Arguments: {{arg.name}}=({{arg.types.join(' | ')}}) @@ -20,7 +26,7 @@

-
+
@@ -36,4 +42,4 @@

- \ No newline at end of file + diff --git a/src/core_plugins/timelion/public/directives/partials/suggestion.js b/src/core_plugins/timelion/public/directives/partials/suggestion.js new file mode 100644 index 0000000000000..f40bdacd9f0b8 --- /dev/null +++ b/src/core_plugins/timelion/public/directives/partials/suggestion.js @@ -0,0 +1,18 @@ +import timelionExpressionSuggestionsTemplate from './suggestion.html'; + +const app = require('ui/modules').get('apps/timelion', []); + +app.directive('timelionExpressionSuggestions', () => { + return { + restrict: 'E', + scope: { + suggestions: '=', + selectedIndex: '=', + onClickSuggestion: '&', + }, + replace: true, + template: timelionExpressionSuggestionsTemplate, + link: function () { + } + }; +}); diff --git a/src/core_plugins/timelion/public/index.html b/src/core_plugins/timelion/public/index.html index 2eed7996b7ad7..cb372f382c023 100644 --- a/src/core_plugins/timelion/public/index.html +++ b/src/core_plugins/timelion/public/index.html @@ -24,17 +24,14 @@ ng-submit="search()" >
- + - +

Argument Name