forked from go-gitea/gitea
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Load issue/PR context popup data only when needed (go-gitea#15955)
* Load issue/PR context popup data only when needed * Add SVG icon Vue component * Remove unneeded check
- Loading branch information
1 parent
224ef4b
commit b4a38dc
Showing
3 changed files
with
152 additions
and
55 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
<template> | ||
<div> | ||
<div v-if="loading" class="ui active centered inline loader"/> | ||
<div v-if="!loading && issue !== null"> | ||
<p><small>{{ issue.repository.full_name }} on {{ createdAt }}</small></p> | ||
<p><svg-icon :name="icon" :class="[color]" /> <strong>{{ issue.title }}</strong> #{{ issue.number }}</p> | ||
<p>{{ body }}</p> | ||
<div> | ||
<div | ||
v-for="label in labels" | ||
:key="label.name" | ||
class="ui label" | ||
:style="{ color: label.textColor, backgroundColor: label.color }" | ||
> | ||
{{ label.name }} | ||
</div> | ||
</div> | ||
</div> | ||
</div> | ||
</template> | ||
|
||
<script> | ||
import {SvgIcon} from '../svg.js'; | ||
const {AppSubUrl} = window.config; | ||
export default { | ||
name: 'ContextPopup', | ||
components: { | ||
SvgIcon, | ||
}, | ||
data: () => ({ | ||
loading: false, | ||
issue: null | ||
}), | ||
computed: { | ||
createdAt() { | ||
return new Date(this.issue.created_at).toLocaleDateString(undefined, {year: 'numeric', month: 'short', day: 'numeric'}); | ||
}, | ||
body() { | ||
const body = this.issue.body.replace(/\n+/g, ' '); | ||
if (body.length > 85) { | ||
return `${body.substring(0, 85)}…`; | ||
} | ||
return body; | ||
}, | ||
icon() { | ||
if (this.issue.pull_request !== null) { | ||
if (this.issue.state === 'open') { | ||
return 'octicon-git-pull-request'; // Open PR | ||
} else if (this.issue.pull_request.merged === true) { | ||
return 'octicon-git-merge'; // Merged PR | ||
} | ||
return 'octicon-git-pull-request'; // Closed PR | ||
} else if (this.issue.state === 'open') { | ||
return 'octicon-issue-opened'; // Open Issue | ||
} | ||
return 'octicon-issue-closed'; // Closed Issue | ||
}, | ||
color() { | ||
if (this.issue.state === 'open') { | ||
return 'green'; | ||
} else if (this.issue.pull_request !== null && this.issue.pull_request.merged === true) { | ||
return 'purple'; | ||
} | ||
return 'red'; | ||
}, | ||
labels() { | ||
return this.issue.labels.map((label) => { | ||
const red = parseInt(label.color.substring(0, 2), 16); | ||
const green = parseInt(label.color.substring(2, 4), 16); | ||
const blue = parseInt(label.color.substring(4, 6), 16); | ||
let color = '#ffffff'; | ||
if ((red * 0.299 + green * 0.587 + blue * 0.114) > 125) { | ||
color = '#000000'; | ||
} | ||
return {name: label.name, color: `#${label.color}`, textColor: color}; | ||
}); | ||
} | ||
}, | ||
mounted() { | ||
this.$root.$on('load-context-popup', (data, callback) => { | ||
if (!this.loading && this.issue === null) { | ||
this.load(data, callback); | ||
} | ||
}); | ||
}, | ||
methods: { | ||
load(data, callback) { | ||
this.loading = true; | ||
$.get(`${AppSubUrl}/api/v1/repos/${data.owner}/${data.repo}/issues/${data.index}`, (issue) => { | ||
this.issue = issue; | ||
this.loading = false; | ||
this.$nextTick(() => { | ||
if (callback) { | ||
callback(); | ||
} | ||
}); | ||
}); | ||
} | ||
} | ||
}; | ||
</script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,76 +1,43 @@ | ||
import {htmlEscape} from 'escape-goat'; | ||
import {svg} from '../svg.js'; | ||
import Vue from 'vue'; | ||
|
||
const {AppSubUrl} = window.config; | ||
import ContextPopup from '../components/ContextPopup.vue'; | ||
|
||
export default function initContextPopups() { | ||
const refIssues = $('.ref-issue'); | ||
if (!refIssues.length) return; | ||
|
||
refIssues.each(function () { | ||
const [index, _issues, repo, owner] = $(this).attr('href').replace(/[#?].*$/, '').split('/').reverse(); | ||
issuePopup(owner, repo, index, $(this)); | ||
}); | ||
} | ||
|
||
function issuePopup(owner, repo, index, $element) { | ||
$.get(`${AppSubUrl}/api/v1/repos/${owner}/${repo}/issues/${index}`, (issue) => { | ||
const createdAt = new Date(issue.created_at).toLocaleDateString(undefined, {year: 'numeric', month: 'short', day: 'numeric'}); | ||
const el = document.createElement('div'); | ||
el.className = 'ui custom popup hidden'; | ||
el.innerHTML = '<div></div>'; | ||
this.parentNode.insertBefore(el, this.nextSibling); | ||
|
||
let body = issue.body.replace(/\n+/g, ' '); | ||
if (body.length > 85) { | ||
body = `${body.substring(0, 85)}...`; | ||
} | ||
const View = Vue.extend({ | ||
render: (createElement) => createElement(ContextPopup), | ||
}); | ||
|
||
let labels = ''; | ||
for (let i = 0; i < issue.labels.length; i++) { | ||
const label = issue.labels[i]; | ||
const red = parseInt(label.color.substring(0, 2), 16); | ||
const green = parseInt(label.color.substring(2, 4), 16); | ||
const blue = parseInt(label.color.substring(4, 6), 16); | ||
let color = '#ffffff'; | ||
if ((red * 0.299 + green * 0.587 + blue * 0.114) > 125) { | ||
color = '#000000'; | ||
} | ||
labels += `<div class="ui label" style="color: ${color}; background-color:#${label.color};">${htmlEscape(label.name)}</div>`; | ||
} | ||
if (labels.length > 0) { | ||
labels = `<p>${labels}</p>`; | ||
} | ||
const view = new View(); | ||
|
||
let octicon, color; | ||
if (issue.pull_request !== null) { | ||
if (issue.state === 'open') { | ||
color = 'green'; | ||
octicon = 'octicon-git-pull-request'; // Open PR | ||
} else if (issue.pull_request.merged === true) { | ||
color = 'purple'; | ||
octicon = 'octicon-git-merge'; // Merged PR | ||
} else { | ||
color = 'red'; | ||
octicon = 'octicon-git-pull-request'; // Closed PR | ||
} | ||
} else if (issue.state === 'open') { | ||
color = 'green'; | ||
octicon = 'octicon-issue-opened'; // Open Issue | ||
} else { | ||
color = 'red'; | ||
octicon = 'octicon-issue-closed'; // Closed Issue | ||
try { | ||
view.$mount(el.firstChild); | ||
} catch (err) { | ||
console.error(err); | ||
el.textContent = 'ContextPopup failed to load'; | ||
} | ||
|
||
$element.popup({ | ||
$(this).popup({ | ||
variation: 'wide', | ||
delay: { | ||
show: 250 | ||
}, | ||
html: ` | ||
<div> | ||
<p><small>${htmlEscape(issue.repository.full_name)} on ${createdAt}</small></p> | ||
<p><span class="${color}">${svg(octicon)}</span> <strong>${htmlEscape(issue.title)}</strong> #${index}</p> | ||
<p>${htmlEscape(body)}</p> | ||
${labels} | ||
</div> | ||
` | ||
onShow: () => { | ||
view.$emit('load-context-popup', {owner, repo, index}, () => { | ||
$(this).popup('reposition'); | ||
}); | ||
}, | ||
popup: $(el), | ||
}); | ||
}); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters