mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-26 16:36:22 +01:00 
			
		
		
		
	Migrated a handful Vue components to the `setup` syntax using composition api as it has better Typescript support and is becoming the new default in the Vue ecosystem. - [x] ActionRunStatus.vue - [x] ActivityHeatmap.vue - [x] ContextPopup.vue - [x] DiffFileList.vue - [x] DiffFileTree.vue - [x] DiffFileTreeItem.vue - [x] PullRequestMergeForm.vue - [x] RepoActivityTopAuthors.vue - [x] RepoCodeFrequency.vue - [x] RepoRecentCommits.vue - [x] ScopedAccessTokenSelector.vue Left some larger components untouched for now to not go to crazy in this single PR: - [ ] DiffCommitSelector.vue - [ ] RepoActionView.vue - [ ] RepoContributors.vue - [ ] DashboardRepoList.vue - [ ] RepoBranchTagSelector.vue
		
			
				
	
	
		
			172 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <script lang="ts" setup>
 | |
| import {SvgIcon} from '../svg.ts';
 | |
| import {
 | |
|   Chart,
 | |
|   Legend,
 | |
|   LinearScale,
 | |
|   TimeScale,
 | |
|   PointElement,
 | |
|   LineElement,
 | |
|   Filler,
 | |
| } from 'chart.js';
 | |
| import {GET} from '../modules/fetch.ts';
 | |
| import {Line as ChartLine} from 'vue-chartjs';
 | |
| import {
 | |
|   startDaysBetween,
 | |
|   firstStartDateAfterDate,
 | |
|   fillEmptyStartDaysWithZeroes,
 | |
|   type DayData,
 | |
| } from '../utils/time.ts';
 | |
| import {chartJsColors} from '../utils/color.ts';
 | |
| import {sleep} from '../utils.ts';
 | |
| import 'chartjs-adapter-dayjs-4/dist/chartjs-adapter-dayjs-4.esm';
 | |
| import {onMounted, ref} from 'vue';
 | |
| 
 | |
| const {pageData} = window.config;
 | |
| 
 | |
| Chart.defaults.color = chartJsColors.text;
 | |
| Chart.defaults.borderColor = chartJsColors.border;
 | |
| 
 | |
| Chart.register(
 | |
|   TimeScale,
 | |
|   LinearScale,
 | |
|   Legend,
 | |
|   PointElement,
 | |
|   LineElement,
 | |
|   Filler,
 | |
| );
 | |
| 
 | |
| defineProps<{
 | |
|   locale: {
 | |
|     loadingTitle: string;
 | |
|     loadingTitleFailed: string;
 | |
|     loadingInfo: string;
 | |
|   };
 | |
| }>();
 | |
| 
 | |
| const isLoading = ref(false);
 | |
| const errorText = ref('');
 | |
| const repoLink = ref(pageData.repoLink || []);
 | |
| const data = ref<DayData[]>([]);
 | |
| 
 | |
| onMounted(() => {
 | |
|   fetchGraphData();
 | |
| });
 | |
| 
 | |
| async function fetchGraphData() {
 | |
|   isLoading.value = true;
 | |
|   try {
 | |
|     let response: Response;
 | |
|     do {
 | |
|       response = await GET(`${repoLink.value}/activity/code-frequency/data`);
 | |
|       if (response.status === 202) {
 | |
|         await sleep(1000); // wait for 1 second before retrying
 | |
|       }
 | |
|     } while (response.status === 202);
 | |
|     if (response.ok) {
 | |
|       data.value = await response.json();
 | |
|       const weekValues = Object.values(data.value);
 | |
|       const start = weekValues[0].week;
 | |
|       const end = firstStartDateAfterDate(new Date());
 | |
|       const startDays = startDaysBetween(start, end);
 | |
|       data.value = fillEmptyStartDaysWithZeroes(startDays, data.value);
 | |
|       errorText.value = '';
 | |
|     } else {
 | |
|       errorText.value = response.statusText;
 | |
|     }
 | |
|   } catch (err) {
 | |
|     errorText.value = err.message;
 | |
|   } finally {
 | |
|     isLoading.value = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| function toGraphData(data) {
 | |
|   return {
 | |
|     datasets: [
 | |
|       {
 | |
|         data: data.map((i) => ({x: i.week, y: i.additions})),
 | |
|         pointRadius: 0,
 | |
|         pointHitRadius: 0,
 | |
|         fill: true,
 | |
|         label: 'Additions',
 | |
|         backgroundColor: chartJsColors['additions'],
 | |
|         borderWidth: 0,
 | |
|         tension: 0.3,
 | |
|       },
 | |
|       {
 | |
|         data: data.map((i) => ({x: i.week, y: -i.deletions})),
 | |
|         pointRadius: 0,
 | |
|         pointHitRadius: 0,
 | |
|         fill: true,
 | |
|         label: 'Deletions',
 | |
|         backgroundColor: chartJsColors['deletions'],
 | |
|         borderWidth: 0,
 | |
|         tension: 0.3,
 | |
|       },
 | |
|     ],
 | |
|   };
 | |
| }
 | |
| 
 | |
| const options = {
 | |
|   responsive: true,
 | |
|   maintainAspectRatio: false,
 | |
|   animation: true,
 | |
|   plugins: {
 | |
|     legend: {
 | |
|       display: true,
 | |
|     },
 | |
|   },
 | |
|   scales: {
 | |
|     x: {
 | |
|       type: 'time',
 | |
|       grid: {
 | |
|         display: false,
 | |
|       },
 | |
|       time: {
 | |
|         minUnit: 'month',
 | |
|       },
 | |
|       ticks: {
 | |
|         maxRotation: 0,
 | |
|         maxTicksLimit: 12,
 | |
|       },
 | |
|     },
 | |
|     y: {
 | |
|       ticks: {
 | |
|         maxTicksLimit: 6,
 | |
|       },
 | |
|     },
 | |
|   },
 | |
| };
 | |
| </script>
 | |
| 
 | |
| <template>
 | |
|   <div>
 | |
|     <div class="ui header tw-flex tw-items-center tw-justify-between">
 | |
|       {{ isLoading ? locale.loadingTitle : errorText ? locale.loadingTitleFailed: `Code frequency over the history of ${repoLink.slice(1)}` }}
 | |
|     </div>
 | |
|     <div class="tw-flex ui segment main-graph">
 | |
|       <div v-if="isLoading || errorText !== ''" class="gt-tc tw-m-auto">
 | |
|         <div v-if="isLoading">
 | |
|           <SvgIcon name="octicon-sync" class="tw-mr-2 job-status-rotate"/>
 | |
|           {{ locale.loadingInfo }}
 | |
|         </div>
 | |
|         <div v-else class="text red">
 | |
|           <SvgIcon name="octicon-x-circle-fill"/>
 | |
|           {{ errorText }}
 | |
|         </div>
 | |
|       </div>
 | |
|       <ChartLine
 | |
|         v-memo="data" v-if="data.length !== 0"
 | |
|         :data="toGraphData(data)" :options="options"
 | |
|       />
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <style scoped>
 | |
| .main-graph {
 | |
|   height: 440px;
 | |
| }
 | |
| </style>
 |