Vue3抽屉(Drawer)

效果如下图:在线预览

在这里插入图片描述

APIs

参数说明类型默认值必传
width宽度,在 placementrightleft 时使用string | number378false
height高度,在 placementtopbottom 时使用string | number378false
title标题string | slotundefinedfalse
closable是否显示左上角的关闭按钮booleantruefalse
placement抽屉的方向‘top’ | ‘right’ | ‘bottom’ | ‘left’‘right’false
headerStyle设置 Drawer 头部的样式CSSProperties{}false
bodyStyle设置 Drawer 内容部分的样式CSSProperties{}false
extra抽屉右上角的操作区域string | slotundefinedfalse
footer抽屉的页脚string | slotundefinedfalse
footerStyle抽屉页脚的样式CSSProperties{}false
destroyOnClose关闭时是否销毁 Drawer 里的子元素booleanfalsefalse
zIndex设置 Drawerz-indexnumber1000false
open v-model抽屉是否可见booleanfalsefalse

Events

事件名称说明参数
close点击遮罩层或左上角叉或取消按钮的回调(e: Event) => void

创建抽屉组件Drawer.vue

<script setup lang="ts">
import { computed, useSlots, type CSSProperties } from 'vue'
interface Props {width?: string | number // 宽度,在 placement 为 right 或 left 时使用height?: string | number // 高度,在 placement 为 top 或 bottom 时使用title?: string // 标题 string | slotclosable?: boolean // 是否显示左上角的关闭按钮placement?: 'top' | 'right' | 'bottom' | 'left' // 抽屉的方向headerStyle?: CSSProperties // 设置 Drawer 头部的样式bodyStyle?: CSSProperties // 设置 Drawer 内容部分的样式extra?: string // 抽屉右上角的操作区域 string | slotfooter?: string // 抽屉的页脚 string | slotfooterStyle?: CSSProperties // 抽屉页脚的样式destroyOnClose?: boolean // 关闭时是否销毁 Drawer 里的子元素zIndex?: number // 设置 Drawer 的 z-indexopen?: boolean // (v-model) 抽屉是否可见
}
const props = withDefaults(defineProps<Props>(), {width: 378,height: 378,title: undefined,closable: true,placement: 'right',headerStyle: () => ({}),bodyStyle: () => ({}),extra: undefined,footer: undefined,footerStyle: () => ({}),destroyOnClose: false,zIndex: 1000,open: false
})
const drawerWidth = computed(() => {if (typeof props.width === 'number') {return props.width + 'px'}return props.width
})
const drawerHeight = computed(() => {if (typeof props.height === 'number') {return props.height + 'px'}return props.height
})
const slots = useSlots()
const showHeader = computed(() => {const titleSlots = slots.title?.()const extraSlots = slots.extra?.()let n = 0if (titleSlots && titleSlots.length) {n++}if (extraSlots && extraSlots.length) {n++}return Boolean(n) || props.title || props.extra || props.closable
})
const showFooter = computed(() => {const footerSlots = slots.footer?.()return (footerSlots && footerSlots.length) || props.footer
})
const emits = defineEmits(['update:open', 'close'])
function onBlur(e: Event) {emits('update:open', false)emits('close', e)
}
function onClose(e: Event) {emits('update:open', false)emits('close', e)
}
</script>
<template><div class="m-drawer" tabindex="-1"><Transition name="fade"><div v-show="open" class="m-drawer-mask" @click.self="onBlur"></div></Transition><Transition :name="`motion-${placement}`"><divv-show="open"class="m-drawer-wrapper":class="`drawer-${placement}`":style="`z-index: ${zIndex}; ${['top', 'bottom'].includes(placement) ? 'height:' + drawerHeight : 'width:' + drawerWidth};`"><div class="m-drawer-content"><div class="m-drawer-body-wrapper" v-if="!destroyOnClose"><div class="m-drawer-header" :style="headerStyle" v-show="showHeader"><div class="m-header-title"><svgv-if="closable"focusable="false"@click="onClose"class="u-close"data-icon="close"width="1em"height="1em"fill="currentColor"aria-hidden="true"viewBox="64 64 896 896"><pathd="M563.8 512l262.5-312.9c4.4-5.2.7-13.1-6.1-13.1h-79.8c-4.7 0-9.2 2.1-12.3 5.7L511.6 449.8 295.1 191.7c-3-3.6-7.5-5.7-12.3-5.7H203c-6.8 0-10.5 7.9-6.1 13.1L459.4 512 196.9 824.9A7.95 7.95 0 00203 838h79.8c4.7 0 9.2-2.1 12.3-5.7l216.5-258.1 216.5 258.1c3 3.6 7.5 5.7 12.3 5.7h79.8c6.8 0 10.5-7.9 6.1-13.1L563.8 512z"></path></svg><p class="u-title"><slot name="title">{{ title }}</slot></p></div><div class="m-drawer-extra"><slot name="extra">{{ extra }}</slot></div></div><div class="m-drawer-body" :style="bodyStyle"><slot></slot></div><div class="m-drawer-footer" :style="footerStyle" v-show="showFooter"><slot name="footer">{{ footer }}</slot></div></div><div class="m-drawer-body-wrapper" v-if="destroyOnClose && open"><div class="m-drawer-header" :style="headerStyle" v-show="showHeader"><div class="m-header-title"><svgfocusable="false"@click="onClose"class="u-close"data-icon="close"width="1em"height="1em"fill="currentColor"aria-hidden="true"viewBox="64 64 896 896"><pathd="M563.8 512l262.5-312.9c4.4-5.2.7-13.1-6.1-13.1h-79.8c-4.7 0-9.2 2.1-12.3 5.7L511.6 449.8 295.1 191.7c-3-3.6-7.5-5.7-12.3-5.7H203c-6.8 0-10.5 7.9-6.1 13.1L459.4 512 196.9 824.9A7.95 7.95 0 00203 838h79.8c4.7 0 9.2-2.1 12.3-5.7l216.5-258.1 216.5 258.1c3 3.6 7.5 5.7 12.3 5.7h79.8c6.8 0 10.5-7.9 6.1-13.1L563.8 512z"></path></svg><p class="u-title"><slot name="title">{{ title }}</slot></p></div><div class="m-drawer-extra"><slot name="extra">{{ extra }}</slot></div></div><div class="m-drawer-body" :style="bodyStyle"><slot></slot></div><div class="m-drawer-footer" :style="footerStyle" v-show="showFooter"><slot name="footer">{{ footer }}</slot></div></div></div></div></Transition></div>
</template>
<style lang="less" scoped>
.fade-enter-active,
.fade-leave-active {transition: opacity 0.3s;
}
.fade-enter-from,
.fade-leave-to {opacity: 0;
}
.motion-top-enter-active,
.motion-top-leave-active {transition: all 0.3s;
}
.motion-top-enter-from,
.motion-top-leave-to {transform: translateY(-100%);
}
.motion-right-enter-active,
.motion-right-leave-active {transition: all 0.3s;
}
.motion-right-enter-from,
.motion-right-leave-to {transform: translateX(100%);
}
.motion-bottom-enter-active,
.motion-bottom-leave-active {transition: all 0.3s;
}
.motion-bottom-enter-from,
.motion-bottom-leave-to {transform: translateY(100%);
}
.motion-left-enter-active,
.motion-left-leave-active {transition: all 0.3s;
}
.motion-left-enter-from,
.motion-left-leave-to {transform: translateX(-100%);
}
.m-drawer {position: fixed;inset: 0;z-index: 1000;pointer-events: none;.m-drawer-mask {position: absolute;inset: 0;z-index: 1000;background: rgba(0, 0, 0, 0.45);pointer-events: auto;}.m-drawer-wrapper {position: absolute;transition: all 0.3s;.m-drawer-content {width: 100%;height: 100%;overflow: auto;background: #ffffff;pointer-events: auto;.m-drawer-body-wrapper {display: flex;flex-direction: column;width: 100%;height: 100%;.m-drawer-header {display: flex;flex: 0;align-items: center;padding: 16px 24px;font-size: 16px;line-height: 1.5;border-bottom: 1px solid rgba(5, 5, 5, 0.06);.m-header-title {display: flex;flex: 1;align-items: center;min-width: 0;min-height: 0;.u-close {display: inline-block;margin-inline-end: 12px;width: 16px;height: 16px;fill: rgba(0, 0, 0, 0.45);cursor: pointer;transition: fill 0.2s;&:hover {fill: rgba(0, 0, 0, 0.88);}}.u-title {flex: 1;margin: 0;color: rgba(0, 0, 0, 0.88);font-weight: 600;font-size: 16px;line-height: 1.5;}}.m-drawer-extra {flex: none;color: rgba(0, 0, 0, 0.88);}}.m-drawer-body {flex: 1;min-width: 0;min-height: 0;padding: 24px;overflow: auto;}.m-drawer-footer {flex-shrink: 0;padding: 8px 16px;border-top: 1px solid rgba(5, 5, 5, 0.06);color: rgba(0, 0, 0, 0.88);}}}}.drawer-top {top: 0;inset-inline: 0;box-shadow:0 6px 16px 0 rgba(0, 0, 0, 0.08),0 3px 6px -4px rgba(0, 0, 0, 0.12),0 9px 28px 8px rgba(0, 0, 0, 0.05);}.drawer-right {top: 0;right: 0;bottom: 0;box-shadow:-6px 0 16px 0 rgba(0, 0, 0, 0.08),-3px 0 6px -4px rgba(0, 0, 0, 0.12),-9px 0 28px 8px rgba(0, 0, 0, 0.05);}.drawer-bottom {bottom: 0;inset-inline: 0;box-shadow:0 -6px 16px 0 rgba(0, 0, 0, 0.08),0 -3px 6px -4px rgba(0, 0, 0, 0.12),0 -9px 28px 8px rgba(0, 0, 0, 0.05);}.drawer-left {top: 0;bottom: 0;left: 0;box-shadow:6px 0 16px 0 rgba(0, 0, 0, 0.08),3px 0 6px -4px rgba(0, 0, 0, 0.12),9px 0 28px 8px rgba(0, 0, 0, 0.05);}
}
</style>

在要使用的页面引入

<script setup lang="ts">
import Drawer from './Drawer.vue'
import { ref } from 'vue'const open1 = ref<boolean>(false)
const open2 = ref<boolean>(false)
const open3 = ref<boolean>(false)
const open4 = ref<boolean>(false)
const open5 = ref<boolean>(false)
const options = ref([{label: 'top',value: 'top'},{label: 'right',value: 'right'},{label: 'bottom',value: 'bottom'},{label: 'left',value: 'left'}
])
const placement = ref('right')
const extraPlacement = ref('right')
const footerPlacement = ref('right')
function onClose() {// 点击遮罩层或左上角叉或取消按钮的回调open3.value = falseopen4.value = falseconsole.log('close')
}
</script>
<template><div><h1>{{ $route.name }} {{ $route.meta.title }}</h1><h2 class="mt30 mb10">基本使用</h2><Button type="primary" @click="open1 = true">Open</Button><Drawer v-model:open="open1" title="Basic Drawer" @close="onClose"><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p></Drawer><h2 class="mt30 mb10">自定义位置</h2><Radio v-model:value="placement" :options="options" style="margin-right: 8px" /><Button type="primary" @click="open2 = true">Open</Button><Drawerv-model:open="open2"title="Basic Drawer":closable="false"extra="extra"footer="footer":placement="placement"><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p></Drawer><h2 class="mt30 mb10">额外操作</h2><Radio v-model:value="extraPlacement" :options="options" style="margin-right: 8px" /><Button type="primary" @click="open3 = true">Open</Button><Drawer v-model:open="open3" title="Basic Drawer" :placement="extraPlacement"><template #extra><Button style="margin-right: 8px" @click="onClose">Cancel</Button><Button type="primary" @click="onClose">Submit</Button></template><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p></Drawer><h2 class="mt30 mb10">抽屉页脚</h2><Radio v-model:value="footerPlacement" :options="options" style="margin-right: 8px" /><Button type="primary" @click="open4 = true">Open</Button><Drawerv-model:open="open4"title="Basic Drawer":placement="footerPlacement":footer-style="{ textAlign: 'right' }"><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p><template #footer><Button style="margin-right: 8px" @click="onClose">Cancel</Button><Button type="primary" @click="onClose">Submit</Button></template></Drawer><h2 class="mt30 mb10">自定义 header & body 样式</h2><Button type="primary" @click="open5 = true">Open</Button><Drawerv-model:open="open5":closable="false"title="Basic Drawer":header-style="{ textAlign: 'center' }":body-style="{ textAlign: 'center' }"><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p></Drawer></div>
</template>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/35512.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

iCloud备份的智能压缩与优化:释放存储空间的艺术

iCloud备份的智能压缩与优化&#xff1a;释放存储空间的艺术 iCloud作为Apple提供的云服务&#xff0c;不仅为用户带来了便捷的数据同步功能&#xff0c;还通过智能的备份压缩和优化存储选项&#xff0c;帮助用户高效管理存储空间。本文将详细解析iCloud备份的压缩和优化存储功…

《Nest系列 - 4. 听说人人都会CRUD,可是我还不会怎么办???-《4.2结合前端使用实现CRUD》

终于到了这一步&#xff0c;今天我们就将实现一个CRUD&#xff0c;主要是编写nest 部分&#xff0c;前端部分后面可以看git 代码 下面是效果演示&#xff08;大部分是参考满哥实现&#xff0c;&#x1f923;&#x1f923;&#x1f923;&#xff09; 前期准备 前端接口处理 im…

产业升级具体可从哪些方面入手?

产业升级是一个广泛而复杂的过程&#xff0c;涉及多个方面。以下是产业升级的主要方面&#xff0c;结合参考文章中的相关信息进行清晰分点表示和归纳&#xff1a; 技术创新&#xff1a; 研发和技术改造&#xff1a;通过不断投入研发和技术改造&#xff0c;推动企业生产技术的升…

RT-Thread Studio实现动态线程

1创建项目 我的板子为STM32F03ZET6 点击RT-Thread项目 2选择板子&#xff08;根据自己的板子选择&#xff09; 3找到主函数 4编写代码 4-1创建函数入口 // 线程入口函数 static void thread_entry(void *parameter) {rt_uint32_t count 0;while (1){// 线程执行的代码rt_k…

互斥锁并不能保证任务不能被调度

互斥锁不能保证在临界区的时候&#xff0c;不发送任务调度&#xff0c;所以为了保护共享的资源不被调度访问&#xff0c;需要在两个线程都加互斥锁来保证任务不调度 #include <stdio.h> #include <pthread.h> #include <unistd.h> int shared_resource 0;p…

2025中国(宁波)出口跨境电商博览会

2025中国(宁波)出口跨境电商博览会 时间&#xff1a;2025年5月28-30日 地点&#xff1a;中国宁波国际会展中心 组织单位&#xff1a; 宁波欧德国际商务咨询服务有限公司 凤麟展览(宁波)有限公司 宁波市跨境电子商务协会 宁波市家居产业协会 详询主办方陆先生 I38&…

Java yield()方法在多线程编程中的应用

目录 背景: 代码解释: 1.类定义 2.主方法(main) 3.自定义线程类(MyThread3) 总结: 背景: 在Java中&#xff0c;多线程是构建并发应用程序的关键技术。当我们多个线程需要同时执行时&#xff0c;操作系统必须决定何时为每个线程分配 CPU时。这个过程被称为线程调度&#x…

php中interface接口类,abstract抽象类和trait详解

在PHP中&#xff0c;interface&#xff08;接口&#xff09;、abstract class&#xff08;抽象类&#xff09;和trait都是为了实现代码的复用、提高可维护性和灵活性而设计的。它们各自有不同的用途和特点&#xff1a; Interface&#xff08;接口&#xff09; 用途&#xff1…

上海市计算机学会竞赛平台2023年7月月赛丙组模糊匹配(二)

题目描述 有两个仅包含大写英文字母的字符串 &#x1d446;,&#x1d447;S,T&#xff0c;且字符串 &#x1d447;T 是 &#x1d446;S 的一个子串。 但由于字符串 &#x1d446;S 字迹模糊不清&#xff0c;其某些位置上的字符没有办法进行辨认&#xff0c;这些模糊的位置&am…

星戈瑞DSPE-FITC细胞成像:一种细胞可视化技术

细胞成像技术是现代生物医学研究中的工具&#xff0c;它为我们提供了一种直观、深入地了解细胞结构和功能的方法。其中&#xff0c;DSPE-FITC作为一种荧光标记分子&#xff0c;在细胞成像领域展现出了优势。 DSPE-FITC细胞成像的原理 DSPE-FITC是由磷脂酰丝氨酸&#xff08;DS…

【openmpi】怎样使用openmpi并行运行python脚本?

创作日志&#xff1a; 装过一次openmpi&#xff0c;但是半年之后就忘记怎么用了&#xff0c;所以记录一下 1. 测试openmpi是否安装好 cd /home/xxxx/SnapHiC_Call_Loop/openmpi-4.1.6/examples make mpirun -np 4 hello_c得到如下输出就说明是装好的了 2. 没有导入路径的话导…

rtthread stm32h743的使用(十)i2c设备使用

我们要在rtthread studio 开发环境中建立stm32h743xih6芯片的工程。我们使用一块stm32h743及fpga的核心板完成相关实验&#xff0c;核心板如图&#xff1a; 1.建立新工程&#xff0c;选择相应的芯片型号及debug引脚及调试器 2.打开cubemux&#xff0c;设置外部时钟及串口外设…

劳务工程元宇宙:未来建筑行业的新趋势

随着科技的不断发展&#xff0c;虚拟现实、增强现实、区块链等技术逐渐渗透到各个行业&#xff0c;为传统产业带来了革命性的变革。在建筑行业&#xff0c;劳务工程元宇宙的概念应运而生&#xff0c;预示着未来建筑行业的新趋势。 一、劳务工程元宇宙的定义 劳务工程元宇宙是…

制造业采购堡垒机的四大必要性看这里!

制造业包括的行业广泛&#xff0c;与大家的生活息息相关&#xff0c;例如食品制造业、汽车制造业、纺织业、服装制造业等等。但大家对于制造业不是很了解&#xff0c;不知道制造业也是需要采购堡垒机的&#xff0c;今天我们就来聊聊制造业采购堡垒机的必要性。 制造业采购堡垒机…

python selenium 下载

查看浏览器版本 下载地址&#xff1a; 新版本下载地址 https://googlechromelabs.github.io/chrome-for-testing/ 历史版本也可以用这个下载地址 http://chromedriver.storage.googleapis.com/index.html 找到对应的版本 126.0.xxx 下载

推荐给中小学生的暑假打字神器

暑假是孩子们放松身心、增长知识的好时机。在这个漫长的假期里&#xff0c;家长们不仅希望孩子能够快乐地度过每一天&#xff0c;还希望他们能在学习上有所进步。尤其是随着科技的发展&#xff0c;熟练的打字技巧已经成为现代学习和工作的基本技能之一。今天&#xff0c;我要向…

节流工具,避免操作太频繁

ThrottleUtil 用于保证某个操作在一定时间内只执行一次的工具。 package com.cashpro.kash.lending.loan.utils;/*** <pre>* Created by zhuguohui* Date: 2024/6/26* Time: 13:43* Desc:用于节流执行任务,限制任务执行的频次* </pre>*/import android.os.Handle…

基于requests模块爬取网易云歌曲评论并制作热词云图

本实践大作业要求 本次实践大作业主要要求主要包括&#xff1a; 1、选择一个热点或者你感兴趣的主题作为本次爬虫实践作业要完成的任务。 2、为了完成本次任务&#xff0c;需要确定从网上爬取的数据对象与范围。 3、利用python及网络爬虫相关技术实现从网上爬取相应内容数据。 …

《ai企业级知识库》- 使用api启动rasa热切换模型文件

阿丹&#xff1a; 在线上环境在准备切换模型文件的时候&#xff0c;更希望通过一些简单&#xff0c;并且热点的方式来将训练好的模型文件进行切换。rasa这里准备了相应的接口来完成这个事情。那么这里就记录一下这接口。关于后面的部署是很重要的事情。 api模式 Rasa的API模式…

nvm-desktop window安装,支持动态切换nodejs版本

一、安装 nvm-desktop 概述 1 、卸载干净笔记的nodejs 和nodejs的环境变量 2、安装 nvm-desktop 软件 3、配置环境变量 4、测试功能 # 此时已安装完成 其他&#xff1a;常见nodejs的问题解决参考&#xff1a;官网 mac 安装教程 https://github.com/1111mp/nvm-desktop/blob/…