summaryrefslogtreecommitdiffstats
path: root/web_src
diff options
context:
space:
mode:
authorsilverwind <me@silverwind.io>2021-11-22 09:19:01 +0100
committerGitHub <noreply@github.com>2021-11-22 16:19:01 +0800
commita159c3175f5f60a9de00f4d3c73787ffa6c63ddd (patch)
treedb212e3f9d20eb26bba12774ff3d8fa0bac71cbc /web_src
parent7743f13bed7dc5958c19603ccadd095db24c6b80 (diff)
downloadgitea-a159c3175f5f60a9de00f4d3c73787ffa6c63ddd.tar.gz
gitea-a159c3175f5f60a9de00f4d3c73787ffa6c63ddd.zip
Add new JS linter rules (#17699)
* Add new JS linter rules Adds a few useful rules from eslint-plugin-github. Notable changes: - Forbid dataset usage, its camel-casing behaviour makes it hard to grep for attributes. - Forbid .then() and .catch(), we should generally prefer await for new code. For rare cases where they are useful, a eslint-disable-line directive can be set. - Add docs js to linting * also enable github/array-foreach * small tweak Co-authored-by: Andrew Thornton <art27@cantab.net> Co-authored-by: techknowlogick <techknowlogick@gitea.io> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
Diffstat (limited to 'web_src')
-rw-r--r--web_src/js/features/clipboard.js8
-rw-r--r--web_src/js/features/codeeditor.js9
-rw-r--r--web_src/js/features/common-issue.js22
-rw-r--r--web_src/js/features/comp/ImagePaste.js4
-rw-r--r--web_src/js/features/heatmap.js4
-rw-r--r--web_src/js/features/repo-code.js8
-rw-r--r--web_src/js/features/repo-home.js8
-rw-r--r--web_src/js/features/repo-issue.js17
-rw-r--r--web_src/js/features/repo-legacy.js6
-rw-r--r--web_src/js/features/repo-projects.js83
-rw-r--r--web_src/js/features/tablesort.js4
-rw-r--r--web_src/js/features/user-auth-u2f.js5
-rw-r--r--web_src/js/markup/tasklist.js9
-rw-r--r--web_src/js/standalone/swagger.js2
14 files changed, 100 insertions, 89 deletions
diff --git a/web_src/js/features/clipboard.js b/web_src/js/features/clipboard.js
index b0c4134537..83626b614e 100644
--- a/web_src/js/features/clipboard.js
+++ b/web_src/js/features/clipboard.js
@@ -52,12 +52,8 @@ export default function initGlobalCopyToClipboardListener() {
// in case <button data-clipboard-text><svg></button>, so we just search
// up to 3 levels for performance
for (let i = 0; i < 3 && target; i++) {
- let text;
- if (target.dataset.clipboardText) {
- text = target.dataset.clipboardText;
- } else if (target.dataset.clipboardTarget) {
- text = document.querySelector(target.dataset.clipboardTarget)?.value;
- }
+ const text = target.getAttribute('data-clipboard-text') || document.querySelector(target.getAttribute('data-clipboard-target'))?.value;
+
if (text) {
e.preventDefault();
diff --git a/web_src/js/features/codeeditor.js b/web_src/js/features/codeeditor.js
index d4fd303fbf..a22043c9d4 100644
--- a/web_src/js/features/codeeditor.js
+++ b/web_src/js/features/codeeditor.js
@@ -21,7 +21,7 @@ const baseOptions = {
function getEditorconfig(input) {
try {
- return JSON.parse(input.dataset.editorconfig);
+ return JSON.parse(input.getAttribute('data-editorconfig'));
} catch {
return null;
}
@@ -132,14 +132,15 @@ function getFileBasedOptions(filename, lineWrapExts) {
export async function createCodeEditor(textarea, filenameInput, previewFileModes) {
const filename = basename(filenameInput.value);
const previewLink = document.querySelector('a[data-tab=preview]');
- const markdownExts = (textarea.dataset.markdownFileExts || '').split(',');
- const lineWrapExts = (textarea.dataset.lineWrapExtensions || '').split(',');
+ const markdownExts = (textarea.getAttribute('data-markdown-file-exts') || '').split(',');
+ const lineWrapExts = (textarea.getAttribute('data-line-wrap-extensions') || '').split(',');
const isMarkdown = markdownExts.includes(extname(filename));
const editorConfig = getEditorconfig(filenameInput);
if (previewLink) {
if (isMarkdown && (previewFileModes || []).includes('markdown')) {
- previewLink.dataset.url = previewLink.dataset.url.replace(/(.*)\/.*/i, `$1/markdown`);
+ const newUrl = (previewLink.getAttribute('data-url') || '').replace(/(.*)\/.*/i, `$1/markdown`);
+ previewLink.setAttribute('data-url', newUrl);
previewLink.style.display = '';
} else {
previewLink.style.display = 'none';
diff --git a/web_src/js/features/common-issue.js b/web_src/js/features/common-issue.js
index f89690abc0..07086d9e63 100644
--- a/web_src/js/features/common-issue.js
+++ b/web_src/js/features/common-issue.js
@@ -12,17 +12,25 @@ export function initCommonIssue() {
}
});
- $('.issue-action').on('click', function () {
- let {action, elementId, url} = this.dataset;
+ $('.issue-action').on('click', async function () {
+ let action = this.getAttribute('data-action');
+ let elementId = this.getAttribute('data-element-id');
+ const url = this.getAttribute('data-url');
const issueIDs = $('.issue-checkbox').children('input:checked').map((_, el) => {
- return el.dataset.issueId;
+ return el.getAttribute('data-issue-id');
}).get().join(',');
if (elementId === '0' && url.substr(-9) === '/assignee') {
elementId = '';
action = 'clear';
}
- updateIssuesMeta(url, action, issueIDs, elementId).then(() => {
- // NOTICE: This reset of checkbox state targets Firefox caching behaviour, as the checkboxes stay checked after reload
+ updateIssuesMeta(
+ url,
+ action,
+ issueIDs,
+ elementId
+ ).then(() => { // eslint-disable-line github/no-then
+ // NOTICE: This reset of checkbox state targets Firefox caching behaviour, as the
+ // checkboxes stay checked after reload
if (action === 'close' || action === 'open') {
// uncheck all checkboxes
$('.issue-checkbox input[type="checkbox"]').each((_, e) => { e.checked = false });
@@ -31,8 +39,8 @@ export function initCommonIssue() {
});
});
- // NOTICE: This event trigger targets Firefox caching behaviour, as the checkboxes stay checked after reload
- // trigger ckecked event, if checkboxes are checked on load
+ // NOTICE: This event trigger targets Firefox caching behaviour, as the checkboxes stay
+ // checked after reload trigger ckecked event, if checkboxes are checked on load
$('.issue-checkbox input[type="checkbox"]:checked').first().each((_, e) => {
e.checked = false;
$(e).trigger('click');
diff --git a/web_src/js/features/comp/ImagePaste.js b/web_src/js/features/comp/ImagePaste.js
index f7f076bf80..99f4e069cc 100644
--- a/web_src/js/features/comp/ImagePaste.js
+++ b/web_src/js/features/comp/ImagePaste.js
@@ -59,7 +59,7 @@ export function initCompImagePaste($target) {
if (!dropzone) {
return;
}
- const uploadUrl = dropzone.dataset.uploadUrl;
+ const uploadUrl = dropzone.getAttribute('data-upload-url');
const dropzoneFiles = dropzone.querySelector('.files');
for (const textarea of this.querySelectorAll('textarea')) {
textarea.addEventListener('paste', async (e) => {
@@ -77,7 +77,7 @@ export function initCompImagePaste($target) {
}
export function initSimpleMDEImagePaste(simplemde, dropzone, files) {
- const uploadUrl = dropzone.dataset.uploadUrl;
+ const uploadUrl = dropzone.getAttribute('data-upload-url');
simplemde.codemirror.on('paste', async (_, e) => {
for (const img of clipboardPastedImages(e)) {
const name = img.name.substr(0, img.name.lastIndexOf('.'));
diff --git a/web_src/js/features/heatmap.js b/web_src/js/features/heatmap.js
index 52b7517c1a..e652f4ed98 100644
--- a/web_src/js/features/heatmap.js
+++ b/web_src/js/features/heatmap.js
@@ -8,11 +8,11 @@ export default function initHeatmap() {
try {
const heatmap = {};
- JSON.parse(el.dataset.heatmapData).forEach(({contributions, timestamp}) => {
+ for (const {contributions, timestamp} of JSON.parse(el.getAttribute('data-heatmap-data'))) {
// Convert to user timezone and sum contributions by date
const dateStr = new Date(timestamp * 1000).toDateString();
heatmap[dateStr] = (heatmap[dateStr] || 0) + contributions;
- });
+ }
const values = Object.keys(heatmap).map((v) => {
return {date: new Date(v), count: heatmap[v]};
diff --git a/web_src/js/features/repo-code.js b/web_src/js/features/repo-code.js
index 8a361a750d..1855f86b41 100644
--- a/web_src/js/features/repo-code.js
+++ b/web_src/js/features/repo-code.js
@@ -131,12 +131,14 @@ export function initRepoCodeView() {
}
$(document).on('click', '.fold-file', ({currentTarget}) => {
const box = currentTarget.closest('.file-content');
- const folded = box.dataset.folded !== 'true';
+ const folded = box.getAttribute('data-folded') !== 'true';
currentTarget.innerHTML = svg(`octicon-chevron-${folded ? 'right' : 'down'}`, 18);
- box.dataset.folded = String(folded);
+ box.setAttribute('data-folded', String(folded));
});
$(document).on('click', '.blob-excerpt', async ({currentTarget}) => {
- const {url, query, anchor} = currentTarget.dataset;
+ const url = currentTarget.getAttribute('data-url');
+ const query = currentTarget.getAttribute('data-query');
+ const anchor = currentTarget.getAttribute('data-anchor');
if (!url) return;
const blob = await $.get(`${url}?${query}&anchor=${anchor}`);
currentTarget.closest('tr').outerHTML = blob;
diff --git a/web_src/js/features/repo-home.js b/web_src/js/features/repo-home.js
index f8bf9318ce..29b4470619 100644
--- a/web_src/js/features/repo-home.js
+++ b/web_src/js/features/repo-home.js
@@ -57,13 +57,13 @@ export function initRepoTopicBar() {
const {invalidTopics} = xhr.responseJSON;
const topicLables = topicDropdown.children('a.ui.label');
- topics.split(',').forEach((value, index) => {
+ for (const [index, value] of topics.split(',').entries()) {
for (let i = 0; i < invalidTopics.length; i++) {
if (invalidTopics[i] === value) {
topicLables.eq(index).removeClass('green').addClass('red');
}
}
- });
+ }
} else {
topicPrompts.countPrompt = xhr.responseJSON.message;
}
@@ -101,7 +101,9 @@ export function initRepoTopicBar() {
const query = stripTags(this.urlData.query.trim());
let found_query = false;
const current_topics = [];
- topicDropdown.find('div.label.visible.topic,a.label.visible').each((_, e) => { current_topics.push(e.dataset.value) });
+ topicDropdown.find('div.label.visible.topic,a.label.visible').each((_, el) => {
+ current_topics.push(el.getAttribute('data-value'));
+ });
if (res.topics) {
let found = false;
diff --git a/web_src/js/features/repo-issue.js b/web_src/js/features/repo-issue.js
index 1476571cc7..b224245fdf 100644
--- a/web_src/js/features/repo-issue.js
+++ b/web_src/js/features/repo-issue.js
@@ -182,7 +182,8 @@ export function initRepoIssueCommentDelete() {
export function initRepoIssueDependencyDelete() {
// Delete Issue dependency
$(document).on('click', '.delete-dependency-button', (e) => {
- const {id, type} = e.currentTarget.dataset;
+ const id = e.currentTarget.getAttribute('data-id');
+ const type = e.currentTarget.getAttribute('data-type');
$('.remove-dependency').modal({
closable: false,
@@ -348,22 +349,19 @@ export async function updateIssuesMeta(url, action, issueIds, elementId) {
export function initRepoIssueComments() {
if ($('.repository.view.issue .timeline').length === 0) return;
- $('.re-request-review').on('click', function (event) {
+ $('.re-request-review').on('click', function (e) {
+ e.preventDefault();
const url = $(this).data('update-url');
const issueId = $(this).data('issue-id');
const id = $(this).data('id');
const isChecked = $(this).hasClass('checked');
- event.preventDefault();
updateIssuesMeta(
url,
isChecked ? 'detach' : 'attach',
issueId,
id,
- ).then(() => {
- window.location.reload();
- });
- return false;
+ ).then(() => window.location.reload()); // eslint-disable-line github/no-then
});
$('.dismiss-review-btn').on('click', function (e) {
@@ -550,7 +548,10 @@ export function initRepoIssueWipToggle() {
// Toggle WIP
$('.toggle-wip a, .toggle-wip button').on('click', async (e) => {
e.preventDefault();
- const {title, wipPrefix, updateUrl} = e.currentTarget.closest('.toggle-wip').dataset;
+ const toggleWip = e.currentTarget.closest('.toggle-wip');
+ const title = toggleWip.getAttribute('data-title');
+ const wipPrefix = toggleWip.getAttribute('data-wip-prefix');
+ const updateUrl = toggleWip.getAttribute('data-update-url');
await $.post(updateUrl, {
_csrf: csrfToken,
title: title?.startsWith(wipPrefix) ? title.substr(wipPrefix.length).trim() : `${wipPrefix.trim()} ${title}`,
diff --git a/web_src/js/features/repo-legacy.js b/web_src/js/features/repo-legacy.js
index d8530fe24f..5904b3ce22 100644
--- a/web_src/js/features/repo-legacy.js
+++ b/web_src/js/features/repo-legacy.js
@@ -162,7 +162,7 @@ export function initRepoCommentForm() {
'clear',
$listMenu.data('issue-id'),
'',
- ).then(() => window.location.reload());
+ ).then(() => window.location.reload()); // eslint-disable-line github/no-then
}
$(this).parent().find('.item').each(function () {
@@ -205,7 +205,7 @@ export function initRepoCommentForm() {
'',
$menu.data('issue-id'),
$(this).data('id'),
- ).then(() => window.location.reload());
+ ).then(() => window.location.reload()); // eslint-disable-line github/no-then
}
let icon = '';
@@ -238,7 +238,7 @@ export function initRepoCommentForm() {
'',
$menu.data('issue-id'),
$(this).data('id'),
- ).then(() => window.location.reload());
+ ).then(() => window.location.reload()); // eslint-disable-line github/no-then
}
$list.find('.selected').html('');
diff --git a/web_src/js/features/repo-projects.js b/web_src/js/features/repo-projects.js
index ef664b4808..d9baa58916 100644
--- a/web_src/js/features/repo-projects.js
+++ b/web_src/js/features/repo-projects.js
@@ -6,57 +6,54 @@ async function initRepoProjectSortable() {
const {Sortable} = await import(/* webpackChunkName: "sortable" */'sortablejs');
const boardColumns = document.getElementsByClassName('board-column');
- new Sortable(
- els[0],
- {
- group: 'board-column',
- draggable: '.board-column',
- animation: 150,
- ghostClass: 'card-ghost',
- onSort: () => {
- const board = document.getElementsByClassName('board')[0];
- const boardColumns = board.getElementsByClassName('board-column');
-
- boardColumns.forEach((column, i) => {
- if (parseInt($(column).data('sorting')) !== i) {
- $.ajax({
- url: $(column).data('url'),
- data: JSON.stringify({sorting: i, color: rgbToHex($(column).css('backgroundColor'))}),
- headers: {
- 'X-Csrf-Token': csrfToken,
- 'X-Remote': true,
- },
- contentType: 'application/json',
- method: 'PUT',
- });
- }
- });
- },
- },
- );
- for (const column of boardColumns) {
- new Sortable(
- column.getElementsByClassName('board')[0],
- {
- group: 'shared',
- animation: 150,
- ghostClass: 'card-ghost',
- onAdd: (e) => {
- $.ajax(`${e.to.dataset.url}/${e.item.dataset.issue}`, {
+ new Sortable(els[0], {
+ group: 'board-column',
+ draggable: '.board-column',
+ animation: 150,
+ ghostClass: 'card-ghost',
+ onSort: () => {
+ const board = document.getElementsByClassName('board')[0];
+ const boardColumns = board.getElementsByClassName('board-column');
+
+ for (const [i, column] of boardColumns.entries()) {
+ if (parseInt($(column).data('sorting')) !== i) {
+ $.ajax({
+ url: $(column).data('url'),
+ data: JSON.stringify({sorting: i, color: rgbToHex($(column).css('backgroundColor'))}),
headers: {
'X-Csrf-Token': csrfToken,
'X-Remote': true,
},
contentType: 'application/json',
- type: 'POST',
- error: () => {
- e.from.insertBefore(e.item, e.from.children[e.oldIndex]);
- },
+ method: 'PUT',
});
- },
+ }
+ }
+ },
+ });
+
+ for (const column of boardColumns) {
+ new Sortable(column.getElementsByClassName('board')[0], {
+ group: 'shared',
+ animation: 150,
+ ghostClass: 'card-ghost',
+ onAdd: ({item, from, to, oldIndex}) => {
+ const url = to.getAttribute('data-url');
+ const issue = item.getAttribute('data-issue');
+ $.ajax(`${url}/${issue}`, {
+ headers: {
+ 'X-Csrf-Token': csrfToken,
+ 'X-Remote': true,
+ },
+ contentType: 'application/json',
+ type: 'POST',
+ error: () => {
+ from.insertBefore(item, from.children[oldIndex]);
+ },
+ });
},
- );
+ });
}
}
diff --git a/web_src/js/features/tablesort.js b/web_src/js/features/tablesort.js
index 17da2985a0..55238ac0ac 100644
--- a/web_src/js/features/tablesort.js
+++ b/web_src/js/features/tablesort.js
@@ -1,6 +1,8 @@
export default function initTableSort() {
for (const header of document.querySelectorAll('th[data-sortt-asc]') || []) {
- const {sorttAsc, sorttDesc, sorttDefault} = header.dataset;
+ const sorttAsc = header.getAttribute('sortt-asc');
+ const sorttDesc = header.getAttribute('sortt-desc');
+ const sorttDefault = header.getAttribute('sortt-default');
header.addEventListener('click', () => {
tableSort(sorttAsc, sorttDesc, sorttDefault);
});
diff --git a/web_src/js/features/user-auth-u2f.js b/web_src/js/features/user-auth-u2f.js
index 15beaaa207..8951c8a4ab 100644
--- a/web_src/js/features/user-auth-u2f.js
+++ b/web_src/js/features/user-auth-u2f.js
@@ -1,3 +1,4 @@
+/* eslint-disable github/no-then */
const {appSubUrl, csrfToken} = window.config;
export function initUserAuthU2fAuth() {
@@ -77,11 +78,11 @@ function u2fError(errorType) {
};
u2fErrors[errorType].removeClass('hide');
- Object.keys(u2fErrors).forEach((type) => {
+ for (const type of Object.keys(u2fErrors)) {
if (type !== `${errorType}`) {
u2fErrors[type].addClass('hide');
}
- });
+ }
$('#u2f-error').modal('show');
}
diff --git a/web_src/js/markup/tasklist.js b/web_src/js/markup/tasklist.js
index f8ca5b0406..149027d428 100644
--- a/web_src/js/markup/tasklist.js
+++ b/web_src/js/markup/tasklist.js
@@ -14,14 +14,14 @@ export function initMarkupTasklist() {
const checkboxes = el.querySelectorAll(`.task-list-item input[type=checkbox]`);
for (const checkbox of checkboxes) {
- if (checkbox.dataset.editable) {
+ if (checkbox.hasAttribute('data-editable')) {
return;
}
- checkbox.dataset.editable = 'true';
+ checkbox.setAttribute('data-editable', 'true');
checkbox.addEventListener('input', async () => {
const checkboxCharacter = checkbox.checked ? 'x' : ' ';
- const position = parseInt(checkbox.dataset.sourcePosition) + 1;
+ const position = parseInt(checkbox.getAttribute('data-source-position')) + 1;
const rawContent = container.querySelector('.raw-content');
const oldContent = rawContent.textContent;
@@ -43,7 +43,8 @@ export function initMarkupTasklist() {
try {
const editContentZone = container.querySelector('.edit-content-zone');
- const {updateUrl, context} = editContentZone.dataset;
+ const updateUrl = editContentZone.getAttribute('data-update-url');
+ const context = editContentZone.getAttribute('data-context');
await $.post(updateUrl, {
ignore_attachments: true,
diff --git a/web_src/js/standalone/swagger.js b/web_src/js/standalone/swagger.js
index 5d8b7192b2..cb91089daf 100644
--- a/web_src/js/standalone/swagger.js
+++ b/web_src/js/standalone/swagger.js
@@ -2,7 +2,7 @@ import SwaggerUI from 'swagger-ui-dist/swagger-ui-es-bundle.js';
import 'swagger-ui-dist/swagger-ui.css';
window.addEventListener('load', async () => {
- const url = document.getElementById('swagger-ui').dataset.source;
+ const url = document.getElementById('swagger-ui').getAttribute('data-source');
const res = await fetch(url);
const spec = await res.json();