/**
* Get tag name suggestions from scanning existing tag names.
+ * If no search term is given the 50 most popular tag names are provided.
* @param $searchTerm
* @return array
*/
- public function getNameSuggestions($searchTerm)
+ public function getNameSuggestions($searchTerm = false)
{
- if ($searchTerm === '') return [];
- $query = $this->tag->where('name', 'LIKE', $searchTerm . '%')->groupBy('name')->orderBy('name', 'desc');
+ $query = $this->tag->select('*', \DB::raw('count(*) as count'))->groupBy('name');
+
+ if ($searchTerm) {
+ $query = $query->where('name', 'LIKE', $searchTerm . '%')->orderBy('name', 'desc');
+ } else {
+ $query = $query->orderBy('count', 'desc')->take(50);
+ }
+
$query = $this->permissionService->filterRestrictedEntityRelations($query, 'tags', 'entity_id', 'entity_type');
return $query->get(['name'])->pluck('name');
}
/**
* Get tag value suggestions from scanning existing tag values.
+ * If no search is given the 50 most popular values are provided.
+ * Passing a tagName will only find values for a tags with a particular name.
* @param $searchTerm
* @param $tagName
* @return array
*/
- public function getValueSuggestions($searchTerm, $tagName = false)
+ public function getValueSuggestions($searchTerm = false, $tagName = false)
{
- if ($searchTerm === '') return [];
- $query = $this->tag->where('value', 'LIKE', $searchTerm . '%')->groupBy('value')->orderBy('value', 'desc');
- if ($tagName !== false) {
- $query = $query->where('name', '=', $tagName);
+ $query = $this->tag->select('*', \DB::raw('count(*) as count'))->groupBy('value');
+
+ if ($searchTerm) {
+ $query = $query->where('value', 'LIKE', $searchTerm . '%')->orderBy('value', 'desc');
+ } else {
+ $query = $query->orderBy('count', 'desc')->take(50);
}
+
+ if ($tagName !== false) $query = $query->where('name', '=', $tagName);
+
$query = $this->permissionService->filterRestrictedEntityRelations($query, 'tags', 'entity_id', 'entity_type');
return $query->get(['value'])->pluck('value');
}
-
+
/**
* Save an array of tags to an entity
* @param Entity $entity
};
}]);
- ngApp.directive('tinymce', ['$timeout', function($timeout) {
+ ngApp.directive('tinymce', ['$timeout', function ($timeout) {
return {
restrict: 'A',
scope: {
scope.tinymce.extraSetups.push(tinyMceSetup);
// Custom tinyMCE plugins
- tinymce.PluginManager.add('customhr', function(editor) {
- editor.addCommand('InsertHorizontalRule', function() {
+ tinymce.PluginManager.add('customhr', function (editor) {
+ editor.addCommand('InsertHorizontalRule', function () {
var hrElem = document.createElement('hr');
var cNode = editor.selection.getNode();
var parentNode = cNode.parentNode;
}
}]);
- ngApp.directive('markdownInput', ['$timeout', function($timeout) {
+ ngApp.directive('markdownInput', ['$timeout', function ($timeout) {
return {
restrict: 'A',
scope: {
scope.$on('markdown-update', (event, value) => {
element.val(value);
- scope.mdModel= value;
+ scope.mdModel = value;
scope.mdChange(markdown(value));
});
}
}]);
- ngApp.directive('markdownEditor', ['$timeout', function($timeout) {
+ ngApp.directive('markdownEditor', ['$timeout', function ($timeout) {
return {
restrict: 'A',
link: function (scope, element, attrs) {
if (now - lastScroll > scrollDebounceTime) {
setScrollHeights()
}
- let scrollPercent = (input.scrollTop() / (inputScrollHeight-inputHeight));
+ let scrollPercent = (input.scrollTop() / (inputScrollHeight - inputHeight));
let displayScrollY = (displayScrollHeight - displayHeight) * scrollPercent;
display.scrollTop(displayScrollY);
lastScroll = now;
}
}
}]);
-
- ngApp.directive('toolbox', [function() {
+
+ ngApp.directive('toolbox', [function () {
return {
restrict: 'A',
- link: function(scope, elem, attrs) {
+ link: function (scope, elem, attrs) {
// Get common elements
const $buttons = elem.find('[tab-button]');
$toggle.click((e) => {
elem.toggleClass('open');
});
-
+
// Set an active tab/content by name
function setActive(tabName, openToolbox) {
$buttons.removeClass('active');
setActive($content.first().attr('tab-content'), false);
// Handle tab button click
- $buttons.click(function(e) {
+ $buttons.click(function (e) {
let name = $(this).attr('tab-button');
setActive(name, true);
});
}
}]);
- ngApp.directive('tagAutosuggestions', ['$http', function($http) {
+ ngApp.directive('tagAutosuggestions', ['$http', function ($http) {
return {
restrict: 'A',
- link: function(scope, elem, attrs) {
-
+ link: function (scope, elem, attrs) {
+
// Local storage for quick caching.
const localCache = {};
let active = 0;
// Listen to input events on autosuggest fields
- elem.on('input', '[autosuggest]', function(event) {
+ elem.on('input focus', '[autosuggest]', function (event) {
let $input = $(this);
let val = $input.val();
let url = $input.attr('autosuggest');
let type = $input.attr('autosuggest-type');
- // No suggestions until at least 3 chars
- if (val.length < 3) {
- if (isShowing) {
- $suggestionBox.hide();
- isShowing = false;
- }
- return;
- }
-
// Add name param to request if for a value
if (type.toLowerCase() === 'value') {
let $nameInput = $input.closest('tr').find('[autosuggest-type="name"]').first();
let nameVal = $nameInput.val();
- if (nameVal === '') return;
- url += '?name=' + encodeURIComponent(nameVal);
- console.log(url);
+ if (nameVal !== '') {
+ url += '?name=' + encodeURIComponent(nameVal);
+ }
}
let suggestionPromise = getSuggestions(val.slice(0, 3), url);
suggestionPromise.then(suggestions => {
- if (val.length > 2) {
- suggestions = suggestions.filter(item => {
- return item.toLowerCase().indexOf(val.toLowerCase()) !== -1;
- }).slice(0, 4);
- displaySuggestions($input, suggestions);
- }
+ if (val.length === 0) {
+ displaySuggestions($input, suggestions.slice(0, 6));
+ } else {
+ suggestions = suggestions.filter(item => {
+ return item.toLowerCase().indexOf(val.toLowerCase()) !== -1;
+ }).slice(0, 4);
+ displaySuggestions($input, suggestions);
+ }
});
});
// Hide autosuggestions when input loses focus.
// Slight delay to allow clicks.
- elem.on('blur', '[autosuggest]', function(event) {
+ let lastFocusTime = 0;
+ elem.on('blur', '[autosuggest]', function (event) {
+ let startTime = Date.now();
setTimeout(() => {
- $suggestionBox.hide();
- isShowing = false;
+ if (lastFocusTime < startTime) {
+ $suggestionBox.hide();
+ isShowing = false;
+ }
}, 200)
});
+ elem.on('focus', '[autosuggest]', function (event) {
+ lastFocusTime = Date.now();
+ });
elem.on('keydown', '[autosuggest]', function (event) {
if (!isShowing) return;
// Down arrow
if (event.keyCode === 40) {
- let newActive = (active === suggestCount-1) ? 0 : active + 1;
+ let newActive = (active === suggestCount - 1) ? 0 : active + 1;
changeActiveTo(newActive, suggestionElems);
}
// Up arrow
else if (event.keyCode === 38) {
- let newActive = (active === 0) ? suggestCount-1 : active - 1;
+ let newActive = (active === 0) ? suggestCount - 1 : active - 1;
changeActiveTo(newActive, suggestionElems);
}
// Enter or tab key
// Display suggestions on a field
let prevSuggestions = [];
+
function displaySuggestions($input, suggestions) {
// Hide if no suggestions
if (i === 0) {
suggestion.className = 'active'
active = 0;
- };
+ }
+ ;
$suggestionBox[0].appendChild(suggestion);
}
// Get suggestions & cache
function getSuggestions(input, url) {
let hasQuery = url.indexOf('?') !== -1;
- let searchUrl = url + (hasQuery?'&':'?') + 'search=' + encodeURIComponent(input);
+ let searchUrl = url + (hasQuery ? '&' : '?') + 'search=' + encodeURIComponent(input);
// Get from local cache if exists
- if (localCache[searchUrl]) {
+ if (typeof localCache[searchUrl] !== 'undefined') {
return new Promise((resolve, reject) => {
- resolve(localCache[input]);
+ resolve(localCache[searchUrl]);
});
}
- return $http.get(searchUrl).then((response) => {
- localCache[input] = response.data;
+ return $http.get(searchUrl).then(response => {
+ localCache[searchUrl] = response.data;
return response.data;
});
}