diff options
author | silverwind <me@silverwind.io> | 2021-11-22 09:19:01 +0100 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-11-22 16:19:01 +0800 |
commit | a159c3175f5f60a9de00f4d3c73787ffa6c63ddd (patch) | |
tree | db212e3f9d20eb26bba12774ff3d8fa0bac71cbc /web_src/js | |
parent | 7743f13bed7dc5958c19603ccadd095db24c6b80 (diff) | |
download | gitea-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/js')
-rw-r--r-- | web_src/js/features/clipboard.js | 8 | ||||
-rw-r--r-- | web_src/js/features/codeeditor.js | 9 | ||||
-rw-r--r-- | web_src/js/features/common-issue.js | 22 | ||||
-rw-r--r-- | web_src/js/features/comp/ImagePaste.js | 4 | ||||
-rw-r--r-- | web_src/js/features/heatmap.js | 4 | ||||
-rw-r--r-- | web_src/js/features/repo-code.js | 8 | ||||
-rw-r--r-- | web_src/js/features/repo-home.js | 8 | ||||
-rw-r--r-- | web_src/js/features/repo-issue.js | 17 | ||||
-rw-r--r-- | web_src/js/features/repo-legacy.js | 6 | ||||
-rw-r--r-- | web_src/js/features/repo-projects.js | 83 | ||||
-rw-r--r-- | web_src/js/features/tablesort.js | 4 | ||||
-rw-r--r-- | web_src/js/features/user-auth-u2f.js | 5 | ||||
-rw-r--r-- | web_src/js/markup/tasklist.js | 9 | ||||
-rw-r--r-- | web_src/js/standalone/swagger.js | 2 |
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(); |