diff options
author | ArthurHoaro <arthur@hoa.ro> | 2017-03-25 15:59:01 +0100 |
---|---|---|
committer | ArthurHoaro <arthur@hoa.ro> | 2017-05-25 15:25:04 +0200 |
commit | aa4797ba3679b847adc895e2f817ac058779a171 (patch) | |
tree | d854a1ab8748911dd10e8bced31a2d9b80ccf57b | |
parent | bc988eb0420156219fdeb7af684fff37c8b33f4b (diff) | |
download | Shaarli-aa4797ba3679b847adc895e2f817ac058779a171.tar.gz Shaarli-aa4797ba3679b847adc895e2f817ac058779a171.tar.zst Shaarli-aa4797ba3679b847adc895e2f817ac058779a171.zip |
Adds a taglist view with edit/delete buttons
* The tag list can be sort alphabetically or by most used tag
* Edit/Delete are perform using AJAX, or fallback to 'do=changetag' page
* New features aren't backported to vintage theme
-rw-r--r-- | application/Router.php | 6 | ||||
-rw-r--r-- | application/Utils.php | 31 | ||||
-rw-r--r-- | index.php | 40 | ||||
-rw-r--r-- | tests/UtilsTest.php | 112 | ||||
-rw-r--r-- | tpl/default/changetag.html | 4 | ||||
-rw-r--r-- | tpl/default/css/shaarli.css | 50 | ||||
-rw-r--r-- | tpl/default/js/shaarli.js | 136 | ||||
-rw-r--r-- | tpl/default/tag.list.html | 82 | ||||
-rw-r--r-- | tpl/default/tag.sort.html | 8 |
9 files changed, 453 insertions, 16 deletions
diff --git a/application/Router.php b/application/Router.php index f6896b1c..4df0387c 100644 --- a/application/Router.php +++ b/application/Router.php | |||
@@ -13,6 +13,8 @@ class Router | |||
13 | 13 | ||
14 | public static $PAGE_TAGCLOUD = 'tagcloud'; | 14 | public static $PAGE_TAGCLOUD = 'tagcloud'; |
15 | 15 | ||
16 | public static $PAGE_TAGLIST = 'taglist'; | ||
17 | |||
16 | public static $PAGE_DAILY = 'daily'; | 18 | public static $PAGE_DAILY = 'daily'; |
17 | 19 | ||
18 | public static $PAGE_FEED_ATOM = 'atom'; | 20 | public static $PAGE_FEED_ATOM = 'atom'; |
@@ -79,6 +81,10 @@ class Router | |||
79 | return self::$PAGE_TAGCLOUD; | 81 | return self::$PAGE_TAGCLOUD; |
80 | } | 82 | } |
81 | 83 | ||
84 | if (startsWith($query, 'do='. self::$PAGE_TAGLIST)) { | ||
85 | return self::$PAGE_TAGLIST; | ||
86 | } | ||
87 | |||
82 | if (startsWith($query, 'do='. self::$PAGE_OPENSEARCH)) { | 88 | if (startsWith($query, 'do='. self::$PAGE_OPENSEARCH)) { |
83 | return self::$PAGE_OPENSEARCH; | 89 | return self::$PAGE_OPENSEARCH; |
84 | } | 90 | } |
diff --git a/application/Utils.php b/application/Utils.php index ab463af9..9d0ebc5e 100644 --- a/application/Utils.php +++ b/application/Utils.php | |||
@@ -435,3 +435,34 @@ function get_max_upload_size($limitPost, $limitUpload, $format = true) | |||
435 | $maxsize = min($size1, $size2); | 435 | $maxsize = min($size1, $size2); |
436 | return $format ? human_bytes($maxsize) : $maxsize; | 436 | return $format ? human_bytes($maxsize) : $maxsize; |
437 | } | 437 | } |
438 | |||
439 | /** | ||
440 | * Sort the given array alphabetically using php-intl if available. | ||
441 | * Case sensitive. | ||
442 | * | ||
443 | * Note: doesn't support multidimensional arrays | ||
444 | * | ||
445 | * @param array $data Input array, passed by reference | ||
446 | * @param bool $reverse Reverse sort if set to true | ||
447 | * @param bool $byKeys Sort the array by keys if set to true, by value otherwise. | ||
448 | */ | ||
449 | function alphabetical_sort(&$data, $reverse = false, $byKeys = false) | ||
450 | { | ||
451 | $callback = function($a, $b) use ($reverse) { | ||
452 | // Collator is part of PHP intl. | ||
453 | if (class_exists('Collator')) { | ||
454 | $collator = new Collator(setlocale(LC_COLLATE, 0)); | ||
455 | if (!intl_is_failure(intl_get_error_code())) { | ||
456 | return $collator->compare($a, $b) * ($reverse ? -1 : 1); | ||
457 | } | ||
458 | } | ||
459 | |||
460 | return strcasecmp($a, $b) * ($reverse ? -1 : 1); | ||
461 | }; | ||
462 | |||
463 | if ($byKeys) { | ||
464 | uksort($data, $callback); | ||
465 | } else { | ||
466 | usort($data, $callback); | ||
467 | } | ||
468 | } | ||
@@ -791,7 +791,7 @@ function renderPage($conf, $pluginManager, $LINKSDB, $history) | |||
791 | if ($targetPage == Router::$PAGE_TAGCLOUD) | 791 | if ($targetPage == Router::$PAGE_TAGCLOUD) |
792 | { | 792 | { |
793 | $visibility = ! empty($_SESSION['privateonly']) ? 'private' : 'all'; | 793 | $visibility = ! empty($_SESSION['privateonly']) ? 'private' : 'all'; |
794 | $filteringTags = isset($_GET['searchtags']) ? explode(' ', $_GET['searchtags']) : array(); | 794 | $filteringTags = isset($_GET['searchtags']) ? explode(' ', $_GET['searchtags']) : []; |
795 | $tags = $LINKSDB->linksCountPerTag($filteringTags, $visibility); | 795 | $tags = $LINKSDB->linksCountPerTag($filteringTags, $visibility); |
796 | 796 | ||
797 | // We sort tags alphabetically, then choose a font size according to count. | 797 | // We sort tags alphabetically, then choose a font size according to count. |
@@ -801,17 +801,7 @@ function renderPage($conf, $pluginManager, $LINKSDB, $history) | |||
801 | $maxcount = max($maxcount, $value); | 801 | $maxcount = max($maxcount, $value); |
802 | } | 802 | } |
803 | 803 | ||
804 | // Sort tags alphabetically: case insensitive, support locale if available. | 804 | alphabetical_sort($tags, true, true); |
805 | uksort($tags, function($a, $b) { | ||
806 | // Collator is part of PHP intl. | ||
807 | if (class_exists('Collator')) { | ||
808 | $c = new Collator(setlocale(LC_COLLATE, 0)); | ||
809 | if (!intl_is_failure(intl_get_error_code())) { | ||
810 | return $c->compare($a, $b); | ||
811 | } | ||
812 | } | ||
813 | return strcasecmp($a, $b); | ||
814 | }); | ||
815 | 805 | ||
816 | $tagList = array(); | 806 | $tagList = array(); |
817 | foreach($tags as $key => $value) { | 807 | foreach($tags as $key => $value) { |
@@ -839,6 +829,31 @@ function renderPage($conf, $pluginManager, $LINKSDB, $history) | |||
839 | exit; | 829 | exit; |
840 | } | 830 | } |
841 | 831 | ||
832 | // -------- Tag cloud | ||
833 | if ($targetPage == Router::$PAGE_TAGLIST) | ||
834 | { | ||
835 | $visibility = ! empty($_SESSION['privateonly']) ? 'private' : 'all'; | ||
836 | $filteringTags = isset($_GET['searchtags']) ? explode(' ', $_GET['searchtags']) : []; | ||
837 | $tags = $LINKSDB->linksCountPerTag($filteringTags, $visibility); | ||
838 | |||
839 | if (! empty($_GET['sort']) && $_GET['sort'] === 'alpha') { | ||
840 | alphabetical_sort($tags, false, true); | ||
841 | } | ||
842 | |||
843 | $data = [ | ||
844 | 'search_tags' => implode(' ', $filteringTags), | ||
845 | 'tags' => $tags, | ||
846 | ]; | ||
847 | $pluginManager->executeHooks('render_taglist', $data, ['loggedin' => isLoggedIn()]); | ||
848 | |||
849 | foreach ($data as $key => $value) { | ||
850 | $PAGE->assign($key, $value); | ||
851 | } | ||
852 | |||
853 | $PAGE->renderPage('tag.list'); | ||
854 | exit; | ||
855 | } | ||
856 | |||
842 | // Daily page. | 857 | // Daily page. |
843 | if ($targetPage == Router::$PAGE_DAILY) { | 858 | if ($targetPage == Router::$PAGE_DAILY) { |
844 | showDaily($PAGE, $LINKSDB, $conf, $pluginManager); | 859 | showDaily($PAGE, $LINKSDB, $conf, $pluginManager); |
@@ -1152,6 +1167,7 @@ function renderPage($conf, $pluginManager, $LINKSDB, $history) | |||
1152 | if ($targetPage == Router::$PAGE_CHANGETAG) | 1167 | if ($targetPage == Router::$PAGE_CHANGETAG) |
1153 | { | 1168 | { |
1154 | if (empty($_POST['fromtag']) || (empty($_POST['totag']) && isset($_POST['renametag']))) { | 1169 | if (empty($_POST['fromtag']) || (empty($_POST['totag']) && isset($_POST['renametag']))) { |
1170 | $PAGE->assign('fromtag', ! empty($_GET['fromtag']) ? escape($_GET['fromtag']) : ''); | ||
1155 | $PAGE->renderPage('changetag'); | 1171 | $PAGE->renderPage('changetag'); |
1156 | exit; | 1172 | exit; |
1157 | } | 1173 | } |
diff --git a/tests/UtilsTest.php b/tests/UtilsTest.php index d6a0aad5..3d1aa653 100644 --- a/tests/UtilsTest.php +++ b/tests/UtilsTest.php | |||
@@ -417,4 +417,116 @@ class UtilsTest extends PHPUnit_Framework_TestCase | |||
417 | $this->assertEquals('1048576', get_max_upload_size('1m', '2m', false)); | 417 | $this->assertEquals('1048576', get_max_upload_size('1m', '2m', false)); |
418 | $this->assertEquals('100', get_max_upload_size(100, 100, false)); | 418 | $this->assertEquals('100', get_max_upload_size(100, 100, false)); |
419 | } | 419 | } |
420 | |||
421 | /** | ||
422 | * Test alphabetical_sort by value, not reversed, with php-intl. | ||
423 | */ | ||
424 | public function testAlphabeticalSortByValue() | ||
425 | { | ||
426 | $arr = [ | ||
427 | 'zZz', | ||
428 | 'éee', | ||
429 | 'éae', | ||
430 | 'eee', | ||
431 | 'A', | ||
432 | 'a', | ||
433 | 'zzz', | ||
434 | ]; | ||
435 | $expected = [ | ||
436 | 'a', | ||
437 | 'A', | ||
438 | 'éae', | ||
439 | 'eee', | ||
440 | 'éee', | ||
441 | 'zzz', | ||
442 | 'zZz', | ||
443 | ]; | ||
444 | |||
445 | alphabetical_sort($arr); | ||
446 | $this->assertEquals($expected, $arr); | ||
447 | } | ||
448 | |||
449 | /** | ||
450 | * Test alphabetical_sort by value, reversed, with php-intl. | ||
451 | */ | ||
452 | public function testAlphabeticalSortByValueReversed() | ||
453 | { | ||
454 | $arr = [ | ||
455 | 'zZz', | ||
456 | 'éee', | ||
457 | 'éae', | ||
458 | 'eee', | ||
459 | 'A', | ||
460 | 'a', | ||
461 | 'zzz', | ||
462 | ]; | ||
463 | $expected = [ | ||
464 | 'zZz', | ||
465 | 'zzz', | ||
466 | 'éee', | ||
467 | 'eee', | ||
468 | 'éae', | ||
469 | 'A', | ||
470 | 'a', | ||
471 | ]; | ||
472 | |||
473 | alphabetical_sort($arr, true); | ||
474 | $this->assertEquals($expected, $arr); | ||
475 | } | ||
476 | |||
477 | /** | ||
478 | * Test alphabetical_sort by keys, not reversed, with php-intl. | ||
479 | */ | ||
480 | public function testAlphabeticalSortByKeys() | ||
481 | { | ||
482 | $arr = [ | ||
483 | 'zZz' => true, | ||
484 | 'éee' => true, | ||
485 | 'éae' => true, | ||
486 | 'eee' => true, | ||
487 | 'A' => true, | ||
488 | 'a' => true, | ||
489 | 'zzz' => true, | ||
490 | ]; | ||
491 | $expected = [ | ||
492 | 'a' => true, | ||
493 | 'A' => true, | ||
494 | 'éae' => true, | ||
495 | 'eee' => true, | ||
496 | 'éee' => true, | ||
497 | 'zzz' => true, | ||
498 | 'zZz' => true, | ||
499 | ]; | ||
500 | |||
501 | alphabetical_sort($arr, true, true); | ||
502 | $this->assertEquals($expected, $arr); | ||
503 | } | ||
504 | |||
505 | /** | ||
506 | * Test alphabetical_sort by keys, reversed, with php-intl. | ||
507 | */ | ||
508 | public function testAlphabeticalSortByKeysReversed() | ||
509 | { | ||
510 | $arr = [ | ||
511 | 'zZz' => true, | ||
512 | 'éee' => true, | ||
513 | 'éae' => true, | ||
514 | 'eee' => true, | ||
515 | 'A' => true, | ||
516 | 'a' => true, | ||
517 | 'zzz' => true, | ||
518 | ]; | ||
519 | $expected = [ | ||
520 | 'zZz' => true, | ||
521 | 'zzz' => true, | ||
522 | 'éee' => true, | ||
523 | 'eee' => true, | ||
524 | 'éae' => true, | ||
525 | 'A' => true, | ||
526 | 'a' => true, | ||
527 | ]; | ||
528 | |||
529 | alphabetical_sort($arr, true, true); | ||
530 | $this->assertEquals($expected, $arr); | ||
531 | } | ||
420 | } | 532 | } |
diff --git a/tpl/default/changetag.html b/tpl/default/changetag.html index 8d263a16..49dd20d9 100644 --- a/tpl/default/changetag.html +++ b/tpl/default/changetag.html | |||
@@ -11,7 +11,7 @@ | |||
11 | <h2 class="window-title">{"Manage tags"|t}</h2> | 11 | <h2 class="window-title">{"Manage tags"|t}</h2> |
12 | <form method="POST" action="#" name="changetag" id="changetag"> | 12 | <form method="POST" action="#" name="changetag" id="changetag"> |
13 | <div> | 13 | <div> |
14 | <input type="text" name="fromtag" placeholder="{'Tag'|t}" | 14 | <input type="text" name="fromtag" placeholder="{'Tag'|t}" value="{$fromtag}" |
15 | list="tagsList" autocomplete="off" class="awesomplete autofocus" data-minChars="1"> | 15 | list="tagsList" autocomplete="off" class="awesomplete autofocus" data-minChars="1"> |
16 | <datalist id="tagsList"> | 16 | <datalist id="tagsList"> |
17 | {loop="$tags"}<option>{$key}</option>{/loop} | 17 | {loop="$tags"}<option>{$key}</option>{/loop} |
@@ -31,6 +31,8 @@ | |||
31 | <input type="submit" value="{'Delete'|t}" name="deletetag" class="button button-red confirm-delete"> | 31 | <input type="submit" value="{'Delete'|t}" name="deletetag" class="button button-red confirm-delete"> |
32 | </div> | 32 | </div> |
33 | </form> | 33 | </form> |
34 | |||
35 | <p>You can also edit tags in the <a href="?do=taglist&sort=usage">tag list</a>.</p> | ||
34 | </div> | 36 | </div> |
35 | </div> | 37 | </div> |
36 | {include="page.footer"} | 38 | {include="page.footer"} |
diff --git a/tpl/default/css/shaarli.css b/tpl/default/css/shaarli.css index 4415a1b7..2eda5df4 100644 --- a/tpl/default/css/shaarli.css +++ b/tpl/default/css/shaarli.css | |||
@@ -751,10 +751,11 @@ body, .pure-g [class*="pure-u"] { | |||
751 | .page-form a { | 751 | .page-form a { |
752 | color: #1b926c; | 752 | color: #1b926c; |
753 | font-weight: bold; | 753 | font-weight: bold; |
754 | text-decoration: none; | ||
754 | } | 755 | } |
755 | 756 | ||
756 | .page-form p { | 757 | .page-form p { |
757 | padding: 0 10px; | 758 | padding: 5px 10px; |
758 | margin: 0; | 759 | margin: 0; |
759 | } | 760 | } |
760 | 761 | ||
@@ -1070,7 +1071,7 @@ form[name="linkform"].page-form { | |||
1070 | } | 1071 | } |
1071 | 1072 | ||
1072 | #cloudtag, #cloudtag a { | 1073 | #cloudtag, #cloudtag a { |
1073 | color: #000; | 1074 | color: #252525; |
1074 | text-decoration: none; | 1075 | text-decoration: none; |
1075 | } | 1076 | } |
1076 | 1077 | ||
@@ -1079,6 +1080,38 @@ form[name="linkform"].page-form { | |||
1079 | } | 1080 | } |
1080 | 1081 | ||
1081 | /** | 1082 | /** |
1083 | * TAG LIST | ||
1084 | */ | ||
1085 | #taglist { | ||
1086 | padding: 0 10px; | ||
1087 | } | ||
1088 | |||
1089 | #taglist a { | ||
1090 | color: #252525; | ||
1091 | text-decoration: none; | ||
1092 | } | ||
1093 | |||
1094 | #taglist .count { | ||
1095 | display: inline-block; | ||
1096 | width: 35px; | ||
1097 | text-align: right; | ||
1098 | color: #7f7f7f; | ||
1099 | } | ||
1100 | |||
1101 | #taglist .delete-tag { | ||
1102 | color: #ac2925; | ||
1103 | display: none; | ||
1104 | } | ||
1105 | |||
1106 | #taglist .rename-tag { | ||
1107 | color: #0b5ea6; | ||
1108 | } | ||
1109 | |||
1110 | #taglist .validate-rename-tag { | ||
1111 | color: #1b926c; | ||
1112 | } | ||
1113 | |||
1114 | /** | ||
1082 | * Picture wall CSS | 1115 | * Picture wall CSS |
1083 | */ | 1116 | */ |
1084 | #picwall_container { | 1117 | #picwall_container { |
@@ -1227,3 +1260,16 @@ form[name="linkform"].page-form { | |||
1227 | .pure-button { | 1260 | .pure-button { |
1228 | -moz-user-select: auto; | 1261 | -moz-user-select: auto; |
1229 | } | 1262 | } |
1263 | |||
1264 | .tag-sort { | ||
1265 | margin-top: 30px; | ||
1266 | text-align: center; | ||
1267 | } | ||
1268 | |||
1269 | .tag-sort a { | ||
1270 | display: inline-block; | ||
1271 | margin: 0 15px; | ||
1272 | color: white; | ||
1273 | text-decoration: none; | ||
1274 | font-weight: bold; | ||
1275 | } | ||
diff --git a/tpl/default/js/shaarli.js b/tpl/default/js/shaarli.js index ceb1d1b8..e19e9001 100644 --- a/tpl/default/js/shaarli.js +++ b/tpl/default/js/shaarli.js | |||
@@ -412,8 +412,139 @@ window.onload = function () { | |||
412 | } | 412 | } |
413 | }); | 413 | }); |
414 | } | 414 | } |
415 | |||
416 | /** | ||
417 | * Tag list operations | ||
418 | * | ||
419 | * TODO: support error code in the backend for AJAX requests | ||
420 | */ | ||
421 | // Display/Hide rename form | ||
422 | var renameTagButtons = document.querySelectorAll('.rename-tag'); | ||
423 | [].forEach.call(renameTagButtons, function(rename) { | ||
424 | rename.addEventListener('click', function(event) { | ||
425 | event.preventDefault(); | ||
426 | var block = findParent(event.target, 'div', {'class': 'tag-list-item'}); | ||
427 | var form = block.querySelector('.rename-tag-form'); | ||
428 | form.style.display = form.style.display == 'none' ? 'block' : 'none'; | ||
429 | }); | ||
430 | }); | ||
431 | |||
432 | // Rename a tag with an AJAX request | ||
433 | var renameTagSubmits = document.querySelectorAll('.validate-rename-tag'); | ||
434 | [].forEach.call(renameTagSubmits, function(rename) { | ||
435 | rename.addEventListener('click', function(event) { | ||
436 | event.preventDefault(); | ||
437 | var block = findParent(event.target, 'div', {'class': 'tag-list-item'}); | ||
438 | var input = block.querySelector('.rename-tag-input'); | ||
439 | var totag = input.value.replace('/"/g', '\\"'); | ||
440 | if (totag.trim() == '') { | ||
441 | return; | ||
442 | } | ||
443 | var fromtag = block.getAttribute('data-tag'); | ||
444 | var token = document.getElementById('token').value; | ||
445 | |||
446 | xhr = new XMLHttpRequest(); | ||
447 | xhr.open('POST', '?do=changetag'); | ||
448 | xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded'); | ||
449 | xhr.onload = function() { | ||
450 | if (xhr.status !== 200) { | ||
451 | alert('An error occurred. Return code: '+ xhr.status); | ||
452 | location.reload(); | ||
453 | } else { | ||
454 | block.setAttribute('data-tag', totag); | ||
455 | input.setAttribute('name', totag); | ||
456 | input.setAttribute('value', totag); | ||
457 | input.parentNode.style.display = 'none'; | ||
458 | block.querySelector('a.tag-link').innerHTML = htmlEntities(totag); | ||
459 | block.querySelector('a.tag-link').setAttribute('href', '?searchtags='+ encodeURIComponent(totag)); | ||
460 | block.querySelector('a.rename-tag').setAttribute('href', '?do=changetag&fromtag='+ encodeURIComponent(totag)); | ||
461 | } | ||
462 | }; | ||
463 | xhr.send('renametag=1&fromtag='+ encodeURIComponent(fromtag) +'&totag='+ encodeURIComponent(totag) +'&token='+ token); | ||
464 | refreshToken(); | ||
465 | }); | ||
466 | }); | ||
467 | |||
468 | // Validate input with enter key | ||
469 | var renameTagInputs = document.querySelectorAll('.rename-tag-input'); | ||
470 | [].forEach.call(renameTagInputs, function(rename) { | ||
471 | rename.addEventListener('keypress', function(event) { | ||
472 | if (event.keyCode === 13) { // enter | ||
473 | findParent(event.target, 'div', {'class': 'tag-list-item'}).querySelector('.validate-rename-tag').click(); | ||
474 | } | ||
475 | }); | ||
476 | }); | ||
477 | |||
478 | // Delete a tag with an AJAX query (alert popup confirmation) | ||
479 | var deleteTagButtons = document.querySelectorAll('.delete-tag'); | ||
480 | [].forEach.call(deleteTagButtons, function(rename) { | ||
481 | rename.style.display = 'inline'; | ||
482 | rename.addEventListener('click', function(event) { | ||
483 | event.preventDefault(); | ||
484 | var block = findParent(event.target, 'div', {'class': 'tag-list-item'}); | ||
485 | var tag = block.getAttribute('data-tag'); | ||
486 | var token = document.getElementById('token').value; | ||
487 | |||
488 | if (confirm('Are you sure you want to delete the tag "'+ tag +'"?')) { | ||
489 | xhr = new XMLHttpRequest(); | ||
490 | xhr.open('POST', '?do=changetag'); | ||
491 | xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded'); | ||
492 | xhr.onload = function() { | ||
493 | block.remove(); | ||
494 | }; | ||
495 | xhr.send(encodeURI('deletetag=1&fromtag='+ tag +'&token='+ token)); | ||
496 | refreshToken(); | ||
497 | } | ||
498 | }); | ||
499 | }); | ||
415 | }; | 500 | }; |
416 | 501 | ||
502 | function findParent(element, tagName, attributes) | ||
503 | { | ||
504 | while (element) { | ||
505 | if (element.tagName.toLowerCase() == tagName) { | ||
506 | var match = true; | ||
507 | for (var key in attributes) { | ||
508 | if (! element.hasAttribute(key) | ||
509 | || (attributes[key] != '' && element.getAttribute(key).indexOf(attributes[key]) == -1) | ||
510 | ) { | ||
511 | match = false; | ||
512 | break; | ||
513 | } | ||
514 | } | ||
515 | |||
516 | if (match) { | ||
517 | return element; | ||
518 | } | ||
519 | } | ||
520 | element = element.parentElement; | ||
521 | } | ||
522 | return null; | ||
523 | } | ||
524 | |||
525 | function refreshToken() | ||
526 | { | ||
527 | var xhr = new XMLHttpRequest(); | ||
528 | xhr.open('GET', '?do=token'); | ||
529 | xhr.onload = function() { | ||
530 | var token = document.getElementById('token'); | ||
531 | token.setAttribute('value', xhr.responseText); | ||
532 | }; | ||
533 | xhr.send(); | ||
534 | } | ||
535 | |||
536 | /** | ||
537 | * html_entities in JS | ||
538 | * | ||
539 | * @see http://stackoverflow.com/questions/18749591/encode-html-entities-in-javascript | ||
540 | */ | ||
541 | function htmlEntities(str) | ||
542 | { | ||
543 | return str.replace(/[\u00A0-\u9999<>\&]/gim, function(i) { | ||
544 | return '&#'+i.charCodeAt(0)+';'; | ||
545 | }); | ||
546 | } | ||
547 | |||
417 | function activateFirefoxSocial(node) { | 548 | function activateFirefoxSocial(node) { |
418 | var loc = location.href; | 549 | var loc = location.href; |
419 | var baseURL = loc.substring(0, loc.lastIndexOf("/")); | 550 | var baseURL = loc.substring(0, loc.lastIndexOf("/")); |
@@ -445,8 +576,11 @@ function activateFirefoxSocial(node) { | |||
445 | * @param currentContinent Current selected continent | 576 | * @param currentContinent Current selected continent |
446 | * @param reset Set to true to reset the selected value | 577 | * @param reset Set to true to reset the selected value |
447 | */ | 578 | */ |
448 | function hideTimezoneCities(cities, currentContinent, reset = false) { | 579 | function hideTimezoneCities(cities, currentContinent) { |
449 | var first = true; | 580 | var first = true; |
581 | if (reset == null) { | ||
582 | reset = false; | ||
583 | } | ||
450 | [].forEach.call(cities, function (option) { | 584 | [].forEach.call(cities, function (option) { |
451 | if (option.getAttribute('data-continent') != currentContinent) { | 585 | if (option.getAttribute('data-continent') != currentContinent) { |
452 | option.className = 'hidden'; | 586 | option.className = 'hidden'; |
diff --git a/tpl/default/tag.list.html b/tpl/default/tag.list.html new file mode 100644 index 00000000..98971051 --- /dev/null +++ b/tpl/default/tag.list.html | |||
@@ -0,0 +1,82 @@ | |||
1 | <!DOCTYPE html> | ||
2 | <html> | ||
3 | <head> | ||
4 | {include="includes"} | ||
5 | </head> | ||
6 | <body> | ||
7 | {include="page.header"} | ||
8 | |||
9 | {include="tag.sort"} | ||
10 | |||
11 | <div class="pure-g"> | ||
12 | <div class="pure-u-lg-1-6 pure-u-1-24"></div> | ||
13 | <div class="pure-u-lg-2-3 pure-u-22-24 page-form page-visitor"> | ||
14 | {$countTags=count($tags)} | ||
15 | <h2 class="window-title">{'Tag list'|t} - {$countTags} {'tags'|t}</h2> | ||
16 | |||
17 | <div id="search-tagcloud" class="pure-g"> | ||
18 | <div class="pure-u-lg-1-4"></div> | ||
19 | <div class="pure-u-1 pure-u-lg-1-2"> | ||
20 | <form method="GET"> | ||
21 | <input type="hidden" name="do" value="taglist"> | ||
22 | <input type="text" name="searchtags" placeholder="{'Filter by tag'|t}" | ||
23 | {if="!empty($search_tags)"} | ||
24 | value="{$search_tags}" | ||
25 | {/if} | ||
26 | autocomplete="off" data-multiple data-autofirst data-minChars="1" | ||
27 | data-list="{loop="$tags"}{$key}, {/loop}" | ||
28 | > | ||
29 | <button type="submit" class="search-button"><i class="fa fa-search"></i></button> | ||
30 | </form> | ||
31 | </div> | ||
32 | <div class="pure-u-lg-1-4"></div> | ||
33 | </div> | ||
34 | |||
35 | <div id="plugin_zone_start_tagcloud" class="plugin_zone"> | ||
36 | {loop="$plugin_start_zone"} | ||
37 | {$value} | ||
38 | {/loop} | ||
39 | </div> | ||
40 | |||
41 | <div id="taglist"> | ||
42 | {loop="tags"} | ||
43 | <div class="tag-list-item pure-g" data-tag="{$key}"> | ||
44 | <div class="pure-u-1"> | ||
45 | {if="isLoggedIn()===true"} | ||
46 | <a href="#" class="delete-tag"><i class="fa fa-trash"></i></a> | ||
47 | <a href="?do=changetag&fromtag={$key|urlencode}" class="rename-tag"> | ||
48 | <i class="fa fa-pencil-square-o {$key}"></i> | ||
49 | </a> | ||
50 | {/if} | ||
51 | |||
52 | <a href="?addtag={$key|urlencode}" title="{'Filter by tag'|t}" class="count">{$value}</a> | ||
53 | <a href="?searchtags={$key|urlencode}" class="tag-link">{$key}</a> | ||
54 | |||
55 | {loop="$value.tag_plugin"} | ||
56 | {$value} | ||
57 | {/loop} | ||
58 | </div> | ||
59 | {if="isLoggedIn()===true"} | ||
60 | <div class="rename-tag-form pure-u-1" style="display:none;"> | ||
61 | <input type="text" name="{$key}" value="{$key}" class="rename-tag-input" /> | ||
62 | <a href="#" class="validate-rename-tag"><i class="fa fa-check"></i></a> | ||
63 | </div> | ||
64 | {/if} | ||
65 | </div> | ||
66 | {/loop} | ||
67 | </div> | ||
68 | |||
69 | <div id="plugin_zone_end_tagcloud" class="plugin_zone"> | ||
70 | {loop="$plugin_end_zone"} | ||
71 | {$value} | ||
72 | {/loop} | ||
73 | </div> | ||
74 | </div> | ||
75 | </div> | ||
76 | |||
77 | {include="tag.sort"} | ||
78 | |||
79 | {include="page.footer"} | ||
80 | </body> | ||
81 | </html> | ||
82 | |||
diff --git a/tpl/default/tag.sort.html b/tpl/default/tag.sort.html new file mode 100644 index 00000000..89acda0d --- /dev/null +++ b/tpl/default/tag.sort.html | |||
@@ -0,0 +1,8 @@ | |||
1 | <div class="pure-g"> | ||
2 | <div class="pure-u-1 pure-alert pure-alert-success tag-sort"> | ||
3 | {'Sort by:'|t} | ||
4 | <a href="?do=tagcloud" title="cloud">{'Cloud'|t}</a> · | ||
5 | <a href="?do=taglist&sort=usage" title="cloud">{'Most used'|t}</a> · | ||
6 | <a href="?do=taglist&sort=alpha" title="cloud">{'Alphabetical'|t}</a> | ||
7 | </div> | ||
8 | </div> \ No newline at end of file | ||