mirror of
https://github.com/go-gitea/gitea.git
synced 2024-11-28 06:16:31 +01:00
parent
348d1d0f32
commit
b7fb20e73e
23
package-lock.json
generated
23
package-lock.json
generated
@ -11,7 +11,7 @@
|
|||||||
"@citation-js/plugin-software-formats": "0.6.1",
|
"@citation-js/plugin-software-formats": "0.6.1",
|
||||||
"@github/markdown-toolbar-element": "2.2.3",
|
"@github/markdown-toolbar-element": "2.2.3",
|
||||||
"@github/relative-time-element": "4.4.3",
|
"@github/relative-time-element": "4.4.3",
|
||||||
"@github/text-expander-element": "2.7.1",
|
"@github/text-expander-element": "2.8.0",
|
||||||
"@mcaptcha/vanilla-glue": "0.1.0-alpha-3",
|
"@mcaptcha/vanilla-glue": "0.1.0-alpha-3",
|
||||||
"@primer/octicons": "19.11.0",
|
"@primer/octicons": "19.11.0",
|
||||||
"@silverwind/vue3-calendar-heatmap": "2.0.6",
|
"@silverwind/vue3-calendar-heatmap": "2.0.6",
|
||||||
@ -40,6 +40,7 @@
|
|||||||
"monaco-editor": "0.51.0",
|
"monaco-editor": "0.51.0",
|
||||||
"monaco-editor-webpack-plugin": "7.1.0",
|
"monaco-editor-webpack-plugin": "7.1.0",
|
||||||
"pdfobject": "2.3.0",
|
"pdfobject": "2.3.0",
|
||||||
|
"perfect-debounce": "1.0.0",
|
||||||
"postcss": "8.4.41",
|
"postcss": "8.4.41",
|
||||||
"postcss-loader": "8.1.1",
|
"postcss-loader": "8.1.1",
|
||||||
"postcss-nesting": "13.0.0",
|
"postcss-nesting": "13.0.0",
|
||||||
@ -3115,13 +3116,13 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@github/text-expander-element": {
|
"node_modules/@github/text-expander-element": {
|
||||||
"version": "2.7.1",
|
"version": "2.8.0",
|
||||||
"resolved": "https://registry.npmjs.org/@github/text-expander-element/-/text-expander-element-2.7.1.tgz",
|
"resolved": "https://registry.npmjs.org/@github/text-expander-element/-/text-expander-element-2.8.0.tgz",
|
||||||
"integrity": "sha512-CWxfYxJRkeWVCUhJveproLs6pHsPrWtK8TsjL8ByYVcSCs8CJmNzF8b7ZawrUgfai0F2jb4aIdw2FoBTykj9XA==",
|
"integrity": "sha512-kkS2rZ/CG8HGKblpLDQ8vcK/K7l/Jsvzi/N4ovwPAsFSOImcIbJh2MgCv9tzqE3wAm/qXlscvh3Ms4Hh1vtZvw==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@github/combobox-nav": "^2.0.2",
|
"@github/combobox-nav": "^2.0.2",
|
||||||
"dom-input-range": "^1.1.6"
|
"dom-input-range": "^1.2.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@humanwhocodes/config-array": {
|
"node_modules/@humanwhocodes/config-array": {
|
||||||
@ -7409,9 +7410,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/dom-input-range": {
|
"node_modules/dom-input-range": {
|
||||||
"version": "1.1.6",
|
"version": "1.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/dom-input-range/-/dom-input-range-1.1.6.tgz",
|
"resolved": "https://registry.npmjs.org/dom-input-range/-/dom-input-range-1.2.0.tgz",
|
||||||
"integrity": "sha512-4o/SkTpscD0n81BeErrrtmE58lG8vTks++92vk//ld0NmkQTb4AVJ2rexh2yor6rtBf5IMte26u+fF3EgCppPQ==",
|
"integrity": "sha512-8HVA5Oy5Vt872S7IXsjjp6/5Hqsm5YZLhurxwwQXp80T9qVsj8/mEUH3sQlFujLLUoWfxiaThHHuJ3/q1MHVuA==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"workspaces": [
|
"workspaces": [
|
||||||
"demos"
|
"demos"
|
||||||
@ -12460,6 +12461,12 @@
|
|||||||
"integrity": "sha512-w/9pXDXTDs3IDmOri/w8lM/w6LHR0/F4fcBLLzH+4csSoyshQ5su0TE7k0FLHZO7aOjVLDGecqd1M89+PVpVAA==",
|
"integrity": "sha512-w/9pXDXTDs3IDmOri/w8lM/w6LHR0/F4fcBLLzH+4csSoyshQ5su0TE7k0FLHZO7aOjVLDGecqd1M89+PVpVAA==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/perfect-debounce": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/perfect-debounce/-/perfect-debounce-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-xCy9V055GLEqoFaHoC1SoLIaLmWctgCUaBaWxDZ7/Zx4CTyX7cJQLJOok/orfjZAh9kEYpjJa4d0KcJmCbctZA==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/picocolors": {
|
"node_modules/picocolors": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
||||||
|
@ -10,7 +10,7 @@
|
|||||||
"@citation-js/plugin-software-formats": "0.6.1",
|
"@citation-js/plugin-software-formats": "0.6.1",
|
||||||
"@github/markdown-toolbar-element": "2.2.3",
|
"@github/markdown-toolbar-element": "2.2.3",
|
||||||
"@github/relative-time-element": "4.4.3",
|
"@github/relative-time-element": "4.4.3",
|
||||||
"@github/text-expander-element": "2.7.1",
|
"@github/text-expander-element": "2.8.0",
|
||||||
"@mcaptcha/vanilla-glue": "0.1.0-alpha-3",
|
"@mcaptcha/vanilla-glue": "0.1.0-alpha-3",
|
||||||
"@primer/octicons": "19.11.0",
|
"@primer/octicons": "19.11.0",
|
||||||
"@silverwind/vue3-calendar-heatmap": "2.0.6",
|
"@silverwind/vue3-calendar-heatmap": "2.0.6",
|
||||||
@ -39,6 +39,7 @@
|
|||||||
"monaco-editor": "0.51.0",
|
"monaco-editor": "0.51.0",
|
||||||
"monaco-editor-webpack-plugin": "7.1.0",
|
"monaco-editor-webpack-plugin": "7.1.0",
|
||||||
"pdfobject": "2.3.0",
|
"pdfobject": "2.3.0",
|
||||||
|
"perfect-debounce": "1.0.0",
|
||||||
"postcss": "8.4.41",
|
"postcss": "8.4.41",
|
||||||
"postcss-loader": "8.1.1",
|
"postcss-loader": "8.1.1",
|
||||||
"postcss-nesting": "13.0.0",
|
"postcss-nesting": "13.0.0",
|
||||||
|
93
routers/web/repo/issue_suggestions.go
Normal file
93
routers/web/repo/issue_suggestions.go
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package repo
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/db"
|
||||||
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
|
"code.gitea.io/gitea/models/unit"
|
||||||
|
issue_indexer "code.gitea.io/gitea/modules/indexer/issues"
|
||||||
|
"code.gitea.io/gitea/modules/optional"
|
||||||
|
"code.gitea.io/gitea/services/context"
|
||||||
|
)
|
||||||
|
|
||||||
|
type issueSuggestion struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
State string `json:"state"`
|
||||||
|
PullRequest *struct {
|
||||||
|
Merged bool `json:"merged"`
|
||||||
|
Draft bool `json:"draft"`
|
||||||
|
} `json:"pull_request,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// IssueSuggestions returns a list of issue suggestions
|
||||||
|
func IssueSuggestions(ctx *context.Context) {
|
||||||
|
keyword := ctx.Req.FormValue("q")
|
||||||
|
|
||||||
|
canReadIssues := ctx.Repo.CanRead(unit.TypeIssues)
|
||||||
|
canReadPulls := ctx.Repo.CanRead(unit.TypePullRequests)
|
||||||
|
|
||||||
|
var isPull optional.Option[bool]
|
||||||
|
if canReadPulls && !canReadIssues {
|
||||||
|
isPull = optional.Some(true)
|
||||||
|
} else if canReadIssues && !canReadPulls {
|
||||||
|
isPull = optional.Some(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
searchOpt := &issue_indexer.SearchOptions{
|
||||||
|
Paginator: &db.ListOptions{
|
||||||
|
Page: 0,
|
||||||
|
PageSize: 5,
|
||||||
|
},
|
||||||
|
Keyword: keyword,
|
||||||
|
RepoIDs: []int64{ctx.Repo.Repository.ID},
|
||||||
|
IsPull: isPull,
|
||||||
|
IsClosed: nil,
|
||||||
|
SortBy: issue_indexer.SortByUpdatedDesc,
|
||||||
|
}
|
||||||
|
|
||||||
|
ids, _, err := issue_indexer.SearchIssues(ctx, searchOpt)
|
||||||
|
if err != nil {
|
||||||
|
ctx.ServerError("SearchIssues", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
issues, err := issues_model.GetIssuesByIDs(ctx, ids, true)
|
||||||
|
if err != nil {
|
||||||
|
ctx.ServerError("FindIssuesByIDs", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
suggestions := make([]*issueSuggestion, 0, len(issues))
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
suggestion := &issueSuggestion{
|
||||||
|
ID: issue.ID,
|
||||||
|
Title: issue.Title,
|
||||||
|
State: string(issue.State()),
|
||||||
|
}
|
||||||
|
|
||||||
|
if issue.IsPull {
|
||||||
|
if err := issue.LoadPullRequest(ctx); err != nil {
|
||||||
|
ctx.ServerError("LoadPullRequest", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if issue.PullRequest != nil {
|
||||||
|
suggestion.PullRequest = &struct {
|
||||||
|
Merged bool `json:"merged"`
|
||||||
|
Draft bool `json:"draft"`
|
||||||
|
}{
|
||||||
|
Merged: issue.PullRequest.HasMerged,
|
||||||
|
Draft: issue.PullRequest.IsWorkInProgress(ctx),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suggestions = append(suggestions, suggestion)
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, suggestions)
|
||||||
|
}
|
@ -1178,6 +1178,7 @@ func registerRoutes(m *web.Router) {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
}, context.RepoRef())
|
}, context.RepoRef())
|
||||||
|
m.Get("/issues/suggestions", repo.IssueSuggestions)
|
||||||
}, ignSignIn, context.RepoAssignment, reqRepoIssuesOrPullsReader)
|
}, ignSignIn, context.RepoAssignment, reqRepoIssuesOrPullsReader)
|
||||||
// end "/{username}/{reponame}": view milestone, label, issue, pull, etc
|
// end "/{username}/{reponame}": view milestone, label, issue, pull, etc
|
||||||
|
|
||||||
|
@ -44,7 +44,7 @@ Template Attributes:
|
|||||||
<button class="markdown-toolbar-button markdown-switch-easymde" data-tooltip-content="{{ctx.Locale.Tr "editor.buttons.switch_to_legacy.tooltip"}}">{{svg "octicon-arrow-switch"}}</button>
|
<button class="markdown-toolbar-button markdown-switch-easymde" data-tooltip-content="{{ctx.Locale.Tr "editor.buttons.switch_to_legacy.tooltip"}}">{{svg "octicon-arrow-switch"}}</button>
|
||||||
</div>
|
</div>
|
||||||
</markdown-toolbar>
|
</markdown-toolbar>
|
||||||
<text-expander keys=": @" suffix="">
|
<text-expander keys=": @ #" multiword="#" suffix="">
|
||||||
<textarea class="markdown-text-editor"{{if .TextareaName}} name="{{.TextareaName}}"{{end}}{{if .TextareaPlaceholder}} placeholder="{{.TextareaPlaceholder}}"{{end}}{{if .TextareaAriaLabel}} aria-label="{{.TextareaAriaLabel}}"{{end}}{{if .DisableAutosize}} data-disable-autosize="{{.DisableAutosize}}"{{end}}>{{.TextareaContent}}</textarea>
|
<textarea class="markdown-text-editor"{{if .TextareaName}} name="{{.TextareaName}}"{{end}}{{if .TextareaPlaceholder}} placeholder="{{.TextareaPlaceholder}}"{{end}}{{if .TextareaAriaLabel}} aria-label="{{.TextareaAriaLabel}}"{{end}}{{if .DisableAutosize}} data-disable-autosize="{{.DisableAutosize}}"{{end}}>{{.TextareaContent}}</textarea>
|
||||||
</text-expander>
|
</text-expander>
|
||||||
<script>
|
<script>
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import {SvgIcon} from '../svg.ts';
|
import {SvgIcon} from '../svg.ts';
|
||||||
import {GET} from '../modules/fetch.ts';
|
import {GET} from '../modules/fetch.ts';
|
||||||
|
import {getIssueColor, getIssueIcon} from '../features/issue.ts';
|
||||||
import {computed, onMounted, ref} from 'vue';
|
import {computed, onMounted, ref} from 'vue';
|
||||||
import type {Issue} from '../types';
|
|
||||||
|
|
||||||
const {appSubUrl, i18n} = window.config;
|
const {appSubUrl, i18n} = window.config;
|
||||||
|
|
||||||
@ -21,37 +21,6 @@ const body = computed(() => {
|
|||||||
return body;
|
return body;
|
||||||
});
|
});
|
||||||
|
|
||||||
function getIssueIcon(issue: Issue) {
|
|
||||||
if (issue.pull_request) {
|
|
||||||
if (issue.state === 'open') {
|
|
||||||
if (issue.pull_request.draft === true) {
|
|
||||||
return 'octicon-git-pull-request-draft'; // WIP PR
|
|
||||||
}
|
|
||||||
return 'octicon-git-pull-request'; // Open PR
|
|
||||||
} else if (issue.pull_request.merged === true) {
|
|
||||||
return 'octicon-git-merge'; // Merged PR
|
|
||||||
}
|
|
||||||
return 'octicon-git-pull-request'; // Closed PR
|
|
||||||
} else if (issue.state === 'open') {
|
|
||||||
return 'octicon-issue-opened'; // Open Issue
|
|
||||||
}
|
|
||||||
return 'octicon-issue-closed'; // Closed Issue
|
|
||||||
}
|
|
||||||
|
|
||||||
function getIssueColor(issue: Issue) {
|
|
||||||
if (issue.pull_request) {
|
|
||||||
if (issue.pull_request.draft === true) {
|
|
||||||
return 'grey'; // WIP PR
|
|
||||||
} else if (issue.pull_request.merged === true) {
|
|
||||||
return 'purple'; // Merged PR
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (issue.state === 'open') {
|
|
||||||
return 'green'; // Open Issue
|
|
||||||
}
|
|
||||||
return 'red'; // Closed Issue
|
|
||||||
}
|
|
||||||
|
|
||||||
const root = ref<HTMLElement | null>(null);
|
const root = ref<HTMLElement | null>(null);
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
|
@ -1,5 +1,41 @@
|
|||||||
import {matchEmoji, matchMention} from '../../utils/match.ts';
|
import {matchEmoji, matchMention, matchIssue} from '../../utils/match.ts';
|
||||||
import {emojiString} from '../emoji.ts';
|
import {emojiString} from '../emoji.ts';
|
||||||
|
import {svg} from '../../svg.ts';
|
||||||
|
import {parseIssueHref} from '../../utils.ts';
|
||||||
|
import {createElementFromAttrs, createElementFromHTML} from '../../utils/dom.ts';
|
||||||
|
import {getIssueColor, getIssueIcon} from '../issue.ts';
|
||||||
|
import {debounce} from 'perfect-debounce';
|
||||||
|
|
||||||
|
const debouncedSuggestIssues = debounce((key: string, text: string) => new Promise<{matched:boolean; fragment?: HTMLElement}>(async (resolve) => {
|
||||||
|
const {owner, repo, index} = parseIssueHref(window.location.href);
|
||||||
|
const matches = await matchIssue(owner, repo, index, text);
|
||||||
|
if (!matches.length) return resolve({matched: false});
|
||||||
|
|
||||||
|
const ul = document.createElement('ul');
|
||||||
|
ul.classList.add('suggestions');
|
||||||
|
for (const issue of matches) {
|
||||||
|
const li = createElementFromAttrs('li', {
|
||||||
|
role: 'option',
|
||||||
|
'data-value': `${key}${issue.id}`,
|
||||||
|
class: 'tw-flex tw-gap-2',
|
||||||
|
});
|
||||||
|
|
||||||
|
const icon = svg(getIssueIcon(issue), 16, ['text', getIssueColor(issue)].join(' '));
|
||||||
|
li.append(createElementFromHTML(icon));
|
||||||
|
|
||||||
|
const id = document.createElement('span');
|
||||||
|
id.textContent = issue.id.toString();
|
||||||
|
li.append(id);
|
||||||
|
|
||||||
|
const nameSpan = document.createElement('span');
|
||||||
|
nameSpan.textContent = issue.title;
|
||||||
|
li.append(nameSpan);
|
||||||
|
|
||||||
|
ul.append(li);
|
||||||
|
}
|
||||||
|
|
||||||
|
resolve({matched: true, fragment: ul});
|
||||||
|
}), 100);
|
||||||
|
|
||||||
export function initTextExpander(expander) {
|
export function initTextExpander(expander) {
|
||||||
expander?.addEventListener('text-expander-change', ({detail: {key, provide, text}}) => {
|
expander?.addEventListener('text-expander-change', ({detail: {key, provide, text}}) => {
|
||||||
@ -49,12 +85,14 @@ export function initTextExpander(expander) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
provide({matched: true, fragment: ul});
|
provide({matched: true, fragment: ul});
|
||||||
|
} else if (key === '#') {
|
||||||
|
provide(debouncedSuggestIssues(key, text));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
expander?.addEventListener('text-expander-value', ({detail}) => {
|
expander?.addEventListener('text-expander-value', ({detail}) => {
|
||||||
if (detail?.item) {
|
if (detail?.item) {
|
||||||
// add a space after @mentions as it's likely the user wants one
|
// add a space after @mentions and #issue as it's likely the user wants one
|
||||||
const suffix = detail.key === '@' ? ' ' : '';
|
const suffix = ['@', '#'].includes(detail.key) ? ' ' : '';
|
||||||
detail.value = `${detail.item.getAttribute('data-value')}${suffix}`;
|
detail.value = `${detail.item.getAttribute('data-value')}${suffix}`;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
32
web_src/js/features/issue.ts
Normal file
32
web_src/js/features/issue.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import type {Issue} from '../types.ts';
|
||||||
|
|
||||||
|
export function getIssueIcon(issue: Issue) {
|
||||||
|
if (issue.pull_request) {
|
||||||
|
if (issue.state === 'open') {
|
||||||
|
if (issue.pull_request.draft === true) {
|
||||||
|
return 'octicon-git-pull-request-draft'; // WIP PR
|
||||||
|
}
|
||||||
|
return 'octicon-git-pull-request'; // Open PR
|
||||||
|
} else if (issue.pull_request.merged === true) {
|
||||||
|
return 'octicon-git-merge'; // Merged PR
|
||||||
|
}
|
||||||
|
return 'octicon-git-pull-request'; // Closed PR
|
||||||
|
} else if (issue.state === 'open') {
|
||||||
|
return 'octicon-issue-opened'; // Open Issue
|
||||||
|
}
|
||||||
|
return 'octicon-issue-closed'; // Closed Issue
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getIssueColor(issue: Issue) {
|
||||||
|
if (issue.pull_request) {
|
||||||
|
if (issue.pull_request.draft === true) {
|
||||||
|
return 'grey'; // WIP PR
|
||||||
|
} else if (issue.pull_request.merged === true) {
|
||||||
|
return 'purple'; // Merged PR
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (issue.state === 'open') {
|
||||||
|
return 'green'; // Open Issue
|
||||||
|
}
|
||||||
|
return 'red'; // Closed Issue
|
||||||
|
}
|
@ -1,8 +1,10 @@
|
|||||||
import emojis from '../../../assets/emoji.json';
|
import emojis from '../../../assets/emoji.json';
|
||||||
|
import type {Issue} from '../features/issue.ts';
|
||||||
|
import {GET} from '../modules/fetch.ts';
|
||||||
|
|
||||||
const maxMatches = 6;
|
const maxMatches = 6;
|
||||||
|
|
||||||
function sortAndReduce(map: Map<string, number>) {
|
function sortAndReduce<T>(map: Map<T, number>): T[] {
|
||||||
const sortedMap = new Map(Array.from(map.entries()).sort((a, b) => a[1] - b[1]));
|
const sortedMap = new Map(Array.from(map.entries()).sort((a, b) => a[1] - b[1]));
|
||||||
return Array.from(sortedMap.keys()).slice(0, maxMatches);
|
return Array.from(sortedMap.keys()).slice(0, maxMatches);
|
||||||
}
|
}
|
||||||
@ -27,11 +29,12 @@ export function matchEmoji(queryText: string): string[] {
|
|||||||
return sortAndReduce(results);
|
return sortAndReduce(results);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function matchMention(queryText: string): string[] {
|
type MentionSuggestion = {value: string; name: string; fullname: string; avatar: string};
|
||||||
|
export function matchMention(queryText: string): MentionSuggestion[] {
|
||||||
const query = queryText.toLowerCase();
|
const query = queryText.toLowerCase();
|
||||||
|
|
||||||
// results is a map of weights, lower is better
|
// results is a map of weights, lower is better
|
||||||
const results = new Map();
|
const results = new Map<MentionSuggestion, number>();
|
||||||
for (const obj of window.config.mentionValues ?? []) {
|
for (const obj of window.config.mentionValues ?? []) {
|
||||||
const index = obj.key.toLowerCase().indexOf(query);
|
const index = obj.key.toLowerCase().indexOf(query);
|
||||||
if (index === -1) continue;
|
if (index === -1) continue;
|
||||||
@ -41,3 +44,13 @@ export function matchMention(queryText: string): string[] {
|
|||||||
|
|
||||||
return sortAndReduce(results);
|
return sortAndReduce(results);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function matchIssue(owner: string, repo: string, issueIndexStr: string, query: string): Promise<Issue[]> {
|
||||||
|
const res = await GET(`${window.config.appSubUrl}/${owner}/${repo}/issues/suggestions?q=${encodeURIComponent(query)}`);
|
||||||
|
|
||||||
|
const issues: Issue[] = await res.json();
|
||||||
|
const issueIndex = parseInt(issueIndexStr);
|
||||||
|
|
||||||
|
// filter out issue with same id
|
||||||
|
return issues.filter((i) => i.id !== issueIndex);
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user