mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-11-04 01:34:27 +00:00 
			
		
		
		
	
							
								
								
									
										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);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user