Files
git-velocity/web/src/views/Repository.vue
T
2025-12-12 23:07:29 +00:00

182 lines
6.6 KiB
Vue

<script setup>
import { ref, computed, onMounted, watch } from 'vue'
import { useRoute } from 'vue-router'
import PageHeader from '../components/PageHeader.vue'
import LoadingState from '../components/LoadingState.vue'
import ErrorState from '../components/ErrorState.vue'
import StatCard from '../components/StatCard.vue'
import DataTable from '../components/DataTable.vue'
import ContributorRow from '../components/ContributorRow.vue'
import SectionHeader from '../components/SectionHeader.vue'
import GithubLink from '../components/GithubLink.vue'
import { formatNumber } from '../composables/formatters'
const route = useRoute()
const repository = ref(null)
const loading = ref(true)
const error = ref(null)
const searchQuery = ref('')
const allContributors = computed(() => repository.value?.contributors || [])
const filteredContributors = computed(() => {
if (!searchQuery.value.trim()) return allContributors.value
const query = searchQuery.value.toLowerCase().trim()
return allContributors.value.filter(contributor => {
const name = (contributor.name || '').toLowerCase()
const login = (contributor.login || '').toLowerCase()
return name.includes(query) || login.includes(query)
})
})
const breadcrumbs = computed(() => [
{ label: 'Dashboard', to: '/' },
{ label: 'Repositories' },
{ label: repository.value?.name || route.params.name }
])
const tableColumns = [
{ key: 'contributor', label: 'Contributor', align: 'left' },
{ key: 'commits', label: 'Commits', align: 'center' },
{ key: 'prs', label: 'PRs', align: 'center' },
{ key: 'reviews', label: 'Reviews', align: 'center' },
{ key: 'lines', label: 'Lines +/-', align: 'center' },
{ key: 'score', label: 'Score', align: 'right' }
]
async function loadRepository() {
loading.value = true
error.value = null
try {
const response = await fetch(`./data/repos/${route.params.owner}/${route.params.name}/metrics.json`)
if (!response.ok) throw new Error('Repository not found')
repository.value = await response.json()
} catch (e) {
error.value = e.message
} finally {
loading.value = false
}
}
onMounted(loadRepository)
watch(() => route.params, loadRepository)
</script>
<template>
<div>
<LoadingState v-if="loading" message="Loading repository..." />
<ErrorState v-else-if="error" :message="error" />
<template v-else-if="repository">
<PageHeader
:title="repository.name"
icon="fas fa-code-branch"
icon-color="text-accent-500"
:breadcrumbs="breadcrumbs"
>
<template #subtitle>
<GithubLink :url="`https://github.com/${repository.owner}/${repository.name}`">
{{ repository.owner }}/{{ repository.name }}
</GithubLink>
</template>
</PageHeader>
<!-- Stats -->
<section class="py-8 px-4">
<div class="container mx-auto">
<div class="grid grid-cols-2 md:grid-cols-4 gap-4">
<StatCard
:value="repository.total_commits"
label="Commits"
icon="fas fa-code-commit"
icon-color="text-green-500"
/>
<StatCard
:value="repository.total_prs"
label="Pull Requests"
icon="fas fa-code-pull-request"
icon-color="text-blue-500"
/>
<StatCard
:value="repository.total_reviews"
label="Reviews"
icon="fas fa-eye"
icon-color="text-purple-500"
/>
<StatCard
:value="repository.active_contributors"
label="Contributors"
icon="fas fa-users"
icon-color="text-orange-500"
/>
</div>
</div>
</section>
<!-- Contributors -->
<section class="py-8 px-4">
<div class="container mx-auto">
<div class="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4 mb-6">
<SectionHeader title="Contributors" icon="fas fa-users" icon-color="text-blue-500" class="mb-0" />
<!-- Search Input -->
<div class="relative w-full sm:w-72 lg:w-96">
<i class="fas fa-search absolute left-3 top-1/2 -translate-y-1/2 text-gray-500"></i>
<input
v-model="searchQuery"
type="text"
placeholder="Search contributors..."
class="w-full pl-10 pr-4 py-2 rounded-lg border border-gray-700 bg-gray-800 text-gray-100 placeholder-gray-500 focus:outline-none focus:ring-2 focus:ring-primary-500 focus:border-transparent transition text-sm"
/>
<button
v-if="searchQuery"
@click="searchQuery = ''"
class="absolute right-3 top-1/2 -translate-y-1/2 text-gray-400 hover:text-gray-200"
>
<i class="fas fa-times"></i>
</button>
</div>
</div>
<p v-if="searchQuery && filteredContributors.length !== allContributors.length" class="mb-4 text-sm text-gray-400">
Showing {{ filteredContributors.length }} of {{ allContributors.length }} contributors
</p>
<DataTable
:columns="tableColumns"
:items="filteredContributors"
empty-icon="fas fa-users"
empty-message="No contributors found"
row-class="hover:bg-gray-800/30 transition group"
>
<template #contributor="{ item }">
<ContributorRow :contributor="item" />
</template>
<template #commits="{ item }">
<span class="text-white">{{ formatNumber(item.commit_count) }}</span>
</template>
<template #prs="{ item }">
<span class="text-white">{{ formatNumber(item.prs_opened) }}</span>
</template>
<template #reviews="{ item }">
<span class="text-white">{{ formatNumber(item.reviews_given) }}</span>
</template>
<template #lines="{ item }">
<span class="text-green-500">+{{ formatNumber(item.lines_added) }}</span>
<span class="text-gray-400 mx-1">/</span>
<span class="text-red-500">-{{ formatNumber(item.lines_deleted) }}</span>
</template>
<template #score="{ item }">
<span class="text-lg font-bold bg-gradient-to-r from-primary-400 to-accent-400 bg-clip-text text-transparent">
{{ formatNumber(item.score?.total || 0) }}
</span>
</template>
</DataTable>
</div>
</section>
</template>
</div>
</template>