Vue3警告提示(Alert)

简介: 这是一个基于 Vue 的警告提示组件 Alert,提供了成功、信息、警告和错误四种样式,并支持自定义内容、图标及操作项。

效果如下图:在线预览

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

APIs

Alert

参数 说明 类型 默认值
message 警告提示内容 string | slot undefined
description 警告提示的辅助性文字介绍 string | slot undefined
type 指定警告提示的样式,有四种选择 successinfowarningerror ‘success’ | ‘info’ | ‘warning’ | ‘error’ ‘info’
closable 是否显示关闭按钮 boolean false
closeText 自定义关闭按钮 string | slot undefined
icon 自定义图标,showIcontrue 时有效 string | slot undefined
showIcon 是否显示辅助图标 boolean false
actions 自定义操作项 slot undefined

Events

名称 说明 类型
close 关闭时触发的回调函数 (e: Event) => void

创建警告提示组件Alert.vue

其中引入使用了以下工具函数:

<script setup lang="ts">
import { ref, computed, watchPostEffect } from 'vue'
import type { Slot } from 'vue'
import { useSlotsExist } from '../utils'
interface Props {
  message?: string // 警告提示内容 string | slot
  description?: string // 警告提示的辅助性文字介绍 string | slot
  type?: 'success' | 'info' | 'warning' | 'error' // 指定警告提示的样式
  closable?: boolean // 是否显示关闭按钮
  closeText?: string // 自定义关闭按钮 string | slot
  icon?: string // 自定义图标,showIcon 为 true 时有效 string | slot
  showIcon?: boolean // 是否显示辅助图标
  actions?: Slot // 自定义操作项 slot
}
const props = withDefaults(defineProps<Props>(), {
  message: undefined,
  description: undefined,
  type: 'info',
  closable: false,
  closeText: undefined,
  icon: undefined,
  showIcon: false,
  actions: undefined
})
const alert = ref()
const closeAlert = ref(false)
const emit = defineEmits(['close'])
const slotsExist = useSlotsExist(['description'])
const showDesc = computed(() => {
  return slotsExist.description || props.description
})
watchPostEffect(() => {
  if (props.closable && !closeAlert.value) {
    alert.value.style.height = alert.value.offsetHeight + 'px'
  }
})
function onClose(e: Event): void {
  closeAlert.value = true
  emit('close', e)
}
</script>
<template>
  <Transition name="alert-motion">
    <div
      v-if="!closeAlert"
      ref="alert"
      class="m-alert"
      :class="[
        `alert-${type}`,
        {
          'alert-width-description': showDesc
        }
      ]"
    >
      <template v-if="showIcon">
        <span v-if="!showDesc" class="m-alert-icon">
          <slot name="icon">
            <img v-if="icon" :src="icon" class="icon-img" />
            <svg
              v-else-if="type === 'info'"
              class="icon-svg"
              focusable="false"
              data-icon="info-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm32 664c0 4.4-3.6 8-8 8h-48c-4.4 0-8-3.6-8-8V456c0-4.4 3.6-8 8-8h48c4.4 0 8 3.6 8 8v272zm-32-344a48.01 48.01 0 010-96 48.01 48.01 0 010 96z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'success'"
              class="icon-svg"
              focusable="false"
              data-icon="check-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm193.5 301.7l-210.6 292a31.8 31.8 0 01-51.7 0L318.5 484.9c-3.8-5.3 0-12.7 6.5-12.7h46.9c10.2 0 19.9 4.9 25.9 13.3l71.2 98.8 157.2-218c6-8.3 15.6-13.3 25.9-13.3H699c6.5 0 10.3 7.4 6.5 12.7z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'warning'"
              class="icon-svg"
              focusable="false"
              data-icon="exclamation-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm-32 232c0-4.4 3.6-8 8-8h48c4.4 0 8 3.6 8 8v272c0 4.4-3.6 8-8 8h-48c-4.4 0-8-3.6-8-8V296zm32 440a48.01 48.01 0 010-96 48.01 48.01 0 010 96z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'error'"
              class="icon-svg"
              focusable="false"
              data-icon="close-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm165.4 618.2l-66-.3L512 563.4l-99.3 118.4-66.1.3c-4.4 0-8-3.5-8-8 0-1.9.7-3.7 1.9-5.2l130.1-155L340.5 359a8.32 8.32 0 01-1.9-5.2c0-4.4 3.6-8 8-8l66.1.3L512 464.6l99.3-118.4 66-.3c4.4 0 8 3.5 8 8 0 1.9-.7 3.7-1.9 5.2L553.5 514l130 155c1.2 1.5 1.9 3.3 1.9 5.2 0 4.4-3.6 8-8 8z"
              ></path>
            </svg>
          </slot>
        </span>
        <span class="m-big-icon" v-else>
          <slot name="icon">
            <img v-if="icon" :src="icon" class="big-icon-img" />
            <svg
              v-else-if="type === 'info'"
              class="icon-svg"
              focusable="false"
              data-icon="info-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm0 820c-205.4 0-372-166.6-372-372s166.6-372 372-372 372 166.6 372 372-166.6 372-372 372z"
              ></path>
              <path
                d="M464 336a48 48 0 1096 0 48 48 0 10-96 0zm72 112h-48c-4.4 0-8 3.6-8 8v272c0 4.4 3.6 8 8 8h48c4.4 0 8-3.6 8-8V456c0-4.4-3.6-8-8-8z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'success'"
              class="icon-svg"
              focusable="false"
              data-icon="check-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M699 353h-46.9c-10.2 0-19.9 4.9-25.9 13.3L469 584.3l-71.2-98.8c-6-8.3-15.6-13.3-25.9-13.3H325c-6.5 0-10.3 7.4-6.5 12.7l124.6 172.8a31.8 31.8 0 0051.7 0l210.6-292c3.9-5.3.1-12.7-6.4-12.7z"
              ></path>
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm0 820c-205.4 0-372-166.6-372-372s166.6-372 372-372 372 166.6 372 372-166.6 372-372 372z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'warning'"
              class="icon-svg"
              focusable="false"
              data-icon="exclamation-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm0 820c-205.4 0-372-166.6-372-372s166.6-372 372-372 372 166.6 372 372-166.6 372-372 372z"
              ></path>
              <path
                d="M464 688a48 48 0 1096 0 48 48 0 10-96 0zm24-112h48c4.4 0 8-3.6 8-8V296c0-4.4-3.6-8-8-8h-48c-4.4 0-8 3.6-8 8v272c0 4.4 3.6 8 8 8z"
              ></path>
            </svg>
            <svg
              v-else-if="type === 'error'"
              class="icon-svg"
              focusable="false"
              data-icon="close-circle"
              width="1em"
              height="1em"
              fill="currentColor"
              aria-hidden="true"
              viewBox="64 64 896 896"
            >
              <path
                d="M685.4 354.8c0-4.4-3.6-8-8-8l-66 .3L512 465.6l-99.3-118.4-66.1-.3c-4.4 0-8 3.5-8 8 0 1.9.7 3.7 1.9 5.2l130.1 155L340.5 670a8.32 8.32 0 00-1.9 5.2c0 4.4 3.6 8 8 8l66.1-.3L512 564.4l99.3 118.4 66 .3c4.4 0 8-3.5 8-8 0-1.9-.7-3.7-1.9-5.2L553.5 515l130.1-155c1.2-1.4 1.8-3.3 1.8-5.2z"
              ></path>
              <path
                d="M512 65C264.6 65 64 265.6 64 513s200.6 448 448 448 448-200.6 448-448S759.4 65 512 65zm0 820c-205.4 0-372-166.6-372-372s166.6-372 372-372 372 166.6 372 372-166.6 372-372 372z"
              ></path>
            </svg>
          </slot>
        </span>
      </template>
      <div class="m-alert-content">
        <div class="alert-message">
          <slot name="message">{
  { message }}</slot>
        </div>
        <div class="alert-description" v-if="showDesc">
          <slot name="description">{
  { description }}</slot>
        </div>
      </div>
      <div class="m-alert-actions">
        <slot name="actions"></slot>
      </div>
      <a v-if="closable" tabindex="0" class="m-alert-close" @click="onClose" @keydown.enter.prevent="onClose">
        <slot name="closeText">
          <span v-if="closeText">{
  { closeText }}</span>
          <svg
            v-else
            class="alert-close"
            focusable="false"
            data-icon="close"
            width="1em"
            height="1em"
            fill="currentColor"
            aria-hidden="true"
            viewBox="64 64 896 896"
          >
            <path
              d="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>
        </slot>
      </a>
    </div>
  </Transition>
</template>
<style lang="less" scoped>
.alert-motion-enter-active,
.alert-motion-leave-active {
  overflow: hidden;
  transition:
    height 0.3s cubic-bezier(0.78, 0.14, 0.15, 0.86),
    opacity 0.3s cubic-bezier(0.78, 0.14, 0.15, 0.86),
    padding 0.3s cubic-bezier(0.78, 0.14, 0.15, 0.86);
}
.alert-motion-leave-to {
  height: 0 !important;
  opacity: 0 !important;
  padding-block: 0 !important;
}
.m-alert {
  padding: 8px 12px;
  color: rgba(0, 0, 0, 0.88);
  font-size: 14px;
  line-height: 1.5714285714285714;
  position: relative;
  display: flex;
  align-items: center;
  word-break: break-all;
  border-radius: 8px;
  .m-alert-icon {
    display: inline-block;
    margin-right: 8px;
    line-height: 0;
  }
  .m-big-icon {
    margin-right: 12px;
    font-size: 24px;
    line-height: 0;
  }
  .icon-img {
    display: inline-block;
    width: 14px;
    height: 14px;
  }
  .big-icon-img {
    display: inline-block;
    width: 24px;
    height: 24px;
  }
  .icon-svg {
    display: inline-block;
  }
  .m-alert-content {
    flex: 1;
    min-width: 0;
  }
  .m-alert-actions {
    margin-left: 8px;
  }
  .m-alert-close {
    margin-left: 8px;
    font-size: 12px;
    color: rgba(0, 0, 0, 0.45);
    line-height: 12px;
    cursor: pointer;
    outline: none;
    transition: color 0.2s;
    &:hover {
      color: rgba(0, 0, 0, 0.88);
    }
    .alert-close {
      display: inline-block;
      vertical-align: bottom;
      fill: rgba(0, 0, 0, 0.45);
      transition: fill 0.2s;
      &:hover {
        fill: rgba(0, 0, 0, 0.88);
      }
    }
  }
}
.alert-info {
  background-color: #e6f4ff;
  border: 1px solid #91caff;
  .m-alert-icon,
  .m-big-icon {
    color: @themeColor;
    .icon-svg,
    :deep(svg) {
      fill: currentColor;
    }
  }
}
.alert-success {
  background-color: #f6ffed;
  border: 1px solid #b7eb8f;
  .m-alert-icon,
  .m-big-icon {
    color: #52c41a;
    .icon-svg,
    :deep(svg) {
      fill: currentColor;
    }
  }
}
.alert-warning {
  background-color: #fffbe6;
  border: 1px solid #ffe58f;
  .m-alert-icon,
  .m-big-icon {
    color: #faad14;
    .icon-svg,
    :deep(svg) {
      fill: currentColor;
    }
  }
}
.alert-error {
  background-color: #fff2f0;
  border: 1px solid #ffccc7;
  .m-alert-icon,
  .m-big-icon {
    color: #ff4d4f;
    .icon-svg,
    :deep(svg) {
      fill: currentColor;
    }
  }
}
.alert-width-description {
  align-items: flex-start;
  padding: 20px 24px;
  .alert-message {
    display: block;
    margin-bottom: 8px;
    color: rgba(0, 0, 0, 0.88);
    font-size: 16px;
  }
  .alert-description {
    display: block;
  }
}
</style>

在使用的页面引入

其中引入使用了以下组件:

<script setup lang="ts">
import Alert from './Alert.vue'
import { SmileOutlined } from '@ant-design/icons-vue'
function onClose(e: Event) {
  console.log(e, 'I was closed.')
}
</script>
<template>
  <div style="width: 425px">
    <h1>{
  { $route.name }} {
  { $route.meta.title }}</h1>
    <h2 class="mt30 mb10">共有四种样式 success、info、warning、error</h2>
    <Flex vertical>
      <Alert message="Info Text" />
      <Alert message="Success Text" type="success" />
      <Alert message="Warning Text" type="warning" />
      <Alert message="Error Text" type="error" />
    </Flex>
    <h2 class="mt30 mb10">可关闭的警告提示</h2>
    <Flex vertical>
      <Alert
        message="Error Text"
        description="Error Description Error Description Error Description Error Description Error Description Error Description"
        type="error"
        closable
        @close="onClose"
      />
      <Alert
        message="Warning Text Warning Text Warning Text Warning Text Warning Text Warning Text Warning Text"
        type="warning"
        closable
        @close="onClose"
      />
    </Flex>
    <h2 class="mt30 mb10">辅助性文字介绍</h2>
    <Flex vertical>
      <Alert message="Success Text" type="success">
        <template #description>
          <p>
            Success Description
            <span style="color: red">Success</span>
            Description Success Description
          </p>
        </template>
      </Alert>
      <Alert
        message="Info Text"
        description="Info Description Info Description Info Description Info Description"
        type="info"
      />
      <Alert
        message="Warning Text"
        description="Warning Description Warning Description Warning Description Warning Description"
        type="warning"
      />
      <Alert
        message="Error Text"
        description="Error Description Error Description Error Description Error Description"
        type="error"
      />
    </Flex>
    <h2 class="mt30 mb10">辅助图标</h2>
    <Flex vertical>
      <Alert message="Success Tips" type="success" show-icon />
      <Alert message="Informational Notes" type="info" show-icon />
      <Alert message="Warning" type="warning" show-icon />
      <Alert message="Error" type="error" show-icon />
      <Alert
        message="Success Tips"
        description="Detailed description and advices about successful copywriting."
        type="success"
        show-icon
      />
      <Alert
        message="Informational Notes"
        description="Additional description and informations about copywriting."
        type="info"
        show-icon
      />
      <Alert message="Warning" description="This is a warning notice about copywriting." type="warning" show-icon />
      <Alert message="Error" description="This is an error message about copywriting." type="error" show-icon />
    </Flex>
    <h2 class="mt30 mb10">自定义关闭文字</h2>
    <Alert message="Info Text" type="info" closable close-text="Close Now" />
    <h2 class="mt30 mb10">自定义图标</h2>
    <Flex vertical>
      <Alert message="Success Tips" type="success" show-icon>
        <template #icon>
          <SmileOutlined />
        </template>
      </Alert>
      <Alert message="Informational Notes" type="info" show-icon>
        <template #icon>
          <SmileOutlined />
        </template>
      </Alert>
      <Alert message="Warning" type="warning" show-icon>
        <template #icon>
          <SmileOutlined />
        </template>
      </Alert>
      <Alert message="Error" type="error" show-icon>
        <template #icon>
          <SmileOutlined />
        </template>
      </Alert>
      <Alert message="Success" type="success" show-icon>
        <template #icon>
          <Avatar size="small" src="https://cdn.jsdelivr.net/gh/themusecatcher/resources@0.0.5/1.jpg" />
        </template>
      </Alert>
      <Alert
        message="Success Tips"
        description="Detailed description and advices about successful copywriting."
        type="success"
        show-icon
      >
        <template #icon>
          <smile-outlined />
        </template>
      </Alert>
      <Alert
        message="Informational Notes"
        description="Additional description and informations about copywriting."
        type="info"
        show-icon
      >
        <template #icon>
          <smile-outlined />
        </template>
      </Alert>
      <Alert message="Warning" description="This is a warning notice about copywriting." type="warning" show-icon>
        <template #icon>
          <smile-outlined />
        </template>
      </Alert>
      <Alert message="Error" description="This is an error message about copywriting." type="error" show-icon>
        <template #icon>
          <smile-outlined />
        </template>
      </Alert>
      <Alert
        message="Info Tips"
        description="Detailed description and advices about successful copywriting."
        type="info"
        show-icon
      >
        <template #icon>
          <Avatar size="large" src="https://cdn.jsdelivr.net/gh/themusecatcher/resources@0.0.5/2.jpg" />
        </template>
      </Alert>
    </Flex>
    <h2 class="mt30 mb10">自定义操作项</h2>
    <Flex vertical>
      <Alert message="Success Tips" type="success" show-icon closable>
        <template #actions>
          <Button size="small" type="text">UNDO</Button>
        </template>
      </Alert>
      <Alert
        message="Error Text"
        show-icon
        description="Error Description Error Description Error Description Error Description"
        type="error"
      >
        <template #actions>
          <Button size="small" type="danger" ghost>Detail</Button>
        </template>
      </Alert>
      <Alert message="Warning Text" type="warning" closable>
        <template #actions>
          <Button size="small" type="text">Done</Button>
        </template>
      </Alert>
      <Alert
        message="Info Text"
        description="Info Description Info Description Info Description Info Description"
        type="info"
        closable
      >
        <template #actions>
          <Space vertical gap="small" align="center">
            <Button size="small" type="primary">Accept</Button>
            <Button size="small" type="danger" ghost>Decline</Button>
          </Space>
        </template>
      </Alert>
    </Flex>
  </div>
</template>
相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
143 64
|
2月前
|
JavaScript 前端开发 API
Vue 3 中 v-model 与 Vue 2 中 v-model 的区别是什么?
总的来说,Vue 3 中的 `v-model` 在灵活性、与组合式 API 的结合、对自定义组件的支持等方面都有了明显的提升和改进,使其更适应现代前端开发的需求和趋势。但需要注意的是,在迁移过程中可能需要对一些代码进行调整和适配。
115 60
|
10天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
39 3
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
39 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
33 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
42 1
|
2月前
|
JavaScript
在 Vue 3 中,如何使用 v-model 来处理自定义组件的双向数据绑定?
需要注意的是,在实际开发中,根据具体的业务需求和组件设计,可能需要对上述步骤进行适当的调整和优化,以确保双向数据绑定的正确性和稳定性。同时,深入理解 Vue 3 的响应式机制和组件通信原理,将有助于更好地运用 `v-model` 实现自定义组件的双向数据绑定。
|
4天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
39 1
|
15天前
|
JavaScript 关系型数据库 MySQL
基于VUE的校园二手交易平台系统设计与实现毕业设计论文模板
基于Vue的校园二手交易平台是一款专为校园用户设计的在线交易系统,提供简洁高效、安全可靠的二手商品买卖环境。平台利用Vue框架的响应式数据绑定和组件化特性,实现用户友好的界面,方便商品浏览、发布与管理。该系统采用Node.js、MySQL及B/S架构,确保稳定性和多功能模块设计,涵盖管理员和用户功能模块,促进物品循环使用,降低开销,提升环保意识,助力绿色校园文化建设。
|
2月前
|
JavaScript 前端开发 开发者
vue学习第一章
欢迎来到我的博客!我是瑞雨溪,一名热爱前端的大一学生,专注于JavaScript与Vue,正向全栈进发。博客分享Vue学习心得、命令式与声明式编程对比、列表展示及计数器案例等。关注我,持续更新中!🎉🎉🎉
46 1
vue学习第一章

热门文章

最新文章