從WTM vue2版到 WtmPlus vue3


從2019年開始注意到WTM,覺得代碼生成器真是能省不少事,大部分的項目都缺不了系統管理(包含用戶,用戶組,角色,角色權限,數據權限),這個通用功能WTM項目已經集成,開發者可以把主要精力投入到業務開發。對於幾個人做的小型項目,想要MVC模式開發,在這個基礎上進行開發,風格就能達到統一;

到2020年,vue版本發布了,想着現在前后端分離是前端開發熱門,就嘗試了一下,我談談我的粗淺認識。

對這個Vue版本,WTM群友的第一感覺就是前端寫出了后端的感覺,能把這個WTM-Vue版本吃透,Vue從基礎到高級用法差不多了解得差不多了。

對於初次接觸的人,Vue 的 Mixins(混入)比較費解,但要了解WTM vue2 代碼生成器生成的每個放在ClientApp/src/pages目錄下的每個窗體頁面,了解數據從webapi獲取,在前端頁面上面顯示,然后響應按鈕事件,Vue頁面的生命周期,要細細,反反復復的看form-mixin.ts,action-minxin.ts,search.ts 這幾個文件,要不都弄不清 store目錄下的api.ts 里面寫的

import { contentType } from "@/config/enum";
const reqPath = config.headerApi + "/_actionlog/";

// 列表
const search = {
  url: reqPath + "Search",
  method: "post",
  dataType: "array"
};
// 批量刪除
const batchDelete = {
  url: reqPath + "BatchDelete",
  method: "post"
};
// 詳情
const detail = {
  url: reqPath + "{ID}",
  method: "get"
};
const exportExcel = {
  url: reqPath + "ExportExcel",
  method: "post",
  contentType: contentType.stream
};
const exportExcelByIds = {
  url: reqPath + "ExportExcelByIds",
  method: "post",
  contentType: contentType.stream
};
const getExcelTemplate = {
  url: reqPath + "GetExcelTemplate",
  method: "get",
  contentType: contentType.stream
};

// 導入
const imported = {
  url: reqPath + "Import",
  method: "post"
};
export default {
  search,
  batchDelete,
  detail,
  exportExcel,
  exportExcelByIds,
  getExcelTemplate,
  imported
};

 

 這些是怎么被調用。

另外就是 ClientApp/src/components/ 下面的文件,這個可以

<template>
    <wtm-dialog-box :is-show.sync="isShow" :status="status" :events="formEvent">
        <wtm-create-form :ref="refName" :status="status" :options="formOptions">
        </wtm-create-form>
    </wtm-dialog-box>
</template>

<script lang="ts">
import { Component, Vue } from "vue-property-decorator";
import { Action, State } from "vuex-class";
import formMixin from "@/vue-custom/mixin/form-mixin";

@Component({ mixins: [formMixin()] })
export default class Index extends Vue {
    // 表單結構
    get formOptions() {
        return {
            formProps: {
                "label-width": this.$t("actionlog.LabelWidthForm")
            },
            formItem: {
                "Entity.ID": {
                    isHidden: true
                },
                "Entity.ModuleName": {
                    type: "input",
                    label: this.$t("actionlog.ModuleName")
                },
                "Entity.ActionName": {
                    type: "input",
                    label: this.$t("actionlog.ActionName")
                },
                "Entity.ITCode": {
                    type: "input",
                    label: "ITCode"
                },
                "Entity.ActionUrl": {
                    type: "input",
                    label: "Url"
                },
                "Entity.ActionTime": {
                    type: "input",
                    label: this.$t("actionlog.ActionTime")
                },
                "Entity.Duration": {
                    type: "input",
                    label: this.$t("actionlog.Duration")
                },
                "Entity.IP": {
                    type: "input",
                    label: "IP"
                },
                "Entity.Remark": {
                    type: "input",
                    label: this.$t("actionlog.Remark")
                }
            }
        };
    }
}
</script>

把表單結構轉換成element組件,注意分析下 ClientApp/src/components/page/CreateForm/utils.tsx 文件,里面的generateWtmFormItemComponent 值得好好分析;

生成普通的頁面,例如下圖頁面就很容易,

但是generateWtmFormItemComponent 生成復雜頁面就力不從心,

 

要怎么把element組件集成到wtm-create-form里面 ,

 

   <wtm-create-form :ref="refName" :status="status" :options="formOptions">
   </wtm-create-form>

我自己是修改 ClientApp/Src/Components/page/CreateForm/index.ts

render(h) {
    const components = _.keys(this.options.formItem).map((key) => {
      const createItem=this.options.formProps["createItem"];
     
      const item = this.options.formItem[key];
      if (_.isFunction(item.isHidden)) {
        if (item.isHidden(this.getFormData(), this.status)) {
          return;
        }
      }
      if ((_.isBoolean(item.isHidden) && item.isHidden) || !item.type) {
        return;
      }
      const itemComp = componentObj[item.type];
        const option = { ...item, key };
        if (createItem == false) {            
            return;
        }
      const contentComp = itemComp ? itemComp.call(this, h, option) : null;

      return componentObj.wtmFormItem.call(this, h, option, contentComp);
    });
    const props = {
      ...this.options.formProps,
      disabled: this.status === "detail",
      model: this.sourceFormData || this.formData,
    };
    const slots = this.$scopedSlots["default"];
    if(props["createItem"]==false){
       return  (      
      <el-form ref={this.elFormRefKey} {...{ props }}>      
         <el-row class={this.elRowClass}>
           {slots && slots({})}
         </el-row> 
      </el-form>
     );
    }    
    if(props["tabs"]){       
       const tbpanes=props["tabs"].panes.map((key)=>{
         const tbpanes_Components=components.slice(Number(key.start),Number(key.end)+1);    
         return (  <el-tab-pane label={key.label} name={key.name}>
                      {tbpanes_Components}
                    </el-tab-pane>
         );
         });       
       return (      
      <el-form ref={this.elFormRefKey} {...{ props }}>
         <el-row class={this.elRowClass}>
           <el-tabs v-model={props["tabs"].activeName} type="border-card">
               {tbpanes}                       
           </el-tabs>
              {slots && slots({})}
           </el-row>
      </el-form>
      );
    }
    else{
    return (      
      <el-form ref={this.elFormRefKey} {...{ props }}>
                           <el-row class={this.elRowClass}>
                             {components}
                              {slots && slots({})}
                              </el-row>

      </el-form>
     );
    }
  

其中一個用法是根據組件順序,把指定范圍的組件放到 一個 el-tab-pane,但是用到這種的情況比較少,另一個就是

 

<template>
    <wtm-dialog-box :is-show.sync="isShow" :status="status" :events="formEvent">
        <wtm-create-form :ref="refName" :status="status" :options="formOptions" :sourceFormData="mergeFormData">       
                <el-form-item prop="Entity.Name" :rules="formOptions.formItem['Entity.Name'].rules">
                    <el-input  v-model="mergeFormData.Entity.Name" placeholder="請輸入庫名" @blur="blur"></el-input>
                </el-form-item>
            <el-form-item>
                <el-input v-model="mergeFormData.Entity.Address" placeholder="請輸入地址" @blur="blur"></el-input>
            </el-form-item>
        </wtm-create-form>
    </wtm-dialog-box>
</template>

<script lang="ts">
import { Component, Vue } from "vue-property-decorator";
import { Action, State } from "vuex-class";
import formMixin from "@/vue-custom/mixin/form-mixin";
import UploadImg from "@/components/page/UploadImg.vue";
    //:sourceFormData="mergeFormData"
@Component({
    mixins: [formMixin()]
})
export default class Index extends Vue {

    mergeFormData ={
       Entity:{
        "ID":0,
        Name:"Name",
        Address:"Address",
      }
    }
    

    // 表單結構
    get formOptions() {
        const filterMethod = (query, item) => {
            return item.label.indexOf(query) > -1;
        };
        return {
            formProps: {
                "label-width": "100px",
                "createItem":false,
            },
            formItem: {
                "Entity.ID": {
                    isHidden: true
                },
             "Entity.Name":{
                 label: "Name",
                 rules: [{ required: true, message:"庫名不能為空",trigger: "blur" }],
                 type: "input",
                 modifier:"trim",//v-model.trim v-model.number,v-model.lazy
                 slotKey:"SlotName"
            },
             "Entity.Address":{
                 label: "Address",
                 rules: [{ required: true, message: "地址"+this.$t("form.notnull"),trigger: "blur" }],
                 type: "input"
            }
                
            }
        };
    }
     beforeRequest(formData?: object): object | void {
      console.log("beforeRequest:", formData);
      return formData;
    }

    blur(){
      console.log("this.form",this.$refs[this.refName]);
      console.log("blur",this.mergeFormData);      
      this.$refs[this.refName].setFormDataItem("Entity.Name",this.mergeFormData.Entity.Name);
      this.$refs[this.refName].validate((valid) => {
          if (valid) {          
          } else {            
            return false;
          }
        });
    }
}
</script>

當 "createItem":false 時,在 wtm-create-form 里面自己綁定組件;總之Vu2版本要寫項目,還是要進行各種改造,不像 Layui版本那種綁定風格,自由度感覺受到限制;

有看過 WTM react 版本,react版本跟layUI版本的綁定比較類似,自己把實體綁定到每個組件上,后來聽說WTM vue版本要進行WTM VU3重制,我不知道了解信息有沒有錯誤,是WTM react操刀人要進行WTM vue3版本重制,我就想風格應該會跟React類似;

千呼萬喚,2021.11.18 WTMPlus Vu3版本正式發布,聽說劉總為了發布都熬通宵了;

第一時間生成幾個Model試了一下,看看有啥變化。

 整個目錄結構都變了,UI也由Vue element 換成了 Vue antd,看綁定,是不是跟 layUI版本有點類似了。

<template>
  <WtmDetails :queryKey="queryKey" :loading="Entities.loading" :onFinish="onFinish">
    <a-row :gutter="6">
      <a-col :span="12">
         <WtmField entityKey="NameAdd"/>
      </a-col>
      <a-col :span="12">
         <WtmField entityKey="Id_NoAdd"/>
      </a-col>
      <a-col :span="12">
         <WtmField entityKey="Day_of_birthAdd"/>
      </a-col>
    </a-row>

      <div style="text-align:right;">


      </div>

  <template #button>
  
   <a-divider type="vertical" />
   <a-button type="primary" html-type="submit">
       <template v-slot:icon>
           <SaveOutlined style='margin-right:5px'/>
       </template>
       <i18n-t :keypath="$locales.SysSubmit" />
   </a-button>
   <a-divider type="vertical" />
   <a-button type="primary" @click.stop.prevent="__wtmBackDetails()">
       <template v-slot:icon>
           <RedoOutlined style='margin-right:5px'/>
       </template>
       <i18n-t :keypath="$locales.SysClose" />
   </a-button>
  </template>
  </WtmDetails>
</template>

 

 還剛接觸Vue3版本的,等認識更深刻后再發表看法;

最后誇一誇 WTMPlus,擁有多個微軟MVP的團隊傾力打造的低碼平台,而且項目源碼全部可以下載,開發者可以進行自由發揮,對於一個程序員,源碼在手,天下我有的感覺是不一樣的。

網址:http://wtmplus.walkingtec.cn

 

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM