mirror of https://github.com/go-gitea/gitea
Refactor branch/tag selector to Vue SFC (#23421)
Follow #23394
There were many bad smells in old code. This PR only moves the code into
Vue SFC, doesn't touch the unrelated logic.
update: after
5f23218c85
, there should be no usage of the vue-rumtime-compiler anymore
(hopefully), so I think this PR could close #19851
---------
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
pull/23456/head^2
parent
d56bb74201
commit
ac8d71ff07
@ -1,208 +0,0 @@ |
||||
import {createApp, nextTick} from 'vue'; |
||||
import $ from 'jquery'; |
||||
|
||||
export function initRepoBranchTagDropdown(selector) { |
||||
$(selector).each(function (dropdownIndex, elRoot) { |
||||
const data = { |
||||
csrfToken: window.config.csrfToken, |
||||
items: [], |
||||
searchTerm: '', |
||||
menuVisible: false, |
||||
createTag: false, |
||||
release: null, |
||||
|
||||
isViewTag: false, |
||||
isViewBranch: false, |
||||
isViewTree: false, |
||||
|
||||
active: 0, |
||||
|
||||
...window.config.pageData.branchDropdownDataList[dropdownIndex], |
||||
}; |
||||
|
||||
// the "data.defaultBranch" is ambiguous, it could be "branch name" or "tag name"
|
||||
|
||||
if (data.showBranchesInDropdown && data.branches) { |
||||
for (const branch of data.branches) { |
||||
data.items.push({name: branch, url: branch, branch: true, tag: false, selected: branch === data.defaultBranch}); |
||||
} |
||||
} |
||||
if (!data.noTag && data.tags) { |
||||
for (const tag of data.tags) { |
||||
if (data.release) { |
||||
data.items.push({name: tag, url: tag, branch: false, tag: true, selected: tag === data.release.tagName}); |
||||
} else { |
||||
data.items.push({name: tag, url: tag, branch: false, tag: true, selected: tag === data.defaultBranch}); |
||||
} |
||||
} |
||||
} |
||||
|
||||
const view = createApp({ |
||||
delimiters: ['${', '}'], |
||||
data() { |
||||
return data; |
||||
}, |
||||
computed: { |
||||
filteredItems() { |
||||
const items = this.items.filter((item) => { |
||||
return ((this.mode === 'branches' && item.branch) || (this.mode === 'tags' && item.tag)) && |
||||
(!this.searchTerm || item.name.toLowerCase().includes(this.searchTerm.toLowerCase())); |
||||
}); |
||||
|
||||
// no idea how to fix this so linting rule is disabled instead
|
||||
this.active = (items.length === 0 && this.showCreateNewBranch ? 0 : -1); // eslint-disable-line vue/no-side-effects-in-computed-properties
|
||||
return items; |
||||
}, |
||||
showNoResults() { |
||||
return this.filteredItems.length === 0 && !this.showCreateNewBranch; |
||||
}, |
||||
showCreateNewBranch() { |
||||
if (this.disableCreateBranch || !this.searchTerm) { |
||||
return false; |
||||
} |
||||
|
||||
return this.items.filter((item) => item.name.toLowerCase() === this.searchTerm.toLowerCase()).length === 0; |
||||
} |
||||
}, |
||||
|
||||
watch: { |
||||
menuVisible(visible) { |
||||
if (visible) { |
||||
this.focusSearchField(); |
||||
} |
||||
} |
||||
}, |
||||
|
||||
beforeMount() { |
||||
switch (data.viewType) { |
||||
case 'tree': |
||||
this.isViewTree = true; |
||||
break; |
||||
case 'tag': |
||||
this.isViewTag = true; |
||||
break; |
||||
default: |
||||
this.isViewBranch = true; |
||||
break; |
||||
} |
||||
|
||||
document.body.addEventListener('click', (event) => { |
||||
if (elRoot.contains(event.target)) return; |
||||
if (this.menuVisible) { |
||||
this.menuVisible = false; |
||||
} |
||||
}); |
||||
}, |
||||
|
||||
methods: { |
||||
selectItem(item) { |
||||
const prev = this.getSelected(); |
||||
if (prev !== null) { |
||||
prev.selected = false; |
||||
} |
||||
item.selected = true; |
||||
const url = (item.tag) ? this.tagURLPrefix + item.url + this.tagURLSuffix : this.branchURLPrefix + item.url + this.branchURLSuffix; |
||||
if (!this.branchForm) { |
||||
window.location.href = url; |
||||
} else { |
||||
this.isViewTree = false; |
||||
this.isViewTag = false; |
||||
this.isViewBranch = false; |
||||
this.$refs.dropdownRefName.textContent = item.name; |
||||
if (this.setAction) { |
||||
$(`#${this.branchForm}`).attr('action', url); |
||||
} else { |
||||
$(`#${this.branchForm} input[name="refURL"]`).val(url); |
||||
} |
||||
$(`#${this.branchForm} input[name="ref"]`).val(item.name); |
||||
if (item.tag) { |
||||
this.isViewTag = true; |
||||
$(`#${this.branchForm} input[name="refType"]`).val('tag'); |
||||
} else { |
||||
this.isViewBranch = true; |
||||
$(`#${this.branchForm} input[name="refType"]`).val('branch'); |
||||
} |
||||
if (this.submitForm) { |
||||
$(`#${this.branchForm}`).trigger('submit'); |
||||
} |
||||
this.menuVisible = false; |
||||
} |
||||
}, |
||||
createNewBranch() { |
||||
if (!this.showCreateNewBranch) return; |
||||
$(this.$refs.newBranchForm).trigger('submit'); |
||||
}, |
||||
focusSearchField() { |
||||
nextTick(() => { |
||||
this.$refs.searchField.focus(); |
||||
}); |
||||
}, |
||||
getSelected() { |
||||
for (let i = 0, j = this.items.length; i < j; ++i) { |
||||
if (this.items[i].selected) return this.items[i]; |
||||
} |
||||
return null; |
||||
}, |
||||
getSelectedIndexInFiltered() { |
||||
for (let i = 0, j = this.filteredItems.length; i < j; ++i) { |
||||
if (this.filteredItems[i].selected) return i; |
||||
} |
||||
return -1; |
||||
}, |
||||
scrollToActive() { |
||||
let el = this.$refs[`listItem${this.active}`]; |
||||
if (!el || !el.length) return; |
||||
if (Array.isArray(el)) { |
||||
el = el[0]; |
||||
} |
||||
|
||||
const cont = this.$refs.scrollContainer; |
||||
if (el.offsetTop < cont.scrollTop) { |
||||
cont.scrollTop = el.offsetTop; |
||||
} else if (el.offsetTop + el.clientHeight > cont.scrollTop + cont.clientHeight) { |
||||
cont.scrollTop = el.offsetTop + el.clientHeight - cont.clientHeight; |
||||
} |
||||
}, |
||||
keydown(event) { |
||||
if (event.keyCode === 40) { // arrow down
|
||||
event.preventDefault(); |
||||
|
||||
if (this.active === -1) { |
||||
this.active = this.getSelectedIndexInFiltered(); |
||||
} |
||||
|
||||
if (this.active + (this.showCreateNewBranch ? 0 : 1) >= this.filteredItems.length) { |
||||
return; |
||||
} |
||||
this.active++; |
||||
this.scrollToActive(); |
||||
} else if (event.keyCode === 38) { // arrow up
|
||||
event.preventDefault(); |
||||
|
||||
if (this.active === -1) { |
||||
this.active = this.getSelectedIndexInFiltered(); |
||||
} |
||||
|
||||
if (this.active <= 0) { |
||||
return; |
||||
} |
||||
this.active--; |
||||
this.scrollToActive(); |
||||
} else if (event.keyCode === 13) { // enter
|
||||
event.preventDefault(); |
||||
|
||||
if (this.active >= this.filteredItems.length) { |
||||
this.createNewBranch(); |
||||
} else if (this.active >= 0) { |
||||
this.selectItem(this.filteredItems[this.active]); |
||||
} |
||||
} else if (event.keyCode === 27) { // escape
|
||||
event.preventDefault(); |
||||
this.menuVisible = false; |
||||
} |
||||
} |
||||
} |
||||
}); |
||||
view.mount(this); |
||||
}); |
||||
} |
@ -0,0 +1,293 @@ |
||||
<template> |
||||
<div class="ui floating filter dropdown custom"> |
||||
<button class="branch-dropdown-button gt-ellipsis ui basic small compact button gt-df" @click="menuVisible = !menuVisible" @keyup.enter="menuVisible = !menuVisible"> |
||||
<span class="text gt-df gt-ac gt-mr-2"> |
||||
<template v-if="release">{{ textReleaseCompare }}</template> |
||||
<template v-else> |
||||
<svg-icon v-if="isViewTag" name="octicon-tag" /> |
||||
<svg-icon v-else name="octicon-git-branch"/> |
||||
<strong ref="dropdownRefName" class="gt-ml-3">{{ refNameText }}</strong> |
||||
</template> |
||||
</span> |
||||
<svg-icon name="octicon-triangle-down" :size="14" class-name="dropdown icon"/> |
||||
</button> |
||||
<div class="menu transition" :class="{visible: menuVisible}" v-if="menuVisible" v-cloak> |
||||
<div class="ui icon search input"> |
||||
<i class="icon gt-df gt-ac gt-jc gt-m-0"><svg-icon name="octicon-filter" :size="16"/></i> |
||||
<input name="search" ref="searchField" autocomplete="off" v-model="searchTerm" @keydown="keydown($event)" :placeholder="searchFieldPlaceholder"> |
||||
</div> |
||||
<template v-if="showBranchesInDropdown"> |
||||
<div class="header branch-tag-choice"> |
||||
<div class="ui grid"> |
||||
<div class="two column row"> |
||||
<a class="reference column" href="#" @click="createTag = false; mode = 'branches'; focusSearchField()"> |
||||
<span class="text" :class="{black: mode === 'branches'}"> |
||||
<svg-icon name="octicon-git-branch" :size="16" class-name="gt-mr-2"/>{{ textBranches }} |
||||
</span> |
||||
</a> |
||||
<template v-if="!noTag"> |
||||
<a class="reference column" href="#" @click="createTag = true; mode = 'tags'; focusSearchField()"> |
||||
<span class="text" :class="{black: mode === 'tags'}"> |
||||
<svg-icon name="octicon-tag" :size="16" class-name="gt-mr-2"/>{{ textTags }} |
||||
</span> |
||||
</a> |
||||
</template> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</template> |
||||
<div class="scrolling menu" ref="scrollContainer"> |
||||
<div v-for="(item, index) in filteredItems" :key="item.name" class="item" :class="{selected: item.selected, active: active === index}" @click="selectItem(item)" :ref="'listItem' + index"> |
||||
{{ item.name }} |
||||
</div> |
||||
<div class="item" v-if="showCreateNewBranch" :class="{active: active === filteredItems.length}" :ref="'listItem' + filteredItems.length"> |
||||
<a href="#" @click="createNewBranch()"> |
||||
<div v-show="createTag"> |
||||
<i class="reference tags icon"/> |
||||
<!-- eslint-disable-next-line vue/no-v-html --> |
||||
<span v-html="textCreateTag.replace('%s', searchTerm)"/> |
||||
</div> |
||||
<div v-show="!createTag"> |
||||
<svg-icon name="octicon-git-branch"/> |
||||
<!-- eslint-disable-next-line vue/no-v-html --> |
||||
<span v-html="textCreateBranch.replace('%s', searchTerm)"/> |
||||
</div> |
||||
<div class="text small"> |
||||
<span v-if="isViewBranch || release">{{ textCreateBranchFrom.replace('%s', branchName) }}</span> |
||||
<span v-else-if="isViewTag">{{ textCreateBranchFrom.replace('%s', tagName) }}</span> |
||||
<span v-else>{{ textCreateBranchFrom.replace('%s', commitIdShort) }}</span> |
||||
</div> |
||||
</a> |
||||
<form ref="newBranchForm" :action="formActionUrl" method="post"> |
||||
<input type="hidden" name="_csrf" :value="csrfToken"> |
||||
<input type="hidden" name="new_branch_name" v-model="searchTerm"> |
||||
<input type="hidden" name="create_tag" v-model="createTag"> |
||||
<input type="hidden" name="current_path" v-model="treePath" v-if="treePath"> |
||||
</form> |
||||
</div> |
||||
</div> |
||||
<div class="message" v-if="showNoResults"> |
||||
{{ noResults }} |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import {createApp, nextTick} from 'vue'; |
||||
import $ from 'jquery'; |
||||
import {SvgIcon} from '../svg.js'; |
||||
import {pathEscapeSegments} from '../utils/url.js'; |
||||
|
||||
const sfc = { |
||||
components: {SvgIcon}, |
||||
|
||||
// no `data()`, at the moment, the `data()` is provided by the init code, which is not ideal and should be fixed in the future |
||||
|
||||
computed: { |
||||
filteredItems() { |
||||
const items = this.items.filter((item) => { |
||||
return ((this.mode === 'branches' && item.branch) || (this.mode === 'tags' && item.tag)) && |
||||
(!this.searchTerm || item.name.toLowerCase().includes(this.searchTerm.toLowerCase())); |
||||
}); |
||||
|
||||
// TODO: fix this anti-pattern: side-effects-in-computed-properties |
||||
this.active = (items.length === 0 && this.showCreateNewBranch ? 0 : -1); |
||||
return items; |
||||
}, |
||||
showNoResults() { |
||||
return this.filteredItems.length === 0 && !this.showCreateNewBranch; |
||||
}, |
||||
showCreateNewBranch() { |
||||
if (this.disableCreateBranch || !this.searchTerm) { |
||||
return false; |
||||
} |
||||
return this.items.filter((item) => item.name.toLowerCase() === this.searchTerm.toLowerCase()).length === 0; |
||||
}, |
||||
formActionUrl() { |
||||
return `${this.repoLink}/branches/_new/${pathEscapeSegments(this.branchNameSubURL)}`; |
||||
}, |
||||
}, |
||||
|
||||
watch: { |
||||
menuVisible(visible) { |
||||
if (visible) { |
||||
this.focusSearchField(); |
||||
} |
||||
} |
||||
}, |
||||
|
||||
beforeMount() { |
||||
if (this.viewType === 'tree') { |
||||
this.isViewTree = true; |
||||
this.refNameText = this.commitIdShort; |
||||
} else if (this.viewType === 'tag') { |
||||
this.isViewTag = true; |
||||
this.refNameText = this.tagName; |
||||
} else { |
||||
this.isViewBranch = true; |
||||
this.refNameText = this.branchName; |
||||
} |
||||
|
||||
document.body.addEventListener('click', (event) => { |
||||
if (this.$el.contains(event.target)) return; |
||||
if (this.menuVisible) { |
||||
this.menuVisible = false; |
||||
} |
||||
}); |
||||
}, |
||||
|
||||
methods: { |
||||
selectItem(item) { |
||||
const prev = this.getSelected(); |
||||
if (prev !== null) { |
||||
prev.selected = false; |
||||
} |
||||
item.selected = true; |
||||
const url = (item.tag) ? this.tagURLPrefix + item.url + this.tagURLSuffix : this.branchURLPrefix + item.url + this.branchURLSuffix; |
||||
if (!this.branchForm) { |
||||
window.location.href = url; |
||||
} else { |
||||
this.isViewTree = false; |
||||
this.isViewTag = false; |
||||
this.isViewBranch = false; |
||||
this.$refs.dropdownRefName.textContent = item.name; |
||||
if (this.setAction) { |
||||
$(`#${this.branchForm}`).attr('action', url); |
||||
} else { |
||||
$(`#${this.branchForm} input[name="refURL"]`).val(url); |
||||
} |
||||
$(`#${this.branchForm} input[name="ref"]`).val(item.name); |
||||
if (item.tag) { |
||||
this.isViewTag = true; |
||||
$(`#${this.branchForm} input[name="refType"]`).val('tag'); |
||||
} else { |
||||
this.isViewBranch = true; |
||||
$(`#${this.branchForm} input[name="refType"]`).val('branch'); |
||||
} |
||||
if (this.submitForm) { |
||||
$(`#${this.branchForm}`).trigger('submit'); |
||||
} |
||||
this.menuVisible = false; |
||||
} |
||||
}, |
||||
createNewBranch() { |
||||
if (!this.showCreateNewBranch) return; |
||||
$(this.$refs.newBranchForm).trigger('submit'); |
||||
}, |
||||
focusSearchField() { |
||||
nextTick(() => { |
||||
this.$refs.searchField.focus(); |
||||
}); |
||||
}, |
||||
getSelected() { |
||||
for (let i = 0, j = this.items.length; i < j; ++i) { |
||||
if (this.items[i].selected) return this.items[i]; |
||||
} |
||||
return null; |
||||
}, |
||||
getSelectedIndexInFiltered() { |
||||
for (let i = 0, j = this.filteredItems.length; i < j; ++i) { |
||||
if (this.filteredItems[i].selected) return i; |
||||
} |
||||
return -1; |
||||
}, |
||||
scrollToActive() { |
||||
let el = this.$refs[`listItem${this.active}`]; |
||||
if (!el || !el.length) return; |
||||
if (Array.isArray(el)) { |
||||
el = el[0]; |
||||
} |
||||
|
||||
const cont = this.$refs.scrollContainer; |
||||
if (el.offsetTop < cont.scrollTop) { |
||||
cont.scrollTop = el.offsetTop; |
||||
} else if (el.offsetTop + el.clientHeight > cont.scrollTop + cont.clientHeight) { |
||||
cont.scrollTop = el.offsetTop + el.clientHeight - cont.clientHeight; |
||||
} |
||||
}, |
||||
keydown(event) { |
||||
if (event.keyCode === 40) { // arrow down |
||||
event.preventDefault(); |
||||
|
||||
if (this.active === -1) { |
||||
this.active = this.getSelectedIndexInFiltered(); |
||||
} |
||||
|
||||
if (this.active + (this.showCreateNewBranch ? 0 : 1) >= this.filteredItems.length) { |
||||
return; |
||||
} |
||||
this.active++; |
||||
this.scrollToActive(); |
||||
} else if (event.keyCode === 38) { // arrow up |
||||
event.preventDefault(); |
||||
|
||||
if (this.active === -1) { |
||||
this.active = this.getSelectedIndexInFiltered(); |
||||
} |
||||
|
||||
if (this.active <= 0) { |
||||
return; |
||||
} |
||||
this.active--; |
||||
this.scrollToActive(); |
||||
} else if (event.keyCode === 13) { // enter |
||||
event.preventDefault(); |
||||
|
||||
if (this.active >= this.filteredItems.length) { |
||||
this.createNewBranch(); |
||||
} else if (this.active >= 0) { |
||||
this.selectItem(this.filteredItems[this.active]); |
||||
} |
||||
} else if (event.keyCode === 27) { // escape |
||||
event.preventDefault(); |
||||
this.menuVisible = false; |
||||
} |
||||
} |
||||
} |
||||
}; |
||||
|
||||
export function initRepoBranchTagSelector(selector) { |
||||
for (const [elIndex, elRoot] of document.querySelectorAll(selector).entries()) { |
||||
const data = { |
||||
csrfToken: window.config.csrfToken, |
||||
items: [], |
||||
searchTerm: '', |
||||
refNameText: '', |
||||
menuVisible: false, |
||||
createTag: false, |
||||
release: null, |
||||
|
||||
isViewTag: false, |
||||
isViewBranch: false, |
||||
isViewTree: false, |
||||
|
||||
active: 0, |
||||
|
||||
...window.config.pageData.branchDropdownDataList[elIndex], |
||||
}; |
||||
|
||||
// the "data.defaultBranch" is ambiguous, it could be "branch name" or "tag name" |
||||
|
||||
if (data.showBranchesInDropdown && data.branches) { |
||||
for (const branch of data.branches) { |
||||
data.items.push({name: branch, url: branch, branch: true, tag: false, selected: branch === data.defaultBranch}); |
||||
} |
||||
} |
||||
if (!data.noTag && data.tags) { |
||||
for (const tag of data.tags) { |
||||
if (data.release) { |
||||
data.items.push({name: tag, url: tag, branch: false, tag: true, selected: tag === data.release.tagName}); |
||||
} else { |
||||
data.items.push({name: tag, url: tag, branch: false, tag: true, selected: tag === data.defaultBranch}); |
||||
} |
||||
} |
||||
} |
||||
|
||||
const comp = {...sfc, data() { return data }}; |
||||
createApp(comp).mount(elRoot); |
||||
} |
||||
} |
||||
|
||||
export default sfc; // activate IDE's Vue plugin |
||||
</script> |
@ -0,0 +1,3 @@ |
||||
export function pathEscapeSegments(s) { |
||||
return s.split('/').map(encodeURIComponent).join('/'); |
||||
} |
@ -0,0 +1,7 @@ |
||||
import {expect, test} from 'vitest'; |
||||
import {pathEscapeSegments} from './url.js'; |
||||
|
||||
test('pathEscapeSegments', () => { |
||||
expect(pathEscapeSegments('a/b/c')).toEqual('a/b/c'); |
||||
expect(pathEscapeSegments('a/b/ c')).toEqual('a/b/%20c'); |
||||
}); |
Loading…
Reference in new issue