不使用构建工具,DOM内模版解析Vue3+element-plus示例

主要功能点:

  1. 子组件使用v-model,通过computed修改父组件数据(setup语法糖,  从 Vue 3.4 开始,推荐的实现方式是使用 defineModel() 宏)
  2. dom内局部注册组件和使用组件
  3. 使用 ES 模块构建版本

目录结构

index.html

<!DOCTYPE html>
<html lang="zh">
	<head>
		<meta charset="UTF-8">
		<meta name="viewport" content="width=device-width, initial-scale=1.0">
		<meta http-equiv="X-UA-Compatible" content="ie=edge">
		<title></title>
		<link rel="stylesheet" href="//unpkg.com/element-plus/dist/index.css" />
	</head>
	<body>
		<script type="importmap">
			{
			"imports": {
			 "vue": "https://unpkg.com/vue@3/dist/vue.esm-browser.js",
			"element-plus": "https://unpkg.com/element-plus@2.7.2/dist/index.full.min.mjs",
			"@element-plus/icons-vue": "https://unpkg.com/browse/@element-plus/icons-vue@2.3.1/dist/index.min.js"
			}
		}
		</script>
		<div id="app">
			 <el-tabs type="border-card">
			    <el-tab-pane label="父组件">
					<h3>DOM内模版解析需要遵从html解析行为,使用kebab-case为组件命名,并显式的指定闭合标签</h3>
					<h3>请注意下面讨论只适用于直接在 DOM 中编写模板的情况。如果你使用来自以下来源的字符串模板,就不需要顾虑这些限制了:</h3>
					
					<ul>
						<li>单文件组件</li>
						<li>内联模板字符串 (例如 template: '...')</li>
						<li>&lt;script type="text/x-template"&gt;</li>
					</ul>
					
					<pre>
						<code>
<h2>kebab-case命名法</h2>&lt;component-a&gt;&lt;/component-a&gt; <el-button type="success" circle ><el-icon :size="20"><Check /></el-icon></el-button> 
<h2>camelCase命名法</h2>&lt;componentB>&lt;/componentB&gt;
<h2>PascalCase命名法</h2>&lt;ComponentB&gt;&lt;/componentB&gt;
						</code>
					</pre>
				</el-tab-pane>
			    <el-tab-pane label="组件B">
					<component-b></component-b>
				</el-tab-pane>
			    <el-tab-pane label="组件C">
					<component-c/>
					</el-tab-pane>
			  </el-tabs>
		</div>


		<script type="text/x-template" id="main-template">
	<ComponentA :loginModel="loginModel"  v-model:dialogVisible="loginModel.dialogVisible"/>
	<el-button @click="loginModel.dialogVisible=true">{{ loginModel.title }}</el-button>
		 
		</script>
		<script type="module">
			import {
				createApp,
				ref,
				reactive,
				watchEffect,
				watch
			} from 'vue'
			import ElementPlus from "element-plus"
			import * as ElementPlusIconsVue from '@element-plus/icons-vue'
			import ComponentA from './components/component-a.js'
	
			//定义B组件
			const ComponentB = {
				components: {
					ComponentA,
				},
				setup() {
					const loginModel=ref({
						title:"微信登录",
						signIn:"扫码登录",
						url:"",
						tip:"使用手机微信扫码登录",
						dialogVisible:false
					})
					return {
						loginModel,
					}
				},
				template: `
			  <ComponentA :loginModel="loginModel"  v-model:dialogVisible="loginModel.dialogVisible"/>
			  <el-button @click="loginModel.dialogVisible=true">{{ loginModel.title }}</el-button>
			  `
			}		
			//定义C组件
			const ComponentC = {
				//局部注册组件
				components: {
					ComponentA,
				},
				setup() {
					const loginModel=ref({
						title:"QQ登录",
						signIn:"扫码登录",
						url:"",
						tip:"使用手机QQ扫码登录",
						dialogVisible:false
					})
					return {
						loginModel,
					}
				},
				template: "#main-template"
			}


			const app = createApp({
				components: {
					ComponentB,
					ComponentC
				},
				setup() {
					const msg = ref("msg")
					return {
						msg
					}
				}
			})
			for (const [key, component] of Object.entries(ElementPlusIconsVue)) {
			  app.component(key, component)
			}
			app.use(ElementPlus).mount('#app')
			
			
		</script>
	</body>
</html>

component-a.js

import { ref } from 'vue'
import {ElMessageBox} from "element-plus"
export default {
  props:["loginModel","dialogVisible"],
  //emits可省略但不能写错
  emits:["update:dialogVisible"],
  computed:{
	  visible:{
		  get(){
			  return this.dialogVisible
		  },
		  set(val){
			  this.$emit('update:dialogVisible',val)
		  }
	  }
  },
  methods:{
   handleClose (done)  {
	   this.visible=false
	  }
  },
  template: `
   <el-dialog
      v-model="visible"
      :title="loginModel.signIn"
      width="500"
	  style="text-align:center"
      :before-close="handleClose"
    >
    <img :src="loginModel.url"/>
	<p>{{loginModel.tip}}</p>
      <template #footer>
        <div class="dialog-footer">
          <el-button @click="visible=false">Cancel</el-button>
        </div>
      </template>
    </el-dialog>
  `
}

  • 7
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值