做完按钮之后,我们应该了解了遮罩层的概念,接下来我们来做 Dialog 组件!

返回阅读列表点击 这里

需求分析

  1. 默认是不可见的,在用户触发某个动作后变为可见
  2. 自带白板卡片,分为上中下三个区域,分别放置标题、内容、操作
  3. 有两个基本操作:确定、取消
  4. 卡片后应放置淡黑色遮罩层,遮住原本网页内容
  5. 可以自定义是否允许取消
  6. 右上角提供小叉叉来允许关闭
  7. 允许通过点击遮罩层来关闭

所以,我们能够得出如下的参数表格

参数 含义 类型 可选值 默认值
visible 是否可见 boolean false / true false
title 标题 string 任意字符串 必填
ok 确定回调 ()=>boolean 返回 boolean 的函数 ()=>true
cancel 取消回调 ()=>boolean 返回 boolean 的函数 ()=>true

注意:可以通过设置返回值为 true 来允许事件发生,反之不允许。可以通过设置返回 false 来取消事件

骨架

我们复用之前做好的 Button 组件

一般情况下,我们不希望对话框弹窗在 DOM 树上的位置,而希望是 body 的直接子元素,那么我们可以使用 vue3teleport 组件。

代码如下:

<template>
<template v-if="visible">
<teleport to="body">
<div class="jeremy-dialog-overlay" @click="close"></div>
<div class="jeremy-dialog">
<header class="jeremy-dialog-header">
{{ title }}
<span class="jeremy-dialog-close" @click="close"></span>
</header>
<div class="jeremy-dialog-divider" />
<main class="jeremy-dialog-main">
<slot></slot>
</main>
<div class="jeremy-dialog-divider" />
<footer class="jeremy-dialog-footer">
<jeremy-button level="plain" @click="close">取消</jeremy-button>
&nbsp;&nbsp;&nbsp;&nbsp;
<jeremy-button @click="task" :loading="loading">确定</jeremy-button>
</footer>
</div>
</teleport>
</template>
</template>

这样,在渲染时,teleport 内部的内容就会出现在 body 的子级上。

功能

现在 ts 中声明参数:

declare const props: {
visible: boolean;
title: string;
ok: () => boolean;
cancel: () => boolean;
};
declare const context: SetupContext;

然后在 export default 中,写入我们的参数:

export default {
install: function (Vue) {
Vue.component(this.name, this);
},
name: "JeremyDialog",
props: {
visible: {
type: Boolean,
default: false,
},
title: {
type: String,
required: true,
},
ok: {
type: Function,
default: () => {
return true;
},
},
cancel: {
type: Function,
default: () => {
return true;
},
},
},
components: {
JeremyButton,
},
};

再补全 setup 方法,此处选用 Promise 制造提交等待响应的感觉

  setup(props, context) {
const loading = ref(false);
const close = () => {
if (loading.value) {
return;
}
new Promise((resolve, reject) => {
resolve(props.cancel());
}).then((result) => {
if (result !== false) {
context.emit("update:visible", false);
}
});
};
const task = () => {
new Promise((resolve, reject) => {
loading.value = true;
resolve(props.ok());
}).then((result) => {
if (result === true) {
loading.value = false;
context.emit("update:visible", false);
}
});
};
return { loading, close, task };
},

样式表

最后再补全样式表:

<style lang="scss">
.jeremy-dialog-overlay {
z-index: 20;
position: fixed;
left: 0;
top: 0;
background: fade-out($color: #000000, $amount: 0.7);
width: 100vw;
height: 100vh;
}
.jeremy-dialog {
z-index: 20;
position: fixed;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
min-width: 300px;
min-height: 200px;
border-radius: 8px;
background: white;
display: flex;
flex-direction: column;
> * {
padding: 8px;
}
> .jeremy-dialog-divider {
border: 1px solid #8c6fef;
padding: 0;
}
> .jeremy-dialog-header {
display: flex;
justify-content: space-between;
> .jeremy-dialog-close {
position: relative;
display: inline-block;
width: 16px;
height: 16px;
cursor: pointer;
&::before,
&::after {
content: "";
position: absolute;
height: 1px;
background: black;
width: 100%;
top: 50%;
left: 50%;
}
&::before {
transform: translate(-50%, -50%) rotate(-45deg);
}
&::after {
transform: translate(-50%, -50%) rotate(45deg);
}
}
}
> .jeremy-dialog-main {
flex-grow: 1;
background: white;
}
> .jeremy-dialog-footer {
display: flex;
justify-content: flex-end;
}
}
</style>

一行代码打开

多数时候我们是不希望使用组件式的,而是直接用函数生成一个弹窗。那么,我们只要使用 vue3 提供的 createApph 函数就可以做到了。

我们再创建一个 ts 文件,即 createDialog.ts ,代码如下:

import { createApp, h } from 'vue'
import JeremyDialog from './Dialog.vue'
export const createDialog = options => {
const { title, content, ok, cancel } = options
const div = document.createElement('div')
document.body.appendChild(div)
const close = () => {
app.unmount(div)
div.remove()
}
const app = createApp({
render() {
return h(JeremyDialog, {
visible: true,
'onUpdate:visible': newVisible => {
if (newVisible === false) {
close();
}
},
title,
ok, cancel
}, { default() { return content } })
}
})
app.mount(div)
}

然后再需要使用的地方导入即可:

import {createDialog} from './createDialog.ts'

注意:该函数要求传入一个 options 对象,该对象包含 title, content, ok, cancel 等 4 个部分,content 指代组件式中的插槽,其余含义见需求分析

然后使用 h 函数渲染新 app 中的内容,并作为参数传入 createApp 函数用以创建新的 app,最后挂载到 DOM 树上。

运行效果

接下来,我们将组件引入到页面中,看一下实际运行效果

感谢阅读

06 - Vue3 UI Framework - Dialog 组件的更多相关文章

  1. 05 - Vue3 UI Framework - Button 组件

    官网基本做好了,接下来开始做核心组件 返回阅读列表点击 这里 目录准备 在项目 src 目录下创建 lib 文件夹,用来存放所有的核心组件吧.然后再在 lib 文件夹下创建 Button.vue 文件 ...

  2. 08 - Vue3 UI Framework - Input 组件

    接下来再做一个常用的组件 - input 组件 返回阅读列表点击 这里 需求分析 开始之前我们先做一个简单的需求分析 input 组件有两种类型,即 input 和 textarea 类型 当类型为 ...

  3. 09 - Vue3 UI Framework - Table 组件

    接下来做个自定义的表格组件,即 table 组件 返回阅读列表点击 这里 需求分析 开始之前我们先做一个简单的需求分析 基于原生 table 标签的强语义 允许用户自定义表头.表体 可选是否具有边框 ...

  4. 10 - Vue3 UI Framework - Tabs 组件

    标签页是非常常用的组件,接下来我们来制作一个简单的 Tabs 组件 返回阅读列表点击 这里 需求分析 我们先做一个简单的需求分析 可以选择标签页排列的方向 选中的标签页应当有下划线高亮显示 切换选中时 ...

  5. 11 - Vue3 UI Framework - Card 组件

    卡片是非常常用也是非常重要的组件,特别是在移动端的众多应用场景中,随便打开一个手机 App ,您会发现充斥着各种各样的卡片. 所以,我们也来制作一个简易的 Card 组件 返回阅读列表点击 这里 需求 ...

  6. 00 - Vue3 UI Framework - 阅读辅助列表

    阅读列表 01 - Vue3 UI Framework - 开始 02 - Vue3 UI Framework - 顶部边栏 03 - Vue3 UI Framework - 首页 04 - Vue3 ...

  7. 04 - Vue3 UI Framework - 文档页

    官网的首页做完了,接下来开始做官网的文档页 返回阅读列表点击 这里 路由设计 先想想我们需要文档页通向哪些地方,这里直接给出我的设计: 所属 子标题 跳转路径 文件名(*.vue) 指南 介绍 /do ...

  8. 01 - Vue3 UI Framework - 开始

    写在前面 一年多没写过博客了,工作.生活逐渐磨平了棱角. 写代码容易,写博客难,坚持写高水平的技术博客更难. 技术控决定慢慢拾起这份坚持,用作技术学习的阶段性总结. 返回阅读列表点击 这里 开始 大前 ...

  9. 12 - Vue3 UI Framework - 打包发布

    基础组件库先做到这个阶段,后面我会继续新增.完善 接下来,我们对之前做的组件进行打包发布到 npm 返回阅读列表点击 这里 组件库优化 通用层叠样式表 我想大家都注意到了,前面我们在写组件的时候,sc ...

随机推荐

  1. OPA-Gatekeeper实验:对特定用户的更新时间窗口做限制

    实验目的 OPA-Gatekeeper可以在Kubernetes 中,通过策略来实现一些额外的管理.安全方面的限制,例如:限制特定用户在 Namespace 中的行为权限 本次实验将在test命名空间 ...

  2. Dapr-Actor构建块

    前言: 前篇-绑定 文章对Dapr的绑定构建块进行了解,本篇继续对 Actor 构建块进行了解学习. 一.Actor简介: Actors 为最低级别的"计算单元". 换句话说,您将 ...

  3. 第十四章 kubernetes 核心技术-调度器

    一.概述 一个容器平台的主要功能就是为容器分配运行时所需要的计算,存储和网络资源.容器调 度系统负责选择在最合适的主机上启动容器,并且将它们关联起来.它必须能够自动的处 理容器故障并且能够在更多的主机 ...

  4. python中else的三种用法

    与if搭配 要么--不然-- num = input("输入一个数字") if(num % 2 == 0): print("偶数") else: print(& ...

  5. linux命令-压缩数据

    linux文件压缩工具:bzip2 文件扩展名 .bz2 compress 文件扩展名 .Z linux上很少看到了 uncompress解压 gzip 文件扩展名,.gz,gzip压缩文件,gzca ...

  6. pm2 重启策略(restart strategies)

    使用 PM2 启动应用程序 时,应用程序会在自动退出.事件循环为空 (node.js) 或应用程序崩溃时自动重新启动. 但您也可以配置额外的重启策略,例如: 使用定时任务重新启动应用程序 文件更改后重 ...

  7. Codeforces 891E - Lust(生成函数)

    Codeforces 题面传送门 & 洛谷题面传送门 NaCly_Fish:<简单>的生成函数题 然鹅我连第一步都没 observe 出来 首先注意到如果我们按题意模拟那肯定是不方 ...

  8. 洛谷 P3783 - [SDOI2017]天才黑客(前后缀优化建图)

    题面传送门 神仙题一道. 首先注意到这里的贡献涉及到边的顺序,并且只与相邻的边是什么有关,因此不难想到一个做法--边转点,点转边,具体来说对于每条边 \(e\),我们将其拆成两个点 \(in_e,ou ...

  9. Git常用操作(二)

    仓库拉取 git clone XXX 修改仓库链接 $ git config -l # 显示coding列表 $ git config --get remote.origin.url # 返回orig ...

  10. 【Perl】如何安装Bioperl模块?

    目录 失败尝试一:使用cpanm 失败尝试二:使用CPAN 成功尝试:直接conda安装bioperl 没有尝试:源码安装bioperl 生信软件绕不过Perl,Perl绕不过Bioperl.而Bio ...