vue通過js代碼實例化組件

来源:https://www.cnblogs.com/JayLily/archive/2023/09/16/17706105.html
-Advertisement-
Play Games

最近在寫項目的一些公共組件(一些選擇器),很多個地方都需要用,所以在main.js全局聲明瞭,但發現子頁面調用還是有挺多的地方需寫。 例如,要在template實例化組件,並用ref綁定,然後在js里的methods里寫方法。 main.js 聲明全局組件 第一種方案 一開始想到的是用ref綁定組件 ...


最近在寫項目的一些公共組件(一些選擇器),很多個地方都需要用,所以在main.js全局聲明瞭,但發現子頁面調用還是有挺多的地方需寫。

例如,要在template實例化組件,並用ref綁定,然後在js里的methods里寫方法。

main.js 聲明全局組件

第一種方案

一開始想到的是用ref綁定組件,業務組件實例化公共組件,並賦予ref,然後通過這個ref綁定,直接調用公共組件的方法(為了一定能觸發方法),例如  this.$refs.xxx.open()

如果用戶在公共組件中,選擇好數據操作完成後,公共組件觸發emit方法,通過回調方法的方式通知業務組件,例如 this.$emit("getTableSelect",this.selection)

以下是相關代碼:

公共組件

<template>
  <div>
    <el-dialog
      v-loading="loading"
      :before-close="cancel"
      :close-on-click-modal="false"
      :element-loading-text="loadingText"
      :visible="dialogVisible"
      append-to-body
      title="科目選擇器"
      width="80%"
    >
      <el-form
        ref="queryForm"
        :inline="true"
        :model="queryParams"
        label-width="68px"
        size="small"
      >
        <el-form-item label="名稱" prop="name">
          <el-input
            v-model="queryParams.name"
            clearable
            placeholder="請輸入名稱"
            @keyup.enter.native="handleQuery"
          />
        </el-form-item>
        <el-form-item label="狀態" prop="status">
          <el-select
            v-model="queryParams.status"
            clearable
            placeholder="請選擇狀態"
          >
            <el-option
              v-for="dict in dict.type.sys_normal_disable"
              :key="dict.value"
              :label="dict.label"
              :value="dict.value"
            />
          </el-select>
        </el-form-item>
        <el-form-item>
          <el-button
            icon="el-icon-search"
            size="mini"
            type="primary"
            @click="handleQuery"
            >搜索
          </el-button>
        </el-form-item>
      </el-form>

      <el-table
        v-loading="loading"
        :data="dataList"
        @row-dblclick="rowDblclick"
        @selection-change="handleSelectionChange"
      >
        <el-table-column align="center" type="selection" width="55" />
        <el-table-column align="center" label="科目id" prop="subjectId" />
        <el-table-column align="center" label="名稱" prop="name" />
        <el-table-column align="center" label="介紹" prop="intro" />
        <el-table-column align="center" label="排序" prop="sort" />
        <el-table-column align="center" label="狀態" prop="status">
          <template slot-scope="scope">
            <dict-tag
              :options="dict.type.sys_normal_disable"
              :value="scope.row.status"
            />
          </template>
        </el-table-column>
        <el-table-column align="center" label="備註" prop="remark" />
      </el-table>
      <div slot="footer" style="text-align: center">
        <el-button type="primary" @click="submitForm">確 定</el-button>
        <el-button @click="cancel">關閉</el-button>
      </div>
    </el-dialog>
  </div>
</template>
<script>
import { listSubject } from "@/api/as/subject";

export default {
  name: "TableSelect",
  dicts: ["sys_normal_disable"],
  data() {
    return {
      loading: false,
      loadingText: "數據正在處理中...",
      dialogVisible: false,
      // 查詢參數
      queryParams: {
        pageNum: 1,
        pageSize: 10,
        name: null,
        status: null,
      },
      // 總條數
      total: 0,
      //表格數據
      dataList: [],
      //被選擇的行
      selection: [],
    };
  },

  methods: {
    /**
     * 打開視窗
     */
    open(name) {
      this.dialogVisible = true;
      this.queryParams.name = name;
      this.handleQuery();
    },
    /** 搜索按鈕操作 */
    handleQuery() {
      this.queryParams.pageNum = 1;
      this.getList();
    },
    /** 查詢科目列表 */
    getList() {
      this.loading = true;
      listSubject(this.queryParams).then((response) => {
        this.dataList = response.rows;
        this.total = response.total;
        this.loading = false;
      });
    },
    //取消
    cancel() {
      this.dialogVisible = false;
      this.reset();
    },
    //確定
    submitForm(row) {
      if (row) {
        this.$emit("getTableSelect", row);
      } else if (this.selection.length > 0) {
        this.$emit("getTableSelect", this.selection);
      } else {
        this.$message.warning("請選擇");
        return;
      }

      this.dialogVisible = false;
      this.reset();
    },
    //重置信息
    reset() {
      this.queryParams = {
        pageNum: 1,
        pageSize: 10,
        name: null,
        status: null,
      };
      this.dataList = [];
      this.selection = [];
    },
    //表格選擇改變
    handleSelectionChange(selection) {
      this.selection = selection;
    },
    //雙擊
    rowDblclick(row) {
      this.submitForm(row);
    },
  },
};
</script>
<style lang="scss" scoped></style>

業務組件(簡化)

<template>
  <div class="app-container">
    <el-input
      v-model="queryParams.name"
      clearable
      placeholder="回車搜索科目"
      @keyup.enter.native="handleTableSelect"
    />
    <table-select ref="tableSelect" @getTableSelect="getTableSelect" />
  </div>
</template>

<script>
export default {
  name: "Chapter",

  data() {
    return {
      // 遮罩層
      loading: true,
      // 查詢參數
      queryParams: {
        name: null,
      },
    };
  },

  methods: {
    //去打開視窗選擇
    handleTableSelect() {
      this.$refs.tableSelect.open(this.queryParams.name);
    },
    //獲取選擇的行
    getTableSelect(row) {
      this.queryParams.name = row.name;
    },
  },
};
</script>

效果圖

 

 

 

第一種方案,雖然可以使用,但業務組件若要使用,則需要正確在4個地方寫上對應的代碼

即,1.實例化公共組件,並寫上ref和getTableSelect的事件監聽,2.輸入框寫上回車事件,3.回車事件里觸發公共組件的方法,4.寫getTableSelect事件監聽的方法。

 

若別的小伙伴一不小心,忘記了其中一個地方(大多數都是實例化組件,或者沒寫getTableSelect事件監聽的方法,我們程式員,ctrl+c,ctrl+v,漏拷貝一點點方法,很正常吧),則會需排查原因,相對繁瑣。

後面我想了一下,再通過百度,寫了第二種方案。

第二種方案

為了減少出現拷貝少問題,我把第3步和第4步合併了(3.回車事件里觸發公共組件的方法,4.寫getTableSelect事件監聽的方法),用Promise。

以下是相關的改動

公共組件改動情況

 

 業務組件改動情況

 

 這樣做了之後,好處就是,觸發的事件和接收結果的方法在一起了,減少出錯的概率,代碼也更加方便理解了。第二種方案,也是我一直用的方案,

但第二種方案,依舊要寫3個地方,即1.實例化公共組件,並寫上ref,2.輸入框寫上回車事件,3.回車事件里觸發公共組件的方法並接收值,

這種情況下,還是有小伙伴會出現拷貝少的問題(沒有實例化公共組件),我也曾想過,怎麼去除這個實例化的代碼,用js代碼實例化組件,但受限於個人水平不夠,我也沒想到什麼好的方法解決這個問題,所以這個問題就一直擱置了。

直到今天(2023-9-16),我心血來潮,想看看有沒有辦法解決沒有實例化公共組件問題,經過長時間的百度,和查看vue官方文檔,終於找到了一種可以在js代碼上實例化組件的方案,也就是現在的第三種方案。

第三種方案

公共組件沒有任何改變

新增一個公共的js方法

import Vue from "vue";

/**
 * 獲取表格選擇
 * @param name 搜索值
 */
export function getTableSelect(name) {
  //獲取公共里的實例
  const MyComponent = Vue.component("TableSelect");
  let myComponent = new MyComponent();
  //掛載實例
  let jbxxModal = myComponent.$mount();

  return new Promise((resolve, reject) => {
    jbxxModal
      .open(name)
      .then((row) => {
        resolve(row);
      })
      .catch((res) => {
        reject(res);
      })
      .finally(() => {
        //銷毀實例
        myComponent.$destroy();
      });
  });
}

 

main.js,新增以下代碼

 業務組件改動情況

 通過第三種方案,別的小伙伴只需要調用公共的js方法即可,不需要實例化了組件了,大大的減少了出錯的概率了,

現在只需要1.輸入框寫上回車事件,2.回車事件里觸發公共組件的方法並接收值。

通過這次學習,提升了個人的一丟丟前端知識,記錄存檔,方便以後還知道怎麼解決這個問題,也方便給別的小伙伴一些參考,

若該文章幫助到了你,請幫忙點一下贊好嗎,若有更好的方案,可以評論告訴我,讓我也學習一下。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本章節主要討論瞭如何通過零拷貝技術來優化文件傳輸的性能。零拷貝技術主要通過減少用戶態和內核態之間的上下文切換次數和數據拷貝次數來提高性能。具體來說,介紹了兩種實現零拷貝的方式:mmap + write和sendfile。使用mmap + write可以減少一次數據拷貝過程,而使用sendfile系統... ...
  • 第1個虛擬項目 1. 前言 點燈開啟了我們的FPGA之路,那麼我們來繼續沙盤演練。 用一個虛擬項目,來入門練習,以此步入數字邏輯的大門。 Key Words:FIFO 、SOF 、EOF、計數器、緩存、時序圖、方案設計 2. 項目要求 1) 輸入報文長度64~2048位元組; 2) 輸入報文之間最小間 ...
  • 本文將從 e2e 的基本介紹,e2e 的使用與擴展,session 日誌隔離三個維度為大家帶來 ChunJun e2e & session 日誌隔離的分享。 大量具體代碼和演示請看視頻教程⬇️ 視頻課程: https://www.bilibili.com/video/BV1ru411P7oZ/?sp ...
  • 一、升級路線 10.2.0.5,11.1.0.7,11.2.0.2以上版本可以直接升級到12c。 10.2.0.5以前的版本和11.2.0.1版需要先升級到中間版本,再升級到12c。 二、環境說明 操作系統:Red Hat 8 Linux 64位 源資料庫版本:Oracle 11.2.0.3 目標數 ...
  • NineData SQL開發企業版是一款強大的資料庫管理工具,旨在解決企業在資料庫管理方面所面臨的各種挑戰。該工具提供了一整套企業級能力,包括團隊協作、許可權管控、審計日誌、SSO單點登錄等功能。它還提供超過100條SQL規範審核,幫助企業規範使用和變更資料庫。通過該工具,用戶可以簡化繁瑣的資料庫管理... ...
  • 簡述 SAP HANA 是由 SAP 開發的一款記憶體列式資料庫, 具有預測分析、空間數據處理、文本分析、文本搜索、流分析、圖形數據處理等高級分析功能。 HANA 記憶體列式資料庫特性,即啟動後可以把所有數據載入記憶體,相比傳統基於硬碟的資料庫,性能提升10~10,000倍。 HANA 一般內置在 SAP ...
  • 首先,先搞明白一個概念,這裡的 Java 混合 Kotlin 是指文件層級的混合,即 Java 代碼還是寫在 .java 文件中,Kotlin 代碼還是寫在 .kt 文件中,只不過是可以在 Java 的代碼中可以調用自己寫好的 Kotlin 類,從 Java 的角度看,它並不知道它調用的這個類是 K ...
  • 目錄前言一、監聽UITextField 內容變化1. 代理2. 通知3. 目標動作事件二、監聽UITextView文本內容高度變化三、cell中的文本框,在reloadData or reloadRow後失去焦點四、手機號碼、銀行卡號格式化 前言 本文總結了在使用 UITextField & UIT ...
一周排行
    -Advertisement-
    Play Games
  • 前言 微服務架構已經成為搭建高效、可擴展系統的關鍵技術之一,然而,現有許多微服務框架往往過於複雜,使得我們普通開發者難以快速上手並體驗到微服務帶了的便利。為瞭解決這一問題,於是作者精心打造了一款最接地氣的 .NET 微服務框架,幫助我們輕鬆構建和管理微服務應用。 本框架不僅支持 Consul 服務註 ...
  • 先看一下效果吧: 如果不會寫動畫或者懶得寫動畫,就直接交給Blend來做吧; 其實Blend操作起來很簡單,有點類似於在操作PS,我們只需要設置關鍵幀,滑鼠點來點去就可以了,Blend會自動幫我們生成我們想要的動畫效果. 第一步:要創建一個空的WPF項目 第二步:右鍵我們的項目,在最下方有一個,在B ...
  • Prism:框架介紹與安裝 什麼是Prism? Prism是一個用於在 WPF、Xamarin Form、Uno 平臺和 WinUI 中構建鬆散耦合、可維護和可測試的 XAML 應用程式框架 Github https://github.com/PrismLibrary/Prism NuGet htt ...
  • 在WPF中,屏幕上的所有內容,都是通過畫筆(Brush)畫上去的。如按鈕的背景色,邊框,文本框的前景和形狀填充。藉助畫筆,可以繪製頁面上的所有UI對象。不同畫筆具有不同類型的輸出( 如:某些畫筆使用純色繪製區域,其他畫筆使用漸變、圖案、圖像或繪圖)。 ...
  • 前言 嗨,大家好!推薦一個基於 .NET 8 的高併發微服務電商系統,涵蓋了商品、訂單、會員、服務、財務等50多種實用功能。 項目不僅使用了 .NET 8 的最新特性,還集成了AutoFac、DotLiquid、HangFire、Nlog、Jwt、LayUIAdmin、SqlSugar、MySQL、 ...
  • 本文主要介紹攝像頭(相機)如何採集數據,用於類似攝像頭本地顯示軟體,以及流媒體數據傳輸場景如傳屏、視訊會議等。 攝像頭採集有多種方案,如AForge.NET、WPFMediaKit、OpenCvSharp、EmguCv、DirectShow.NET、MediaCaptre(UWP),網上一些文章以及 ...
  • 前言 Seal-Report 是一款.NET 開源報表工具,擁有 1.4K Star。它提供了一個完整的框架,使用 C# 編寫,最新的版本採用的是 .NET 8.0 。 它能夠高效地從各種資料庫或 NoSQL 數據源生成日常報表,並支持執行複雜的報表任務。 其簡單易用的安裝過程和直觀的設計界面,我們 ...
  • 背景需求: 系統需要對接到XXX官方的API,但因此官方對接以及管理都十分嚴格。而本人部門的系統中包含諸多子系統,系統間為了穩定,程式間多數固定Token+特殊驗證進行調用,且後期還要提供給其他兄弟部門系統共同調用。 原則上:每套系統都必須單獨接入到官方,但官方的接入複雜,還要官方指定機構認證的證書 ...
  • 本文介紹下電腦設備關機的情況下如何通過網路喚醒設備,之前電源S狀態 電腦Power電源狀態- 唐宋元明清2188 - 博客園 (cnblogs.com) 有介紹過遠程喚醒設備,後面這倆天瞭解多了點所以單獨加個隨筆 設備關機的情況下,使用網路喚醒的前提條件: 1. 被喚醒設備需要支持這WakeOnL ...
  • 前言 大家好,推薦一個.NET 8.0 為核心,結合前端 Vue 框架,實現了前後端完全分離的設計理念。它不僅提供了強大的基礎功能支持,如許可權管理、代碼生成器等,還通過採用主流技術和最佳實踐,顯著降低了開發難度,加快了項目交付速度。 如果你需要一個高效的開發解決方案,本框架能幫助大家輕鬆應對挑戰,實 ...