VUE3(三十七)Vue3.2子父组件交互(vue、ts不分离)~

简介: VUE3(三十七)Vue3.2子父组件交互(vue、ts不分离)~

之前在做博客手机端的时候,特意把底部菜单封装成了一个自定义组件,尝试了一下vue3.2语法的子传父、父传子、子调父、父调子。


首先放一下自定义组件的代码:


Menu.vue


<template>
  <!-- 手机底部菜单(别问我为什么把现成的组件又封了一个自定义组件,问就是为了试试子传父、父传子、子调父、父调子) -->
  <!-- <van-grid clickable :column-num="4">
    <van-grid-item icon="wap-home" text="首页" to="/phone/index" @click="goIndex" :icon-color="homeColor" />
    <van-grid-item icon="font" text="杂谈" to="/phone/gossip"  @click="goOther" :icon-color="otherColor"/>
    <van-grid-item icon="comment" text="归档" to="/phone/index" @click="goPlaceFile" :icon-color="placeFileColor"/>
    <van-grid-item icon="friends" text="我的" to="/phone/index" @click="goPersonal" :icon-color="personalColor" />
  </van-grid> -->
  <van-tabbar
    v-model="active"
    @change="onChange"
    active-color="#24657D"
    inactive-color="#323233"
  >
    <van-tabbar-item icon="wap-home" to="/phone/index">首页</van-tabbar-item>
    <van-tabbar-item icon="font" to="/phone/gossip">杂谈</van-tabbar-item>
    <van-tabbar-item icon="comment" to="/phone/placeFile">归档</van-tabbar-item>
    <van-tabbar-item icon="friends" to="/phone/personal">我的</van-tabbar-item>
  </van-tabbar>
</template>
<script lang="ts" setup>
import { reactive, ref, toRefs } from "vue";
// 引入axios钩子
import axios from "/@/request/axios";
// 引入路由
import { useRouter, useRoute } from "vue-router";
// 引入公共js文件
import utils from "/@/assets/js/public/function";
// api 接口文件
import {
  getIndexData,
  getRemarkList,
  getFileList,
  getUserSession,
} from "/@/api/phone/index";
import { common, userinfo } from "/@/hooks/common";
import { Toast } from "vant";
// =============================================================================
// 实例化路由
const router = useRouter();
const route = useRoute();
const data = reactive({
  // tabbar 默认选中索引
  active: 0,
});
/**
 * @name: 将data绑定值dataRef
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-01-10
 */
const { active } = toRefs(data);
//  使用defineEmits创建名称,接受一个数组
const $myemit = defineEmits([
  "getIndexDataList",
  "getOtherData",
  "getPlaceFileData",
  "getPersonalData",
]);
/**
 * @name: tabbar 发生改变时
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-08-11
 * @param:  index number  索引
 */
const onChange = (index: any): void => {};
/**
 * @name: 获取首页数据公共方法
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-08-11
 * @param:  page  number  页码
 * @param:  search  string  搜索值
 * @param:  category_id number  分类
 */
const getIndexDataList = (
  page: number = 1,
  search: string = "",
  category_id: number = 0,
  sign: boolean = true
): void => {
  // 判断是否为数字
  if (parseFloat(page).toString() == "NaN") {
    page = 1;
  }
  // 请求数据
  // getUserSessionData();
  try {
    data.active = 0;
    Toast.loading({
      message: "加载中...",
      forbidClick: true,
    });
    // utils.alertLoadExec(true);
    let param = {
      page: page,
      search: search,
      category_id: utils.encryptCode({ category_id: category_id }),
    };
    getIndexData(param).then(function (response: any) {
      response.page = page;
      response.category_id = category_id;
      // console.log(response);
      // 传递数据给父组件
      $myemit("getIndexDataList", response);
      // 自定义loading消失
      // utils.alertLoadExec(false);
      utils.sleep(1000).then(() => {
        // 这里写sleep之后需要去做的事情
        Toast.clear();
      });
    });
  } catch (error) {
    console.log("chucuole");
    // 自定义loading消失
    // utils.alertLoadExec(false);
    Toast.clear();
  }
};
/**
 * @name: 获取杂谈页数据
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-08-11
 * @param:  page  number  页码
 */
const getOtherData = (sign: Boolean = true): void => {
  Toast.loading({
    message: "加载中...",
    forbidClick: true,
  });
  // utils.alertLoadExec(true);
  // 请求数据
  // getUserSessionData();
  try {
    data.active = 1;
    let param = {};
    getRemarkList(param).then(function (response: any) {
      // console.log(response);
      // 传递数据给父组件
      $myemit("getOtherData", response);
      // 自定义loading消失
      // utils.alertLoadExec(false);
      utils.sleep(1000).then(() => {
        // 这里写sleep之后需要去做的事情
        Toast.clear();
      });
    });
  } catch (error) {
    console.log("chucuole");
    // 自定义loading消失
    // utils.alertLoadExec(false);
    Toast.clear();
  }
};
/**
 * @name: 去归档页
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-08-11
 */
const getPlaceFileData = (sign: Boolean = true): void => {
  // utils.alertLoadExec(true);
  Toast.loading({
    message: "加载中...",
    forbidClick: true,
  });
  // getUserSessionData();
  // 请求数据
  try {
    data.active = 2;
    let param = {};
    getFileList(param).then(function (response: any) {
      // console.log(response);
      // 传递数据给父组件
      $myemit("getPlaceFileData", response);
      // 自定义loading消失
      // utils.alertLoadExec(false);
      utils.sleep(5000).then(() => {
        // 这里写sleep之后需要去做的事情
        Toast.clear();
      });
    });
  } catch (error) {
    console.log("chucuole");
    // 自定义loading消失
    // utils.alertLoadExec(false);
    Toast.clear();
  }
};
/**
 * @name: 去个人中心页
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-08-11
 */
const getPersonalData = (): void => {
  data.active = 3;
};
/**
 * @name: 获取用户登录信息
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const getUserSessionData = (): void => {
  try {
    let param = {};
    getUserSession(param).then(function (response: any) {
      userinfo.email = response.email;
      userinfo.figureurl = response.figureurl;
      userinfo.nickname = response.nickname;
      userinfo.userid = response.userid;
    });
  } catch (error) {
    console.log("登录信息出错!");
  }
};
getUserSessionData();
// 将组件中的属性暴露出去,这样父组件可以获取
defineExpose({
  ...toRefs(data),
  getIndexDataList,
  getOtherData,
  getPlaceFileData,
  getPersonalData,
});
</script>
<style lang="scss">
// @import "../../assets/css/components/pc/Footer.scss";
</style>

 

父组件调用:


<template>
  <!-- header -->
  <div class="header">
    <van-nav-bar title="时间里的" />
    <!-- 搜索框 -->
    <van-search
      v-model="search"
      show-action
      placeholder="请输入搜索关键词(本站采用sphinx搜索引擎)"
    >
      <template #action>
        <div @click="getSearchData">搜索</div>
      </template>
    </van-search>
    <!-- <van-search v-model="search" placeholder="请输入搜索关键词(本站采用sphinx搜索引擎)" clearable @click-left-icon="getSearchData" /> -->
  </div>
  <!-- body -->
  <!-- <van-pull-refresh v-model="loading" @refresh="onRefresh"> -->
  <div class="body">
    <van-tabs v-model:active="active" animated @click-tab="onClickTab">
      <!-- 全部 -->
      <van-tab title="全部" name="0">
        <div
          class="article"
          v-for="(item, index) in dataList"
          :key="index"
          @click="jumpArticleDetail(item.id)"
        >
          <div class="title" v-html="item.arttitle"></div>
          <div class="desc">
            <div class="left-div">
              <div class="author">{{ item.another }} | {{ item.putime }}</div>
              <div class="desc-show" v-html="item.artdesc"></div>
            </div>
            <div class="right-div">
              <img :src="item.artlogo" style="width: 100%" class="lazyload" />
            </div>
          </div>
          <div class="label">
            <div class="label-left">
              <van-icon name="browsing-history-o" />&nbsp;{{ item.click_num }}
            </div>
            <div class="label-right">
              {{ item.labelStr }}
            </div>
          </div>
        </div>
        <div class="next-page" @click="nextPage" v-if="articlePage > page">
          点击加载下一页
        </div>
      </van-tab>
      <!-- 循环 -->
      <van-tab
        v-for="(it, ind) in categoryList"
        :key="ind"
        :title="it.cat_name"
        :name="it.id"
      >
        <div
          class="article"
          v-for="(item, index) in dataList"
          :key="index"
          @click="jumpArticleDetail(item.id)"
        >
          <div class="title" v-html="item.arttitle"></div>
          <div class="desc">
            <div class="left-div">
              <div class="author">{{ item.another }} | {{ item.putime }}</div>
              <div class="desc-show" v-html="item.artdesc"></div>
            </div>
            <div class="right-div">
              <img :src="item.artlogo" style="width: 100%" class="lazyload" />
            </div>
          </div>
          <div class="label">
            <div class="label-left">
              <van-icon name="browsing-history-o" />&nbsp;{{ item.click_num }}
            </div>
            <div class="label-right">
              {{ item.labelStr }}
            </div>
          </div>
        </div>
        <div class="next-page" @click="nextPage" v-if="articlePage > page">
          点击加载下一页
        </div>
      </van-tab>
    </van-tabs>
  </div>
  <img
    src="https://resource.guanchao.site/uploads/gotop/timg.png"
    class="go_top"
    @click="goToTop"
  />
  <!-- </van-pull-refresh> -->
  <!-- navbar -->
  <div class="footer">
    <!-- <Menu @getIndexDataList="goIndex" ref="MenuRef" /> -->
    <Menu @getIndexDataList="goIndex" ref="MenuRef" />
  </div>
</template>
<script lang="ts" setup>
import {
  PropType,
  ref,
  watch,
  reactive,
  toRefs,
  provide,
  inject,
  onBeforeMount, // 在组件挂载之前执行的函数
  onMounted,
  nextTick,
} from "vue";
// 引入路由
import { useRouter, useRoute } from "vue-router";
// 引入公共js文件
import utils from "/@/assets/js/public/function";
// 引入子组件
import Menu from "/@/components/phone/Menu.vue";
import { common, userinfo } from "/@/hooks/common";
import { Toast } from "vant";
// =============================================================================
// 实例化路由
const router = useRouter();
const route = useRoute();
/**
 * @name: 声明data
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-01-18
 */
const data = reactive({
  // 文章列表
  dataList: Array<any>(),
  // 分类列表
  categoryList: Array<any>(),
  // 页码
  page: <number>(route.query.page ? route.query.page : 1),
  // 子分类id
  category_id: <number>(route.query.category_id ? route.query.category_id : 0),
  // 页码(控制加载下一页是否显示)
  articlePage: <number>1,
  // 是否请求
  req: <boolean>(route.query.req ? route.query.req : true),
  // 下拉刷新标识
  loading: <boolean>false,
  // 搜索值
  search: <string>"",
});
/**
 * @name: 将data绑定值dataRef
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-01-10
 */
const {
  dataList,
  categoryList,
  page,
  category_id,
  articlePage,
  loading,
  search,
} = toRefs(data);
// ===============================================================================
// 子组件相关
/**
 * @name: 定义子组件暴露给父组件的值属性列表
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
interface menuData {
  goIndex: () => void;
  getIndexDataList: () => void;
  homeColor: string;
  otherColor: string;
  placeFileColor: string;
  personalColor: string;
}
// 子组件ref(TypeScript语法)下面这这两种写法也可以,推荐使用Typescript语法
const MenuRef = ref<InstanceType<typeof Menu> & menuData>();
// const MenuRef = ref<InstanceType<typeof Menu>>()
// const MenuRef = ref(null)
// ===============================================================================
// 方法
const goToTop = () => {
  // 滚动条回顶部
  let json = document.getElementById("body");
  if (json != null) {
    json.scrollTop = 0;
  }
};
/**
 * @name: 下拉刷新方法
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const onRefresh = () => {
  // 执行子组件方法
  MenuRef.value?.getIndexDataList(data.page);
  Toast("刷新成功");
  data.loading = false;
};
/**
 * @name: 跳转文章详情页
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 * @param:  category_id number  分类
 */
const jumpArticleDetail = (article_id: Number) => {
  router.push({
    path: "/phone/articleDetail",
    query: {
      article_id: utils.encryptCode({ article_id: article_id }),
      path: "index",
    },
  });
};
/**
 * @name: 子组件向父组件抛出的方法
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const goIndex = (response: any): void => {
  try {
    data.page = response.page;
    data.category_id = response.category_id;
    if (data.page > 1) {
      // 数组合并
      data.dataList.push(...response.articleShow);
    } else {
      // 数组赋值
      data.dataList = response.articleShow;
    }
    data.categoryList = response.cateList;
    data.articlePage = response.articlePage;
  } catch (error) {
    console.log("出错了");
  }
};
/**
 * @name: 获取搜索数据
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const getSearchData = (): void => {
  data.page = 1;
  MenuRef.value?.getIndexDataList(data.page, data.search, data.category_id);
};
/**
 * @name: 加载下一页数据
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const nextPage = (): void => {
  data.page += 1;
  // 调用子组件的方法
  MenuRef.value?.getIndexDataList(data.page, data.search, data.category_id);
};
/**
 * @name: 获取分类下的文章列表
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const onClickTab = (obj: number): void => {
  data.category_id = obj;
  data.page = 1;
  // 调用子组件的方法
  MenuRef.value?.getIndexDataList(data.page, "", data.category_id);
};
/**
 * @name: nextTick 页面发生变化即渲染
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
nextTick(() => {
  /*// 获取子组件name
    console.log(MenuRef._value)
    console.log(MenuRef.__v_isRef)
    console.log(MenuRef.__v_isShallow)
    console.log(MenuRef._rawValue)
    console.log(MenuRef.value.count)
    // 加个问号这种写法,没有属性也不会报错
    console.log(MenuRef.value?.homeColor)
    // 执行子组件方法
    MenuRef.value.goIndex()//*/
});
/**
 * @name: 生命周期---页面挂载完成
 * @author: camellia
 * @email: guanchao_gc@qq.com
 * @date: 2022-07-18
 */
const mounted = onMounted(() => {
  // console.log('获取子组件中的性别', MenuRef );
  // console.log('获取子组件中的其他信息', MenuRef.value?.info );
  // console.log('获取子组件中的其他信息', MenuRef.value.homeColor );
  // 执行子组件方法
  MenuRef.value?.getIndexDataList(data.page);
});
</script>
<style>
em {
  color: #f73131;
}
</style>
<style scoped lang="scss">
@import "/@/assets/css/phone/index.scss";
</style>


还是最开始说的,组件封的挺鸡肋的,但是尝试了一下字符组件的交互,简单讲:父组件调用子组件的方法,子组件的方法获取数据,再将数据传递回父组件,父组件拿到数据之后再进行页面的数据渲染。

 

具体使用的方法,请参见VUE3官方文档:cn.vuejs.org/guide/types…


 

或者,参见掘金VUE3.2子父组件总结文章:juejin.cn/post/700610…


 

但是,把html代码和ts代码写到一起可能不是一个太好的习惯。我琢磨琢磨怎么把他们分离开。

 


以上大概就是我自己尝试的VUE3.2语法的子父组件的交互。

 


有好建议,请在下方是输入你的评论。


目录
相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
142 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,都能轻松实现高效的状态管理和数据处理。
38 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` 实现自定义组件的双向数据绑定。
|
2月前
|
JavaScript 索引
Vue 3.x 版本中双向数据绑定的底层实现有哪些变化
从Vue 2.x的`Object.defineProperty`到Vue 3.x的`Proxy`,实现了更高效的数据劫持与响应式处理。`Proxy`不仅能够代理整个对象,动态响应属性的增删,还优化了嵌套对象的处理和依赖追踪,减少了不必要的视图更新,提升了性能。同时,Vue 3.x对数组的响应式处理也更加灵活,简化了开发流程。
|
2月前
|
JavaScript 前端开发 API
从Vue 2到Vue 3的演进
从Vue 2到Vue 3的演进
44 0
|
2月前
|
JavaScript 前端开发 API
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
65 0

热门文章

最新文章