mirror of
https://github.com/pengzhanbo/vuepress-theme-plume.git
synced 2026-04-23 10:58:13 +08:00
feat(theme): add component <RepoCard />
This commit is contained in:
parent
5e9d2d043a
commit
4c44462e61
155
theme/src/client/features/components/RepoCard.vue
Normal file
155
theme/src/client/features/components/RepoCard.vue
Normal file
@ -0,0 +1,155 @@
|
||||
<script setup lang="ts">
|
||||
import { toRef } from 'vue'
|
||||
import { useGithubRepo } from '../composables/github-repo.js'
|
||||
|
||||
const props = defineProps<{
|
||||
repo: string
|
||||
}>()
|
||||
const { loaded, data } = useGithubRepo(toRef(props, 'repo'))
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div v-if="loaded && data" class="vp-repo-card">
|
||||
<p class="repo-name">
|
||||
<span class="vpi-github-repo" />
|
||||
<span class="repo-link">
|
||||
<a :href="data.url" target="_blank" rel="noopener noreferrer">
|
||||
{{ data.ownerType === 'Organization' ? data.fullName : data.name }}
|
||||
</a>
|
||||
</span>
|
||||
<span class="repo-visibility">{{ data.visibility + (data.template ? ' Template' : '') }} </span>
|
||||
</p>
|
||||
<p class="repo-desc">
|
||||
{{ data.description }}
|
||||
</p>
|
||||
<div class="repo-info">
|
||||
<p v-if="data.language">
|
||||
<span
|
||||
class="repo-language" :style="{ 'background-color': data.languageColor }"
|
||||
/><span>{{ data.language }}</span>
|
||||
</p>
|
||||
<p><span class="vpi-github-star" /><span>{{ data.stars }}</span></p>
|
||||
<p><span class="vpi-github-fork" /><span>{{ data.forks }}</span></p>
|
||||
<p v-if="data.license">
|
||||
<span class="vpi-github-license" /><span>{{ data.license.name }}</span>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.vp-repo-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
padding: 16px 20px;
|
||||
margin: 16px 0;
|
||||
border: solid 1px var(--vp-c-divider);
|
||||
border-radius: 8px;
|
||||
transition: border-color var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card:hover {
|
||||
border-color: var(--vp-c-brand-2);
|
||||
}
|
||||
|
||||
.vp-repo-card p {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-name {
|
||||
display: flex;
|
||||
gap: 0 8px;
|
||||
align-items: center;
|
||||
max-width: 100%;
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-link {
|
||||
flex: 1;
|
||||
width: 1px;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-name a {
|
||||
max-width: 100%;
|
||||
font-weight: 600;
|
||||
color: var(--vp-c-brand-1);
|
||||
text-decoration: none;
|
||||
transition: color var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-name a:hover {
|
||||
color: var(--vp-c-brand-2);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-visibility {
|
||||
display: inline-block;
|
||||
padding: 0 8px;
|
||||
font-size: 14px;
|
||||
line-height: 20px;
|
||||
color: var(--vp-c-text-2);
|
||||
border: solid 1px var(--vp-c-divider);
|
||||
border-radius: 22px;
|
||||
transition: color var(--t-color), border var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-desc {
|
||||
flex: 1;
|
||||
font-size: 14px;
|
||||
line-height: 22px;
|
||||
color: var(--vp-c-text-2);
|
||||
transition: color var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-info {
|
||||
display: flex;
|
||||
gap: 16px;
|
||||
align-items: center;
|
||||
justify-content: flex-start;
|
||||
font-size: 14px;
|
||||
line-height: 22px;
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-info p {
|
||||
display: flex;
|
||||
gap: 0 4px;
|
||||
align-items: center;
|
||||
color: var(--vp-c-text-2);
|
||||
transition: color var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-info p [class^="vpi-github-"] {
|
||||
color: var(--vp-c-text-1);
|
||||
transition: color var(--t-color);
|
||||
}
|
||||
|
||||
.vp-repo-card .repo-language {
|
||||
display: inline-block;
|
||||
width: 0.8em;
|
||||
height: 0.8em;
|
||||
border-radius: 100%;
|
||||
}
|
||||
|
||||
.vpi-github-repo {
|
||||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='1em' height='1em' viewBox='0 0 16 16'%3E%3Cpath fill='%23000' d='M2 2.5A2.5 2.5 0 0 1 4.5 0h8.75a.75.75 0 0 1 .75.75v12.5a.75.75 0 0 1-.75.75h-2.5a.75.75 0 0 1 0-1.5h1.75v-2h-8a1 1 0 0 0-.714 1.7a.75.75 0 1 1-1.072 1.05A2.5 2.5 0 0 1 2 11.5Zm10.5-1h-8a1 1 0 0 0-1 1v6.708A2.5 2.5 0 0 1 4.5 9h8ZM5 12.25a.25.25 0 0 1 .25-.25h3.5a.25.25 0 0 1 .25.25v3.25a.25.25 0 0 1-.4.2l-1.45-1.087a.25.25 0 0 0-.3 0L5.4 15.7a.25.25 0 0 1-.4-.2Z'/%3E%3C/svg%3E");
|
||||
|
||||
color: var(--vp-c-text-2);
|
||||
transition: color var(--t-color);
|
||||
transform: translateY(2px);
|
||||
}
|
||||
|
||||
.vpi-github-star {
|
||||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='1em' height='1em' viewBox='0 0 256 256'%3E%3Cpath fill='%23000' d='M243 96a20.33 20.33 0 0 0-17.74-14l-56.59-4.57l-21.84-52.81a20.36 20.36 0 0 0-37.66 0L87.35 77.44L30.76 82a20.45 20.45 0 0 0-11.66 35.88l43.18 37.24l-13.2 55.7A20.37 20.37 0 0 0 79.57 233L128 203.19L176.43 233a20.39 20.39 0 0 0 30.49-22.15l-13.2-55.7l43.18-37.24A20.43 20.43 0 0 0 243 96m-70.47 45.7a12 12 0 0 0-3.84 11.86L181.58 208l-47.29-29.08a12 12 0 0 0-12.58 0L74.42 208l12.89-54.4a12 12 0 0 0-3.84-11.86l-42.27-36.5l55.4-4.47a12 12 0 0 0 10.13-7.38L128 41.89l21.27 51.5a12 12 0 0 0 10.13 7.38l55.4 4.47Z'/%3E%3C/svg%3E");
|
||||
}
|
||||
|
||||
.vpi-github-fork {
|
||||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='1em' height='1em' viewBox='0 0 256 256'%3E%3Cpath fill='%23000' d='M228 64a36 36 0 1 0-48 33.94V112a4 4 0 0 1-4 4H80a4 4 0 0 1-4-4V97.94a36 36 0 1 0-24 0V112a28 28 0 0 0 28 28h36v18.06a36 36 0 1 0 24 0V140h36a28 28 0 0 0 28-28V97.94A36.07 36.07 0 0 0 228 64M64 52a12 12 0 1 1-12 12a12 12 0 0 1 12-12m64 152a12 12 0 1 1 12-12a12 12 0 0 1-12 12m64-128a12 12 0 1 1 12-12a12 12 0 0 1-12 12'/%3E%3C/svg%3E");
|
||||
}
|
||||
|
||||
.vpi-github-license {
|
||||
--icon: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='1em' height='1em' viewBox='0 0 16 16'%3E%3Cpath fill='none' stroke='%23000' stroke-linecap='round' stroke-linejoin='round' d='M4.5 13.5h7M8.01 1v12.06M1.5 3.5h3l1.5-1h4l1.5 1h3M.5 10L3 4.48L5.5 10C4 11 2 11 .5 10m10 0L13 4.48L15.5 10c-1.5 1-3.5 1-5 0'/%3E%3C/svg%3E");
|
||||
}
|
||||
</style>
|
||||
48
theme/src/client/features/composables/github-repo.ts
Normal file
48
theme/src/client/features/composables/github-repo.ts
Normal file
@ -0,0 +1,48 @@
|
||||
import { computed, ref, toValue, watch } from 'vue'
|
||||
import type { MaybeRef } from 'vue'
|
||||
|
||||
export interface GithubRepoInfo {
|
||||
name: string
|
||||
fullName: string
|
||||
description: string
|
||||
url: string
|
||||
stars: number
|
||||
forks: number
|
||||
watchers: number
|
||||
language: string
|
||||
languageColor: string
|
||||
visibility: 'Private' | 'Public' // private, public
|
||||
template: boolean
|
||||
ownerType: 'User' | 'Organization'
|
||||
license: {
|
||||
name: string
|
||||
url: string
|
||||
} | null
|
||||
}
|
||||
|
||||
export function useGithubRepo(repo: MaybeRef<string>) {
|
||||
const repoRef = computed(() => {
|
||||
const info = toValue(repo)
|
||||
const [owner = '', name = ''] = info.split('/')
|
||||
return { owner, name }
|
||||
})
|
||||
const data = ref<GithubRepoInfo | null>(null)
|
||||
const loaded = ref(false)
|
||||
|
||||
async function fetchData() {
|
||||
const { owner, name } = repoRef.value
|
||||
if (__VUEPRESS_SSR__ || !owner || !name)
|
||||
return
|
||||
|
||||
loaded.value = false
|
||||
const res = await fetch(`https://api.pengzhanbo.cn/github/repo/${owner}/${name}`)
|
||||
.then(res => res.json()) as GithubRepoInfo
|
||||
|
||||
data.value = res
|
||||
loaded.value = true
|
||||
}
|
||||
|
||||
watch(repoRef, fetchData, { immediate: true })
|
||||
|
||||
return { data, loaded }
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user