[UI] Sortable Tables Header By Click (#7980)
* [UI] Sortable Tables Header By Click * get rid of padding above header * restart CI * fix lint * convert getArrow JS to SortArrow go func * addopt SortArrow funct * suggestions from @silverwind - tablesort.js Co-authored-by: silverwind <me@silverwind.io> * Update web_src/js/features/tablesort.js Co-authored-by: silverwind <me@silverwind.io> * Update web_src/js/features/tablesort.js Co-authored-by: silverwind <me@silverwind.io> Co-authored-by: silverwind <me@silverwind.io>
This commit is contained in:
parent
ae20de7771
commit
c86478ec47
10 changed files with 106 additions and 18 deletions
|
@ -298,8 +298,30 @@ func NewFuncMap() []template.FuncMap {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
},
|
},
|
||||||
"svg": func(icon string, size int) template.HTML {
|
"svg": SVG,
|
||||||
return template.HTML(fmt.Sprintf(`<svg class="svg %s" width="%d" height="%d" aria-hidden="true"><use xlink:href="#%s" /></svg>`, icon, size, size, icon))
|
"SortArrow": func(normSort, revSort, urlSort string, isDefault bool) template.HTML {
|
||||||
|
// if needed
|
||||||
|
if len(normSort) == 0 || len(urlSort) == 0 {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(urlSort) == 0 && isDefault {
|
||||||
|
// if sort is sorted as default add arrow tho this table header
|
||||||
|
if isDefault {
|
||||||
|
return SVG("octicon-triangle-down", 16)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// if sort arg is in url test if it correlates with column header sort arguments
|
||||||
|
if urlSort == normSort {
|
||||||
|
// the table is sorted with this header normal
|
||||||
|
return SVG("octicon-triangle-down", 16)
|
||||||
|
} else if urlSort == revSort {
|
||||||
|
// the table is sorted with this header reverse
|
||||||
|
return SVG("octicon-triangle-up", 16)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// the table is NOT sorted with this header
|
||||||
|
return ""
|
||||||
},
|
},
|
||||||
}}
|
}}
|
||||||
}
|
}
|
||||||
|
@ -410,6 +432,11 @@ func NewTextFuncMap() []texttmpl.FuncMap {
|
||||||
}}
|
}}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SVG render icons
|
||||||
|
func SVG(icon string, size int) template.HTML {
|
||||||
|
return template.HTML(fmt.Sprintf(`<svg class="svg %s" width="%d" height="%d" aria-hidden="true"><use xlink:href="#%s" /></svg>`, icon, size, size, icon))
|
||||||
|
}
|
||||||
|
|
||||||
// Safe render raw as HTML
|
// Safe render raw as HTML
|
||||||
func Safe(raw string) template.HTML {
|
func Safe(raw string) template.HTML {
|
||||||
return template.HTML(raw)
|
return template.HTML(raw)
|
||||||
|
|
|
@ -33,9 +33,15 @@
|
||||||
<table class="ui very basic striped table">
|
<table class="ui very basic striped table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>{{.i18n.Tr "admin.users.name"}}</th>
|
<th data-sortt-asc="username" data-sortt-desc="reverseusername">
|
||||||
|
{{.i18n.Tr "admin.users.name"}}
|
||||||
|
{{SortArrow "username" "reverseusername" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.users.full_name"}}</th>
|
<th>{{.i18n.Tr "admin.users.full_name"}}</th>
|
||||||
<th>{{.i18n.Tr "email"}}</th>
|
<th data-sortt-asc="email" data-sortt-desc="reverseemail" data-sortt-default="true">
|
||||||
|
{{.i18n.Tr "email"}}
|
||||||
|
{{SortArrow "email" "reverseemail" $.SortType true}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.emails.primary"}}</th>
|
<th>{{.i18n.Tr "admin.emails.primary"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.emails.activated"}}</th>
|
<th>{{.i18n.Tr "admin.emails.activated"}}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
|
@ -16,12 +16,18 @@
|
||||||
<table class="ui very basic striped table">
|
<table class="ui very basic striped table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th data-sortt-asc="oldest" data-sortt-desc="newest">ID{{SortArrow "oldest" "newest" $.SortType false}}</th>
|
||||||
<th>{{.i18n.Tr "admin.orgs.name"}}</th>
|
<th data-sortt-asc="alphabetically" data-sortt-desc="reversealphabetically" data-sortt-default="true">
|
||||||
|
{{.i18n.Tr "admin.orgs.name"}}
|
||||||
|
{{SortArrow "alphabetically" "reversealphabetically" $.SortType true}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.orgs.teams"}}</th>
|
<th>{{.i18n.Tr "admin.orgs.teams"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.orgs.members"}}</th>
|
<th>{{.i18n.Tr "admin.orgs.members"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.repos"}}</th>
|
<th>{{.i18n.Tr "admin.users.repos"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.created"}}</th>
|
<th data-sortt-asc="recentupdate" data-sortt-desc="leastupdate">
|
||||||
|
{{.i18n.Tr "admin.users.created"}}
|
||||||
|
{{SortArrow "recentupdate" "leastupdate" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.users.edit"}}</th>
|
<th>{{.i18n.Tr "admin.users.edit"}}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
|
@ -13,15 +13,27 @@
|
||||||
<table class="ui very basic striped table">
|
<table class="ui very basic striped table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th data-sortt-asc="oldest" data-sortt-desc="newest">ID{{SortArrow "oldest" "newest" $.SortType false}}</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.owner"}}</th>
|
<th>{{.i18n.Tr "admin.repos.owner"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.name"}}</th>
|
<th data-sortt-asc="alphabetically" data-sortt-desc="reversealphabetically">
|
||||||
|
{{.i18n.Tr "admin.repos.name"}}
|
||||||
|
{{SortArrow "alphabetically" "reversealphabetically" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.private"}}</th>
|
<th>{{.i18n.Tr "admin.repos.private"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.watches"}}</th>
|
<th>{{.i18n.Tr "admin.repos.watches"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.stars"}}</th>
|
<th data-sortt-asc="moststars" data-sortt-desc="feweststars">
|
||||||
<th>{{.i18n.Tr "admin.repos.forks"}}</th>
|
{{.i18n.Tr "admin.repos.stars"}}
|
||||||
|
{{SortArrow "moststars" "feweststars" $.SortType false}}
|
||||||
|
</th>
|
||||||
|
<th data-sortt-asc="mostforks" data-sortt-desc="fewestforks">
|
||||||
|
{{.i18n.Tr "admin.repos.forks"}}
|
||||||
|
{{SortArrow "mostforks" "fewestforks" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.issues"}}</th>
|
<th>{{.i18n.Tr "admin.repos.issues"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.repos.size"}}</th>
|
<th data-sortt-asc="size" data-sortt-desc="reversesize">
|
||||||
|
{{.i18n.Tr "admin.repos.size"}}
|
||||||
|
{{SortArrow "size" "reversesize" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.users.created"}}</th>
|
<th>{{.i18n.Tr "admin.users.created"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.notices.op"}}</th>
|
<th>{{.i18n.Tr "admin.notices.op"}}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
|
@ -16,15 +16,21 @@
|
||||||
<table class="ui very basic striped table">
|
<table class="ui very basic striped table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th data-sortt-asc="oldest" data-sortt-desc="newest">ID{{SortArrow "oldest" "newest" .SortType false}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.name"}}</th>
|
<th data-sortt-asc="alphabetically" data-sortt-desc="reversealphabetically" data-sortt-default="true">
|
||||||
|
{{.i18n.Tr "admin.users.name"}}
|
||||||
|
{{SortArrow "alphabetically" "reversealphabetically" $.SortType true}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "email"}}</th>
|
<th>{{.i18n.Tr "email"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.activated"}}</th>
|
<th>{{.i18n.Tr "admin.users.activated"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.admin"}}</th>
|
<th>{{.i18n.Tr "admin.users.admin"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.restricted"}}</th>
|
<th>{{.i18n.Tr "admin.users.restricted"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.repos"}}</th>
|
<th>{{.i18n.Tr "admin.users.repos"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.created"}}</th>
|
<th>{{.i18n.Tr "admin.users.created"}}</th>
|
||||||
<th>{{.i18n.Tr "admin.users.last_login"}}</th>
|
<th data-sortt-asc="recentupdate" data-sortt-desc="leastupdate">
|
||||||
|
{{.i18n.Tr "admin.users.last_login"}}
|
||||||
|
{{SortArrow "recentupdate" "leastupdate" $.SortType false}}
|
||||||
|
</th>
|
||||||
<th>{{.i18n.Tr "admin.users.edit"}}</th>
|
<th>{{.i18n.Tr "admin.users.edit"}}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
|
|
20
web_src/js/features/tablesort.js
Normal file
20
web_src/js/features/tablesort.js
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
export default function initTableSort() {
|
||||||
|
for (const header of document.querySelectorAll('th[data-sortt-asc]') || []) {
|
||||||
|
const {sorttAsc, sorttDesc, sorttDefault} = header.dataset;
|
||||||
|
header.addEventListener('click', () => {
|
||||||
|
tableSort(sorttAsc, sorttDesc, sorttDefault);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function tableSort(normSort, revSort, isDefault) {
|
||||||
|
if (!normSort) return false;
|
||||||
|
if (!revSort) revSort = '';
|
||||||
|
|
||||||
|
const url = new URL(window.location);
|
||||||
|
let urlSort = url.searchParams.get('sort');
|
||||||
|
if (!urlSort && isDefault) urlSort = normSort;
|
||||||
|
|
||||||
|
url.searchParams.set('sort', urlSort !== normSort ? normSort : revSort);
|
||||||
|
window.location.replace(url.href);
|
||||||
|
}
|
|
@ -15,6 +15,7 @@ import initUserHeatmap from './features/userheatmap.js';
|
||||||
import initServiceWorker from './features/serviceworker.js';
|
import initServiceWorker from './features/serviceworker.js';
|
||||||
import attachTribute from './features/tribute.js';
|
import attachTribute from './features/tribute.js';
|
||||||
import createDropzone from './features/dropzone.js';
|
import createDropzone from './features/dropzone.js';
|
||||||
|
import initTableSort from './features/tablesort.js';
|
||||||
import highlight from './features/highlight.js';
|
import highlight from './features/highlight.js';
|
||||||
import ActivityTopAuthors from './components/ActivityTopAuthors.vue';
|
import ActivityTopAuthors from './components/ActivityTopAuthors.vue';
|
||||||
import {initNotificationsTable, initNotificationCount} from './features/notification.js';
|
import {initNotificationsTable, initNotificationCount} from './features/notification.js';
|
||||||
|
@ -2450,6 +2451,7 @@ $(document).ready(async () => {
|
||||||
initRepoStatusChecker();
|
initRepoStatusChecker();
|
||||||
initTemplateSearch();
|
initTemplateSearch();
|
||||||
initContextPopups();
|
initContextPopups();
|
||||||
|
initTableSort();
|
||||||
initNotificationsTable();
|
initNotificationsTable();
|
||||||
initNotificationCount();
|
initNotificationCount();
|
||||||
|
|
||||||
|
|
|
@ -6,8 +6,6 @@
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
|
|
||||||
&:not(.striped) {
|
&:not(.striped) {
|
||||||
padding-top: 5px;
|
|
||||||
|
|
||||||
thead {
|
thead {
|
||||||
th:last-child {
|
th:last-child {
|
||||||
padding-right: 5px !important;
|
padding-right: 5px !important;
|
||||||
|
|
|
@ -1223,6 +1223,17 @@ i.icon.centerlock {
|
||||||
margin-top: 1rem;
|
margin-top: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
table th[data-sortt-asc],
|
||||||
|
table th[data-sortt-desc] {
|
||||||
|
&:hover {
|
||||||
|
background: rgba(0, 0, 0, .1) !important;
|
||||||
|
cursor: pointer !important;
|
||||||
|
}
|
||||||
|
.svg {
|
||||||
|
margin-left: .25rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* limit width of all direct dropdown menu children */
|
/* limit width of all direct dropdown menu children */
|
||||||
/* https://github.com/go-gitea/gitea/pull/10835 */
|
/* https://github.com/go-gitea/gitea/pull/10835 */
|
||||||
.dropdown:not(.selection) > .menu:not(.review-box) > *:not(.header) {
|
.dropdown:not(.selection) > .menu:not(.review-box) > *:not(.header) {
|
||||||
|
|
|
@ -479,7 +479,7 @@ a.ui.basic.green.label:hover {
|
||||||
|
|
||||||
.ui.table thead th,
|
.ui.table thead th,
|
||||||
.ui.table > thead > tr > th {
|
.ui.table > thead > tr > th {
|
||||||
background: #404552 !important;
|
background: #404552;
|
||||||
color: #dbdbdb !important;
|
color: #dbdbdb !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue