mirror of
https://github.com/Sevichecc/Urara-Blog.git
synced 2025-05-01 07:09:30 +08:00
300 lines
10 KiB
Markdown
300 lines
10 KiB
Markdown
---
|
||
title: 为博客写一个Project showcase 页面
|
||
created: 2022-05-26
|
||
summary: 第一次Pull Request的经历
|
||
tags:
|
||
- Svelte
|
||
- Open Source
|
||
---
|
||
|
||
这两天为博客写了一个 Project 的页面用来放我的作品,这里记录一下我是怎么写(模仿)的,我对 Svelte 语法的了解不多,没有特别深入学习,只是在官方看了下文档和用了下他们的 [交互式教程](https://www.sveltejs.cn/tutorial/basics) ,编码的过程是一边学习一边模仿慢慢摸索的,虽然最终没有 merge 到 repo 中,但我觉得整个过程都蛮兴奋的。
|
||
|
||
既然有了博客,那我肯定是要写一下这个过程的。
|
||
|
||
## 1. 分析需求
|
||
|
||
我想要的是一个独立的 Page,而不是一个 Post 页面,最后把它放在导航栏里面。
|
||
想要有以下这几个功能:
|
||
|
||
- 技术栈分类
|
||
- 项目类别筛选
|
||
- 项目展示
|
||
|
||
主要有这些信息的展示:
|
||
|
||
- 项目标题
|
||
- 项目图片
|
||
- 项目描述
|
||
- 技术栈
|
||
- 项目类别
|
||
|
||
## 2. 画原型图
|
||
|
||
明确了需求后,参考了几个项目平台的布局,在 [Whimsical](https://whimsical.com/) 上画了原型图如下:
|
||

|
||
|
||
目前还没有做上面 Tag 的分类功能,之后可能会做吧
|
||
|
||
## 2. 创建组件样式 CSS
|
||
|
||
为了统一风格,我在博客现有框架里四处搜寻可用的组件样式,想在这基础上修改,然后我找到了作者 藍 在 Tailwind Play 上的友链组件,感觉很适合,然后就直接在这个 Tailwind Play Demo 上进行了样式修改,不过此时填写的数据都是死数据,后面再进行修改。
|
||
|
||
因为我之前没有怎么用过 Tailwind,所以是一边对照 Tailwind 文档修改的,然后 Tailwind Play 上的代码提示功能真的很新手友好,hover CSS class 的时候会显示具体的 CSS 原始参数,很直观。
|
||
|
||

|
||
最后我构建的 Demo 样式如下:
|
||
[Tailwind Play](https://play.tailwindcss.com/uQwYojgpuk?layout=horizontal)
|
||
|
||

|
||
|
||
## 4. 编写组件代码
|
||
|
||
整个页面的构建跟 Friend 页面很像,我分析了 Friend 页面所涉及到的代码和结构,然后一点点模仿构建 Project 页面。
|
||
|
||
### 数据
|
||
|
||
首先根据需求确定传入的数据及其格式,以便后面使用 TypeScript 的提示
|
||
|
||
- 参考:`/src/lib/config/friends.ts`
|
||
- 新建:`/src/lib/config/projects.ts`
|
||
|
||
```ts twoslash title="/src/lib/config/friends.ts"
|
||
export interface FriendOld {
|
||
// hCard+XFN
|
||
id: string // HTML id
|
||
rel?: string // XFN, contact / acquaintance / friend
|
||
link?: string // URL
|
||
html?: string // HTML
|
||
title?: string // 标题
|
||
descr?: string // 描述
|
||
avatar?: string // 头像
|
||
name?: string // backwards compatibility
|
||
}
|
||
|
||
export type Friend = {
|
||
id: string // HTML id
|
||
rel?: string // XHTML Friends Network
|
||
link?: string // URL
|
||
html?: string // Custom HTML
|
||
|
||
title?: string // 标题
|
||
name?: string // 人名
|
||
avatar?: string // 头像
|
||
descr?: string // 描述
|
||
class?: {
|
||
avatar?: string // 头像类名
|
||
img?: string // 图片类名
|
||
}
|
||
}
|
||
|
||
export const friends: Friend[] = [
|
||
{
|
||
id: 'id',
|
||
rel: '',
|
||
title: '',
|
||
name: '',
|
||
link: '',
|
||
descr: '',
|
||
avatar: ''
|
||
}
|
||
]
|
||
```
|
||
|
||
```ts twoslash title="/src/lib/config/projects.ts"
|
||
export type Project = {
|
||
id: string
|
||
name?: string
|
||
tags?: string[]
|
||
feature?: string
|
||
description?: string
|
||
img?: string
|
||
link?: string
|
||
}
|
||
|
||
export const projects: Project[] = [
|
||
{
|
||
id: 'coach',
|
||
name: 'Find a Coach',
|
||
tags: ['Vue 3', 'Composition API'],
|
||
feature: 'Vue3',
|
||
description:
|
||
'既然如何, 问题的关键究竟为何? 要想清楚,科学和人文谁更有意义,到底是一种怎么样的存在。 普列姆昌德曾经提到过,希望的灯一旦熄灭,生活刹那间变成了一片黑暗。这启发了我, 那么, 我认为, 总结的来说,',
|
||
img: 'https://uneorange.oss-cn-guangzhou.aliyuncs.com/202205251801454.avif',
|
||
link: 'https://seviche.cc'
|
||
}
|
||
]
|
||
```
|
||
|
||
### 组件
|
||
|
||
将 CSS 复制进去,并注入数据
|
||
|
||
- 参考:`/src/lib/components/extra/friend.svelte`
|
||
- 新建:`/src/lib/components/extra/project.svelte`
|
||
|
||
```html title="/src/lib/components/extra/friend.svelte"
|
||
<script lang="ts">
|
||
import type { Friend } from '$lib/config/friends'
|
||
import Footer from '$lib/components/footer.svelte'
|
||
export let item: unknown
|
||
let friend = item as unknown as Friend
|
||
</script>
|
||
|
||
{#if friend.id === 'footer'}
|
||
<footer rounded="{true}" class="p-4 md:p-8" />
|
||
{:else if friend.html}
|
||
<a id="{friend.id}" rel="{friend.rel}" href="{friend.link}" class="h-card u-url">{@html friend.html}</a>
|
||
{:else}
|
||
<a
|
||
id="{friend.id}"
|
||
rel="{friend.rel}"
|
||
href="{friend.link}"
|
||
class="card bg-base-100 shadow-xl hover:shadow-2xl transition-shadow h-card u-url">
|
||
<div class="absolute text-4xl font-bold opacity-5 rotate-6 leading-tight top-4">{friend.name ?? ''}{friend.title ?? ''}</div>
|
||
<div class="card-body p-4">
|
||
<div class="flex items-center gap-4">
|
||
{#if friend.avatar}
|
||
<div class="avatar {friend.class?.avatar} shrink-0 w-16 mb-auto">
|
||
<img class="{friend.class?.img ?? 'rounded-xl'} u-photo" src="{friend.avatar}" alt="{friend.title}" />
|
||
</div>
|
||
{:else}
|
||
<div class="avatar {friend.class?.avatar} placeholder mb-auto">
|
||
<div class="{friend.class?.img ?? 'bg-neutral-focus text-neutral-content shadow-inner rounded-xl'} w-16">
|
||
<span class="text-3xl">{(friend.name ?? friend.title).charAt(0)}</span>
|
||
</div>
|
||
</div>
|
||
{/if}
|
||
<div class="card-title flex-col gap-0 flex-1 items-end">
|
||
<span class="text-right p-name">{friend.name ?? ''}</span>
|
||
<span class="opacity-50 text-right">{friend.title}</span>
|
||
</div>
|
||
</div>
|
||
{#if friend.descr}
|
||
<div class="prose opacity-70 p-note">{friend.descr}</div>
|
||
{/if}
|
||
</div>
|
||
</a>
|
||
{/if}
|
||
```
|
||
|
||
根据具体的页面效果修改了 Demo 中的 CSS 样式
|
||
|
||
```html title="/src/lib/components/extra/project.svelte"
|
||
<script lang="ts">
|
||
import type { Project } from '$lib/config/projects'
|
||
import Footer from '$lib/components/footer.svelte'
|
||
export let item: unknown
|
||
let project = item as unknown as Project
|
||
let tags = project.tags
|
||
</script>
|
||
|
||
{#if project.id === 'footer'}
|
||
<footer rounded="{true}" class="max-w-4xl mx-auto p-4 md:p-8" />
|
||
{:else}
|
||
<a
|
||
id="{project.id}"
|
||
href="{project.link}"
|
||
class="card mx-auto max-w-4xl bg-base-100 shadow-xl transition-shadow mb-7 h-card u-url hover:shadow-2xl">
|
||
<div class="absolute text-5xl font-bold opacity-5 rotate-6 leading-tight top-2 right-0">{project.feature}</div>
|
||
<div class="card-body p-4">
|
||
<div class="flex flex-col md:flex-row items-start gap-4">
|
||
<div class="mb-auto max-w-full shrink-0 md:max-w-xs">
|
||
<img class="rounded-md" src="{project.img}" alt="{project.description}" />
|
||
</div>
|
||
<div class="card-title flex-1 flex-col items-start gap-4">
|
||
<div>
|
||
<h2 class="p-name text-left text-2xl mb-2">{project.name}</h2>
|
||
<div class="mb-3 text-base font-normal">
|
||
{#each tags as tag}
|
||
<span class="btn btn-sm btn-ghost normal-case border-dotted border-base-content/20 border-2 my-1 mr-1">{tag}</span>
|
||
{/each}
|
||
</div>
|
||
</div>
|
||
<p class="text-left text-base font-normal opacity-70">{@html project.description}</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</a>
|
||
{/if}
|
||
```
|
||
|
||
### 页面
|
||
|
||
- 参考:`/urara/friends/index.svelte`
|
||
- 新建:`/urara/projects/index.svelte`
|
||
|
||
```html title="/urara/friends/index.svelte"
|
||
<script lang="ts">
|
||
// @ts-nocheck
|
||
import Masonry from 'svelte-bricks'
|
||
import { Friend, friends as allFriends } from '$lib/config/friends'
|
||
import Head from '$lib/components/head.svelte'
|
||
import FriendComponent from '$lib/components/extra/friend.svelte'
|
||
|
||
const rnd = Math.random()
|
||
const fy = (a: Friend[], r = 0, c = a.length) => {
|
||
while (c) (r = (rnd * c--) | 0), ([a[c], a[r]] = [a[r], a[c]])
|
||
return a
|
||
}
|
||
let items: { id: string }[] = [...fy(allFriends as { id: string }[]), { id: 'footer' }]
|
||
let width: number, height: number
|
||
</script>
|
||
|
||
<head />
|
||
|
||
<Masonry
|
||
{items}
|
||
minColWidth="{384}"
|
||
maxColWidth="{384}"
|
||
gap="{32}"
|
||
let:item
|
||
class="mx-4 sm:mx-8 md:my-4 lg:mx-16 lg:my-8 xl:mx-32 xl:my-16"
|
||
bind:width
|
||
bind:height>
|
||
<FriendComponent {item} />
|
||
</Masonry>
|
||
```
|
||
|
||
**Projects 页面**
|
||
|
||
因为我没有用到瀑布流布局,所以删掉了一些组件和 function
|
||
|
||
```html title="/urara/projects/index.svelte"
|
||
<script lang="ts">
|
||
import { projects as allProjects } from '$lib/config/projects'
|
||
import Head from '$lib/components/head.svelte'
|
||
import ProjectComponent from '$lib/components/extra/projects.svelte'
|
||
|
||
let items: { id: string }[] = [...(allProjects as { id: string }[]), { id: 'footer' }]
|
||
</script>
|
||
|
||
<head />
|
||
|
||
{#each items as item}
|
||
<ProjectComponent {item} />
|
||
{/each}
|
||
```
|
||
|
||
### 响应式布局
|
||
|
||
参考 [Tailwind 的响应式设计指南](https://tailwindcss.com/docs/responsive-design) ,修改了卡片`flex` 的方向,以及图片的宽度,以适应小尺寸屏幕。
|
||
|
||
## 5. 测试
|
||
|
||
其实有错误的话 `pnpm dev` 以及 `pnpm build` 的时候都会提醒,但我后面发现也可以用 `pnpm check` 来检查。过程中我好像没有遇到什么 Bug。
|
||
|
||
## 6. Pull request 到 Github
|
||
|
||
先看了一下 Repo 作者写的 [contributing docs](https://github.com/importantimport/urara/blob/main/.github/CONTRIBUTING.md),了解 Contribute 的规范。
|
||
|
||
按照相应步骤做了之后,Google 了一下如何 pull request,照着 FreeCodeCamp 的这篇进行了操作:
|
||
[如何在 GitHub 提交第一个 pull request](https://chinese.freecodecamp.org/news/how-to-make-your-first-pull-request-on-github/) ,然后成功 Pull 了一个 Request,但后面发现有的文件没有改,造成了 bug,就删除了原 Request 重新 Pull。
|
||
|
||
最后终于创建成功了我的第一个 Pull request!
|
||
链接:[feat: ✨ add project page by Sevichecc · Pull Request #19 · importantimport/urara · GitHub](https://github.com/importantimport/urara/pull/19)
|
||
|
||
## 7. Last but not least
|
||
|
||
写一篇这样的博文,并发表到互联网。
|
||
|
||
||好啦我知道这篇文章有点臭屁,但下次还敢……||
|