Vue動態(tài)構(gòu)建混合數(shù)據(jù)Treeselect選擇樹及巨樹問題的解決
今天在項(xiàng)目中需要通過行政區(qū)域選擇,然后選擇該行政區(qū)域下面的景區(qū),也就是要構(gòu)建行政區(qū)劃、景區(qū)兩表數(shù)據(jù)表的樹。
全國的行政區(qū)域到縣已經(jīng)3500多了,再加上景區(qū)會有幾萬個(gè)點(diǎn),這棵選擇樹不論是在后臺還是在前臺構(gòu)建都比較大,會影響系統(tǒng)性能,需要前后端結(jié)合,動態(tài)構(gòu)建,使用時(shí)用懶加載,提升系統(tǒng)性能。
一、后臺構(gòu)建兩個(gè)表的數(shù)據(jù)選擇查詢功能
1、行政區(qū)域需要按省、市、縣的adcode和級別查詢,以便于動態(tài)構(gòu)建。
兩個(gè)參數(shù),adcode后臺查詢時(shí)會將后面的0去掉匹配,級別如果0或空,查詢匹配下所有的,為數(shù)字只查本級。
這里有兩個(gè)參數(shù),用@requestParam傳遞的,注意要用post,前端VUE使用get傳遞時(shí)會報(bào)錯(cuò)。
? ? @PostMapping("/listjson" ) ? ? @ApiOperation("根據(jù)查詢條件獲取區(qū)劃清單") ? ? public AjaxResult listJson(@RequestParam(name="adcode", required = true) String adcode, @RequestParam(name="level",required = true) String level) ? ? { ? ? ? ? System.out.println(adcode); ? ? ? ? List<Map<String, String>> ?regionJson = iMapRegionService.selectRegionJson(adcode,level); ? ? ? ? return AjaxResult.success(regionJson); ? ? } }
2、查詢行政區(qū)域下的景區(qū),根據(jù)adcode查詢,一個(gè)參數(shù),參數(shù)放在路徑中傳遞的。
? ? @GetMapping("/listByCode/{adcode}") ? ? @ApiOperation("根據(jù)查詢條件獲取景區(qū)清單") ? ? public AjaxResult listByCode(@PathVariable String adcode) ? ? { ? ? ? ? List<Map<String, String>> scenicList = iMapScenicService.selectScenicByCode(adcode); ? ? ? ? return AjaxResult.success(scenicList); ? ? } }
二、在VUE中引入Treeselect
由于后面要用懶加載,在引用Treeselect給件同時(shí),還要引入LOAD_CHILDREN_OPTIONS,Treeselect要加到components中。
? import { LOAD_CHILDREN_OPTIONS, Treeselect } from '@riophae/vue-treeselect' ? import '@riophae/vue-treeselect/dist/vue-treeselect.css' ? export default { ? ? components: { Treeselect }, ? ? data() { ? ? ? return { ? ? ? ? //地點(diǎn)列表 ? ? ? ? regionOptions: [],
三、使用Treeselect組件
noChildrenText=“更新中…” , //由于缺省會將沒有加載下級節(jié)點(diǎn)的children設(shè)置為null,系統(tǒng)缺省會顯示No sub-options,點(diǎn)擊節(jié)點(diǎn)是,顯示更新中更好,更新完成顯示子節(jié)點(diǎn)。 :load-options=“l(fā)oadOptions” //增加此選項(xiàng),在后方擴(kuò)展子節(jié)點(diǎn)是會調(diào)用此方法,第一是點(diǎn)擊左邊的三解形,第二下面將非葉子節(jié)點(diǎn)設(shè)置為不能選擇,點(diǎn)擊節(jié)點(diǎn)也會自動擴(kuò)展。 :disable-branch-nodes=“true”> //將樹枝節(jié)點(diǎn)設(shè)置為不能選擇,樹枝節(jié)點(diǎn)是行政區(qū)域,也不我們需要的景區(qū)ID,所以不能選擇,否則數(shù)據(jù)會亂。
<el-form-item label="景區(qū)名稱" prop="scenicId"> ? <treeselect ? ? v-model="form.scenicId" ? ? :options="regionOptions" ? ? noChildrenText="更新中..." ? ? ? :load-options="loadOptions" ? ? placeholder="請選擇景區(qū)" ? ? :disable-branch-nodes="true"> ? </treeselect> </el-form-item>
四、構(gòu)建初始的省級目錄
組件打開初始化時(shí),構(gòu)建一級選擇樹
?created() { ? ? ? this.getList() ? ? ? this.getTreeselect() ? ? }, ? ? methods: { ? ? ? /** 生成查詢行政區(qū)劃下拉樹結(jié)構(gòu)第一級 */ ? ? ? getTreeselect() { ? ? ? ? let formData = new FormData() ? ?//構(gòu)建需要查詢的參數(shù),先選擇省節(jié)點(diǎn) ? ? ? ? formData.append('adcode', '000000') ? ? ? ? ? formData.append('level', '1') ? ? ? ? listJson(formData).then(response => { ?//listJson對應(yīng)是詢后臺行政區(qū)域的接口函數(shù) ? ? ? ? ? let data = response.data ? ? ? ? ? data.forEach(element => { ? ? ? ? ? ? element['id'] = element.code // 后臺傳遞過來的是code和name,需要換為treeselect所需要的id和label ? ? ? ? ? ? element['label'] = element.name ? ? ? ? ? ? element['level'] = '1' ? ?//設(shè)置為1級,后續(xù)判斷需要 ? ? ? ? ? ? element['children'] = null ?//要設(shè)置為空,才能觸發(fā):load-options="loadOptions"動用 ? ? ? ? ? }) ? ? ? ? ? this.regionOptions = data ? ? ? ? }) ? ? ? },
五、構(gòu)建動態(tài)生成的二、三、四級目錄
//動態(tài)添加樹的子節(jié)點(diǎn) loadOptions({ action, parentNode, callback }) { if (action === LOAD_CHILDREN_OPTIONS) { let formData = new FormData() formData.append('adcode', parentNode.id) switch (parentNode.level) { //判斷選擇級別 case '1' : /** 生成查詢行政區(qū)劃下拉樹結(jié)構(gòu)第二級 */ formData.append('level', '2') listJson(formData).then(res => { //選擇市,構(gòu)建2級 if (res.code === 200) { //后臺返回是用code表示返回狀態(tài)代碼,與后臺匹配 let data = res.data data.forEach(element => { element['id'] = element.code element['label'] = element.name element['level'] = '2' element['children'] = null }) parentNode.children = data //增加子節(jié)點(diǎn) } }) callback() break case '2' : /** 生成查詢行政區(qū)劃下拉樹結(jié)構(gòu)第三級 */ formData.append('level', '3') listJson(formData).then(res => { if (res.code === 200) { let data = res.data data.forEach(element => { element['id'] = element.code element['label'] = element.name element['level'] = '3' element['children'] = null }) parentNode.children = data } }) callback() break case '3' : /** 生成查詢行政區(qū)下的景區(qū)清單 */ listByCode(parentNode.id).then(res => { //第四級是景區(qū),使用的是景區(qū)表,根據(jù)adcode查詢 if (res.code === 200) { let data = res.data console.log(data) data.forEach(element => { //為景區(qū)增加一個(gè)圖標(biāo),以示區(qū)別 element['label'] = element['label'] + '' //使用emoji文件,標(biāo)識這是景區(qū),treeselect沒有icon圖標(biāo) element['level'] = '4' }) parentNode.children = data } }) callback() break } } }
六、最后效果
這個(gè)方法動態(tài)實(shí)現(xiàn)Treeselect構(gòu)建,可以解決巨樹構(gòu)建問題。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Vue中實(shí)現(xiàn)v-for循環(huán)遍歷圖片的方法
這篇文章主要介紹了Vue中實(shí)現(xiàn)v-for循環(huán)遍歷圖片的方法,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-08-08vue2使用ts?vue-class-component的過程
vue-property-decorator?是一個(gè)?Vue.js?的裝飾器庫,它提供了一些裝飾器來讓你在?Vue?組件中定義屬性、計(jì)算屬性、方法、事件等,本文給大家介紹vue2使用ts?vue-class-component的相關(guān)知識,感興趣的朋友一起看看吧2023-11-11vue2文件流下載成功后文件格式錯(cuò)誤、打不開及內(nèi)容缺失的解決方法
使用Vue時(shí)我們前端如何處理后端返回的文件流,下面這篇文章主要給大家介紹了關(guān)于vue2文件流下載成功后文件格式錯(cuò)誤、打不開及內(nèi)容缺失的解決方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-04-04一次用vue3簡單封裝table組件的實(shí)戰(zhàn)過程
之所以封裝全局組件是為了省事,所有的目的,全都是為了偷懶,下面這篇文章主要給大家介紹了關(guān)于用vue3簡單封裝table組件的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-12-12