mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-24 00:02:33 -04:00 
			
		
		
		
	* Improve dashboard's repo list performance - Avoid a lot of database lookups for all the repo's, by adding a undocumented "minimal" mode for this specific task, which returns the data that's only needed by this list which doesn't require any database lookups. - Makes fetching these list faster. - Less CPU overhead when a user visits home page. * Refactor javascript code + fix Fork icon - Use async in the function so we can use `await`. - Remove `archivedFilter` check for count, as it doesn't make sense to show the count of repos when you can't even see them(as they are filited away). * Add `count_only` * Remove uncessary code * Improve comment Co-authored-by: delvh <dev.lh@web.de> * Update web_src/js/components/DashboardRepoList.js Co-authored-by: delvh <dev.lh@web.de> * Update web_src/js/components/DashboardRepoList.js Co-authored-by: delvh <dev.lh@web.de> * By default apply minimal mode * Remove `minimal` paramater * Refactor count header * Simplify init Co-authored-by: wxiaoguang <wxiaoguang@gmail.com> Co-authored-by: delvh <dev.lh@web.de> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com> Co-authored-by: zeripath <art27@cantab.net>
		
			
				
	
	
		
			376 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			376 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import Vue from 'vue';
 | |
| import $ from 'jquery';
 | |
| import {initVueSvg, vueDelimiters} from './VueComponentLoader.js';
 | |
| 
 | |
| const {appSubUrl, assetUrlPrefix, pageData} = window.config;
 | |
| 
 | |
| function initVueComponents() {
 | |
|   Vue.component('repo-search', {
 | |
|     delimiters: vueDelimiters,
 | |
|     props: {
 | |
|       searchLimit: {
 | |
|         type: Number,
 | |
|         default: 10
 | |
|       },
 | |
|       subUrl: {
 | |
|         type: String,
 | |
|         required: true
 | |
|       },
 | |
|       uid: {
 | |
|         type: Number,
 | |
|         default: 0
 | |
|       },
 | |
|       teamId: {
 | |
|         type: Number,
 | |
|         required: false,
 | |
|         default: 0
 | |
|       },
 | |
|       organizations: {
 | |
|         type: Array,
 | |
|         default: () => [],
 | |
|       },
 | |
|       isOrganization: {
 | |
|         type: Boolean,
 | |
|         default: true
 | |
|       },
 | |
|       canCreateOrganization: {
 | |
|         type: Boolean,
 | |
|         default: false
 | |
|       },
 | |
|       organizationsTotalCount: {
 | |
|         type: Number,
 | |
|         default: 0
 | |
|       },
 | |
|       moreReposLink: {
 | |
|         type: String,
 | |
|         default: ''
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     data() {
 | |
|       const params = new URLSearchParams(window.location.search);
 | |
| 
 | |
|       let tab = params.get('repo-search-tab');
 | |
|       if (!tab) {
 | |
|         tab = 'repos';
 | |
|       }
 | |
| 
 | |
|       let reposFilter = params.get('repo-search-filter');
 | |
|       if (!reposFilter) {
 | |
|         reposFilter = 'all';
 | |
|       }
 | |
| 
 | |
|       let privateFilter = params.get('repo-search-private');
 | |
|       if (!privateFilter) {
 | |
|         privateFilter = 'both';
 | |
|       }
 | |
| 
 | |
|       let archivedFilter = params.get('repo-search-archived');
 | |
|       if (!archivedFilter) {
 | |
|         archivedFilter = 'unarchived';
 | |
|       }
 | |
| 
 | |
|       let searchQuery = params.get('repo-search-query');
 | |
|       if (!searchQuery) {
 | |
|         searchQuery = '';
 | |
|       }
 | |
| 
 | |
|       let page = 1;
 | |
|       try {
 | |
|         page = parseInt(params.get('repo-search-page'));
 | |
|       } catch {
 | |
|         // noop
 | |
|       }
 | |
|       if (!page) {
 | |
|         page = 1;
 | |
|       }
 | |
| 
 | |
|       return {
 | |
|         tab,
 | |
|         repos: [],
 | |
|         reposTotalCount: 0,
 | |
|         reposFilter,
 | |
|         archivedFilter,
 | |
|         privateFilter,
 | |
|         page,
 | |
|         finalPage: 1,
 | |
|         searchQuery,
 | |
|         isLoading: false,
 | |
|         staticPrefix: assetUrlPrefix,
 | |
|         counts: {},
 | |
|         repoTypes: {
 | |
|           all: {
 | |
|             searchMode: '',
 | |
|           },
 | |
|           forks: {
 | |
|             searchMode: 'fork',
 | |
|           },
 | |
|           mirrors: {
 | |
|             searchMode: 'mirror',
 | |
|           },
 | |
|           sources: {
 | |
|             searchMode: 'source',
 | |
|           },
 | |
|           collaborative: {
 | |
|             searchMode: 'collaborative',
 | |
|           },
 | |
|         }
 | |
|       };
 | |
|     },
 | |
| 
 | |
|     computed: {
 | |
|       // used in `repolist.tmpl`
 | |
|       showMoreReposLink() {
 | |
|         return this.repos.length > 0 && this.repos.length < this.counts[`${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`];
 | |
|       },
 | |
|       searchURL() {
 | |
|         return `${this.subUrl}/repo/search?sort=updated&order=desc&uid=${this.uid}&team_id=${this.teamId}&q=${this.searchQuery
 | |
|         }&page=${this.page}&limit=${this.searchLimit}&mode=${this.repoTypes[this.reposFilter].searchMode
 | |
|         }${this.reposFilter !== 'all' ? '&exclusive=1' : ''
 | |
|         }${this.archivedFilter === 'archived' ? '&archived=true' : ''}${this.archivedFilter === 'unarchived' ? '&archived=false' : ''
 | |
|         }${this.privateFilter === 'private' ? '&is_private=true' : ''}${this.privateFilter === 'public' ? '&is_private=false' : ''
 | |
|         }`;
 | |
|       },
 | |
|       repoTypeCount() {
 | |
|         return this.counts[`${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`];
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     mounted() {
 | |
|       this.changeReposFilter(this.reposFilter);
 | |
|       $(this.$el).find('.tooltip').popup();
 | |
|       $(this.$el).find('.dropdown').dropdown();
 | |
|       this.setCheckboxes();
 | |
|       Vue.nextTick(() => {
 | |
|         this.$refs.search.focus();
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     methods: {
 | |
|       changeTab(t) {
 | |
|         this.tab = t;
 | |
|         this.updateHistory();
 | |
|       },
 | |
| 
 | |
|       setCheckboxes() {
 | |
|         switch (this.archivedFilter) {
 | |
|           case 'unarchived':
 | |
|             $('#archivedFilterCheckbox').checkbox('set unchecked');
 | |
|             break;
 | |
|           case 'archived':
 | |
|             $('#archivedFilterCheckbox').checkbox('set checked');
 | |
|             break;
 | |
|           case 'both':
 | |
|             $('#archivedFilterCheckbox').checkbox('set indeterminate');
 | |
|             break;
 | |
|           default:
 | |
|             this.archivedFilter = 'unarchived';
 | |
|             $('#archivedFilterCheckbox').checkbox('set unchecked');
 | |
|             break;
 | |
|         }
 | |
|         switch (this.privateFilter) {
 | |
|           case 'public':
 | |
|             $('#privateFilterCheckbox').checkbox('set unchecked');
 | |
|             break;
 | |
|           case 'private':
 | |
|             $('#privateFilterCheckbox').checkbox('set checked');
 | |
|             break;
 | |
|           case 'both':
 | |
|             $('#privateFilterCheckbox').checkbox('set indeterminate');
 | |
|             break;
 | |
|           default:
 | |
|             this.privateFilter = 'both';
 | |
|             $('#privateFilterCheckbox').checkbox('set indeterminate');
 | |
|             break;
 | |
|         }
 | |
|       },
 | |
| 
 | |
|       changeReposFilter(filter) {
 | |
|         this.reposFilter = filter;
 | |
|         this.repos = [];
 | |
|         this.page = 1;
 | |
|         Vue.set(this.counts, `${filter}:${this.archivedFilter}:${this.privateFilter}`, 0);
 | |
|         this.searchRepos();
 | |
|       },
 | |
| 
 | |
|       updateHistory() {
 | |
|         const params = new URLSearchParams(window.location.search);
 | |
| 
 | |
|         if (this.tab === 'repos') {
 | |
|           params.delete('repo-search-tab');
 | |
|         } else {
 | |
|           params.set('repo-search-tab', this.tab);
 | |
|         }
 | |
| 
 | |
|         if (this.reposFilter === 'all') {
 | |
|           params.delete('repo-search-filter');
 | |
|         } else {
 | |
|           params.set('repo-search-filter', this.reposFilter);
 | |
|         }
 | |
| 
 | |
|         if (this.privateFilter === 'both') {
 | |
|           params.delete('repo-search-private');
 | |
|         } else {
 | |
|           params.set('repo-search-private', this.privateFilter);
 | |
|         }
 | |
| 
 | |
|         if (this.archivedFilter === 'unarchived') {
 | |
|           params.delete('repo-search-archived');
 | |
|         } else {
 | |
|           params.set('repo-search-archived', this.archivedFilter);
 | |
|         }
 | |
| 
 | |
|         if (this.searchQuery === '') {
 | |
|           params.delete('repo-search-query');
 | |
|         } else {
 | |
|           params.set('repo-search-query', this.searchQuery);
 | |
|         }
 | |
| 
 | |
|         if (this.page === 1) {
 | |
|           params.delete('repo-search-page');
 | |
|         } else {
 | |
|           params.set('repo-search-page', `${this.page}`);
 | |
|         }
 | |
| 
 | |
|         const queryString = params.toString();
 | |
|         if (queryString) {
 | |
|           window.history.replaceState({}, '', `?${queryString}`);
 | |
|         } else {
 | |
|           window.history.replaceState({}, '', window.location.pathname);
 | |
|         }
 | |
|       },
 | |
| 
 | |
|       toggleArchivedFilter() {
 | |
|         switch (this.archivedFilter) {
 | |
|           case 'both':
 | |
|             this.archivedFilter = 'unarchived';
 | |
|             break;
 | |
|           case 'unarchived':
 | |
|             this.archivedFilter = 'archived';
 | |
|             break;
 | |
|           case 'archived':
 | |
|             this.archivedFilter = 'both';
 | |
|             break;
 | |
|           default:
 | |
|             this.archivedFilter = 'unarchived';
 | |
|             break;
 | |
|         }
 | |
|         this.page = 1;
 | |
|         this.repos = [];
 | |
|         this.setCheckboxes();
 | |
|         Vue.set(this.counts, `${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`, 0);
 | |
|         this.searchRepos();
 | |
|       },
 | |
| 
 | |
|       togglePrivateFilter() {
 | |
|         switch (this.privateFilter) {
 | |
|           case 'both':
 | |
|             this.privateFilter = 'public';
 | |
|             break;
 | |
|           case 'public':
 | |
|             this.privateFilter = 'private';
 | |
|             break;
 | |
|           case 'private':
 | |
|             this.privateFilter = 'both';
 | |
|             break;
 | |
|           default:
 | |
|             this.privateFilter = 'both';
 | |
|             break;
 | |
|         }
 | |
|         this.page = 1;
 | |
|         this.repos = [];
 | |
|         this.setCheckboxes();
 | |
|         Vue.set(this.counts, `${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`, 0);
 | |
|         this.searchRepos();
 | |
|       },
 | |
| 
 | |
| 
 | |
|       changePage(page) {
 | |
|         this.page = page;
 | |
|         if (this.page > this.finalPage) {
 | |
|           this.page = this.finalPage;
 | |
|         }
 | |
|         if (this.page < 1) {
 | |
|           this.page = 1;
 | |
|         }
 | |
|         this.repos = [];
 | |
|         Vue.set(this.counts, `${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`, 0);
 | |
|         this.searchRepos();
 | |
|       },
 | |
| 
 | |
|       async searchRepos() {
 | |
|         this.isLoading = true;
 | |
| 
 | |
|         const searchedMode = this.repoTypes[this.reposFilter].searchMode;
 | |
|         const searchedURL = this.searchURL;
 | |
|         const searchedQuery = this.searchQuery;
 | |
| 
 | |
|         let response, json;
 | |
|         try {
 | |
|           if (!this.reposTotalCount) {
 | |
|             const totalCountSearchURL = `${this.subUrl}/repo/search?count_only=1&uid=${this.uid}&team_id=${this.teamId}&q=&page=1&mode=`;
 | |
|             response = await fetch(totalCountSearchURL);
 | |
|             this.reposTotalCount = response.headers.get('X-Total-Count');
 | |
|           }
 | |
| 
 | |
|           response = await fetch(searchedURL);
 | |
|           json = await response.json();
 | |
|         } catch {
 | |
|           if (searchedURL === this.searchURL) {
 | |
|             this.isLoading = false;
 | |
|           }
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         if (searchedURL === this.searchURL) {
 | |
|           this.repos = json.data;
 | |
|           const count = response.headers.get('X-Total-Count');
 | |
|           if (searchedQuery === '' && searchedMode === '' && this.archivedFilter === 'both') {
 | |
|             this.reposTotalCount = count;
 | |
|           }
 | |
|           Vue.set(this.counts, `${this.reposFilter}:${this.archivedFilter}:${this.privateFilter}`, count);
 | |
|           this.finalPage = Math.ceil(count / this.searchLimit);
 | |
|           this.updateHistory();
 | |
|           this.isLoading = false;
 | |
|         }
 | |
|       },
 | |
| 
 | |
|       repoIcon(repo) {
 | |
|         if (repo.fork) {
 | |
|           return 'octicon-repo-forked';
 | |
|         } else if (repo.mirror) {
 | |
|           return 'octicon-mirror';
 | |
|         } else if (repo.template) {
 | |
|           return `octicon-repo-template`;
 | |
|         } else if (repo.private) {
 | |
|           return 'octicon-lock';
 | |
|         } else if (repo.internal) {
 | |
|           return 'octicon-repo';
 | |
|         }
 | |
|         return 'octicon-repo';
 | |
|       }
 | |
|     }
 | |
|   });
 | |
| }
 | |
| 
 | |
| 
 | |
| export function initDashboardRepoList() {
 | |
|   const el = document.getElementById('dashboard-repo-list');
 | |
|   const dashboardRepoListData = pageData.dashboardRepoList || null;
 | |
|   if (!el || !dashboardRepoListData) return;
 | |
| 
 | |
|   initVueSvg();
 | |
|   initVueComponents();
 | |
|   new Vue({
 | |
|     el,
 | |
|     delimiters: vueDelimiters,
 | |
|     data: () => {
 | |
|       return {
 | |
|         searchLimit: dashboardRepoListData.searchLimit || 0,
 | |
|         subUrl: appSubUrl,
 | |
|         uid: dashboardRepoListData.uid || 0,
 | |
|       };
 | |
|     },
 | |
|   });
 | |
| }
 |