记账app_7_开发前端页面
Contents
1 封装WelcomeLayout组件
Welcome页面的四个子组件代码基本一致,考虑将其整体抽离为一个布局组件
-
属性 和 插槽
-
都是父子组件传参的方式
-
插槽传参是通过子组件的形式,而属性是通过标签属性来传递的
-
slot是
setup
函数中context
参数的一个属性,在子组件定义与父组件使用插槽时,都要以函数形式声明 -
父组件在使用时,注意slots对象中的属性都是返回值为jsx的回调函数,使用**
v-slots
属性**向子组件传参
-
用法详见:vuejs/babel-plugin-jsx: JSX for Vue 3 (github.com) Slot部分
-
使用插槽
- 先定义再使用
1 2 3 4 5 6 7 8 9 10 11 12
// First.tsx import ... export const First = defineComponent({ setup(props, context) { const slots = { icon: () => <img src={pig} />, title: () => <h2>会赚钱<br />还要会省钱</h2>, buttons: () => <>...</>, }; return () => <WelcomeLayout v-slots={slots} />; }, });
- 也可以插入标签内使用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
// First.tsx import ... export const First = defineComponent({ setup(props, context) { return () => ( <WelcomeLayout> {{ icon: () => <img src={pig} />, title: () => <h2>会赚钱<br />还要会省钱</h2>, buttons: () => <>...</>, }} </WelcomeLayout> ) } });
-
定义插槽
|
|
-
优化
-
如果没用到ref,可以使用render替换setup,或者直接变成一个函数的形式(React)
用法详见:vuejs/babel-plugin-jsx: JSX for Vue 3 (github.com) with render部分
-
优化First等欢迎页子组件(这不就是React吗)
1 2 3 4 5 6 7 8 9 10 11 12 13
import ... export const Second = () => { return ( <WelcomeLayout> // 这里不是模板字符串,而是对象的含义 {{ icon: () => <img src={clock} />, title: () => <h2>每日提醒<br />不遗漏每一笔账单</h2>, buttons: () => <>...</>, }} </WelcomeLayout> ); };
- 优化WelcomeLayout.tsx,注意类型
1 2 3 4 5 6 7 8 9 10 11 12
import { FunctionalComponent } from "vue"; import s from "./WelcomeLayout.module.scss"; export const WelcomeLayout: FunctionalComponent = (props, context) => { const { slots: { icon, title, buttons } } = context; return ( ... <div class={s.card}> {icon?.()} {title?.()} </div> ); };
-
2 动画、SVG Sprites与滑动手势
2.1 路由与动画
-
需求:切换Welcome子路由时加上过渡动画
-
为了让动画只应用于中间的路由画面,需要将子组件的card与footer分离(代码略)
-
GIt技巧——将远程仓库中的某次提交作为补丁应用到本地仓库
-
git log
复制某次commit id -
git format-patch -1 <id>
生成这次commit的补丁 -
mv 0001-.patch ../.././
将补丁移入本地仓库 -
git am < 0001-.patch
本地仓库应用这次补丁
-
-
首先需要将子路由的components分离,vue-router支持单页面多重RouterView
|
|
- 在Welcome使用时指定RouterView的name属性即可
|
|
- TSX的Transition方案
- Template写法
|
|
- 针对TSX,我们可以参考slots的tsx写法
RouterView
的slots
接受一个函数,函数的参数包含渲染的路由组件与路由属性,返回值为一个新组件(注意,返回的组件需要利用vue提供的h
方法手动渲染,(好像也不需要…))。
|
|
-
注意:在RouterView的类型定义中找到其slots函数的类型
-
那么就可以在
Component
外包上过渡组件Transition
,并指定四个状态的css -
注意:如果指定了Transition的name,那么动画的css需要写到全局样式中。这里我们使用自定义动画,此时样式就可以写为CSS Modules的形式了
|
|
- 最终版
|
|
2.2 SVG Sprites
-
在渲染Welcome页面时,图片是先渲染再请求的,这就导致图片的加载存在卡顿现象
-
安装依赖
-
pnpm i svgo
用于优化 SVG 文件 -
pnpm i svgstore
用于制作 SVG Sprites
-
-
编写Vite插件
|
|
- 需要做一些配置
|
|
|
|
|
|
- 使用
|
|
2.3 封装useSwipe Hook
-
移动端的每个页面都要支持滑动事件,为此我们封装为
useSwipe hook
(也就是Vue中的Composition API) -
hook/Composition API
:界面中重复的部分连同其功能一起提取为可重用的代码段 -
useSwipe
本质上就是给传入的组件添加触控事件监听器,通过计算滑动初始位置与结束位置的差,得到移动的距离和方向,进而执行对应的响应
|
|
- 使用
|
|
2.4 滑动切换路由
-
统一类型定义写法
- 推荐
1
const a = ref<HTMLElement>()
- 不推荐
1
const a = ref<HTMLElement | null>(null)
-
为每个路由添加name,方便跳转
-
在Welcome.tsx中监听main元素的滑动事件(
useSwipe(main)
) -
扩展useSwipe,提供useSwipe的生命周期机制
|
|
- 现在就可以在滑动之前(
beforeStart
)禁用浏览器的默认滑动事件
|
|
- 在**
watchEffect
中监听**isMoving
和direction
的变化(自动监听),从而执行路由切换逻辑
|
|
-
经过测试后发现,由于滑动事件触发过于频繁,导致一瞬间就切换到了最后一页
-
因此需要添加节流处理
|
|
- 优化路由切换逻辑后的最终代码
|
|
3 业务与轮子
我的思想是对的
Author gsemir
LastMod 2022-05-26