vue3實現模擬地圖上,站點名稱按需顯示的功能

来源:https://www.cnblogs.com/jiekzou/p/18260813
-Advertisement-
Play Games

很久很久沒有更新博客了,因為實在是太忙了,每天都有公司的事情忙不完....... 最近在做車輛模擬地圖,在實現控制站點名稱按需顯示時,折騰了好一段時間,特此記錄一下。最終界面如下圖所示: 站點顯示需求:首末站必須顯示,從第一個站開始,如果站點名稱能顯示下,則顯示,如果站點名稱會重疊則隱藏,以此類推。 ...


很久很久沒有更新博客了,因為實在是太忙了,每天都有公司的事情忙不完.......

最近在做車輛模擬地圖,在實現控制站點名稱按需顯示時,折騰了好一段時間,特此記錄一下。最終界面如下圖所示:

站點顯示需求:首末站必須顯示,從第一個站開始,如果站點名稱能顯示下,則顯示,如果站點名稱會重疊則隱藏,以此類推。當界面寬度變化時,車輛模擬地圖自動變化,保證站點名稱能夠最大限度的顯示。

最開始我用的比例換演算法,演算法複雜度是O,結果總是不准。儘管一開始我就覺得演算法的複雜度應該是O2。我之前卻一直想著只遍歷一次就算出來,我也嘗試過把需求描述得很詳細去問chatgpt,可是它就像個傻子一樣輸出各種演算法錯誤代碼。

需要註意的地方:由於站點的名稱內容是千奇百怪的,可以有空格,各種特殊圖標,所以站點文字的長度計算是一個問題,這裡是通過canvas來計算的。還有,這裡我添加了一個限制,站點文字內容我最大顯示120px,超出隱藏並顯示省略號,站點名稱上添加了title顯示全稱。

/**
 * 獲取站點名稱
 * @param item 
 * @param showFullName 是否總是顯示站點全名
 */
/** */
export const getSiteName = (item: any,showFullName?:boolean=false) => {
  const { siteSign } = simulatedMapConf.value;
  let name = '';
  if (siteSign == 'firstWord') {
    name = getSubStrByPreNum(item.stationName);
  } else if (siteSign == 'order') {
    name = item.stationSeq + '';
  } else {
    name=showFullName?item.stationName:(item.show? item.stationName:''); //show控制站點名稱是否顯示
  }
  return name || '';
}
/**
 * 獲取站點名稱寬度
 * @param item 站點對象
 * @param showFullName 是否總是顯示站點全名
 * @returns 
 */
export const getSiteNameWidth = (item: any,showFullName?:boolean=false) => {
  const name =showFullName?item.stationName: getSiteName(item,showFullName);
  const width= calculateStringWidth(name);
  return width>siteMaxWidth?siteMaxWidth:width;
}
/**
 * 根據字元串計算出界面渲染的寬度
 * @param str 
 * @returns 
 */
function calculateStringWidth(str:string) {
  // 創建一個虛擬的 <canvas> 元素
  const canvas = document.createElement('canvas');
  const ctx = canvas.getContext('2d');
  // 設置字體樣式
  ctx.font = '12px sans-serif';
  // 使用 canvas 的 measureText 方法測量字元串的寬度
  const width = ctx.measureText(str).width;
  // 返回計算出的寬度
  return width;
}

最核心的演算法:

    //計算上行站點,控制站點是否顯示在模擬地圖上
    function calcSite(station: any, lineWidth: number) {
        if (station.length < 1) return [];
        station.forEach((f: any, index) => {
            f.show=false;
        });
        const lastSiteLength = getSiteNameWidth(station[station.length - 1], true) / 2;//站點文字半寬
        let lastLeft = getSiteCx(station[station.length - 1], station.length - 1);//最後一個站點left
        lastLeft = toDecimal(lastLeft - lastSiteLength);
        station.forEach((f: any, index) => {
            let siteLength = getSiteNameWidth(f, true);//站點文字寬度
            let bigHalf = siteLength / 2;//獲取當前的半寬
            f.left = getSiteCx(f, index); 
            if (index == 0 || index == station.length - 1) { //第一項和最後一項必須顯示
                f.show = true;
            } else {
                const preShowIndex = getLastTrueIndex(station); //獲取前面最近一個顯示站點的索引
                const preEndLeft = toDecimal(station[preShowIndex].left + getSiteNameWidth(station[preShowIndex], true) / 2);//上一項顯示的站點名稱結束left位置
                f.show = toDecimal(f.left - bigHalf) >=preEndLeft && preEndLeft < lastLeft; //如果上一個顯示站點文字的結尾位置 小於等於 當前站點文字的開始位置  並且小於最後一個站點文字的開始位置
if (f.show && toDecimal(f.left + bigHalf) > lastLeft) {
f.show = false;
}
}
        })
    }

獲取前面最近一個顯示站點的索引:

    //獲取list集合中最後一項show的index位置
    function getLastTrueIndex(dataList: any) {
        // 從數組末尾第2項開始向前遍歷
        for (let i = dataList.length - 2; i >= 0; i--) {
            if (dataList[i].show === true) {
                return i;  // 返回第一個找到的最後一個為true的索引
            }
        }
        return -1;  // 如果未找到符合條件的對象,返回-1
    }

下行站點的計算有些差別,因為下行站點是從右至左,所以left基本上是反著的:

    //計算下行站點,控制站點是否顯示在模擬地圖上 getDownSiteCx
    function calcDownSite(station: any, lineWidth: number) {
        if (station.length < 1) return [];
        station.forEach((f: any, index) => {
            f.show=false;
        });
        const lastSiteLength = getSiteNameWidth(station[station.length - 1], true) / 2;//站點文字半寬
        let lastLeft = getDownSiteCx(station[station.length - 1], station.length - 1);//最後一個站點left
        lastLeft = toDecimal(lastLeft + lastSiteLength);
        station.forEach((f: any, index) => {
            let siteLength = getSiteNameWidth(f, true);//站點文字寬度
            let bigHalf = siteLength / 2;//獲取當前的半寬
            f.left = getDownSiteCx(f, index); 
            if (index == 0 || index == station.length - 1) { //第一項和最後一項必須顯示
                f.show = true;
            } else {
                const preShowIndex = getLastTrueIndex(station); //獲取前面最近一個顯示站點的索引
                const preEndLeft = toDecimal(station[preShowIndex].left - getSiteNameWidth(station[preShowIndex], true) / 2);//上一項顯示站的的結束left位置
                f.show = toDecimal(f.left + bigHalf) <=preEndLeft && preEndLeft > lastLeft;
                if (f.show && toDecimal(f.left - bigHalf) < lastLeft) {
                    f.show = false;
                }
            }
        })
    }

另外獲取站點中心點位置的方法

    //獲取上行站點水平x位置
    const getSiteCx = (item: any, index: number) => {
        return startleft.value + dLayout.lineWidth * index;
    }
    //獲取下行站點水平x位置
    const getDownSiteCx = (item: any, index: number) => {
        return downStartleft.value - layout.endLine - dLayout.downLineWidth * index;
    }

說明:站點的佈局採用css絕對定位。第一個版本這塊我是採用的svg畫的,後來發現擴展起來越來越麻煩,周末就在家花了半天時間全部改造為html實現了。

我最開始的有問題代碼是上下行站點共用的,最大的問題是會出現跳站點顯示的情況,代碼如下的:

    //計算站點,控制站點是否顯示在模擬地圖上
    function calcSite(station: any, lineWidth: number) {
        let availableWidth = (station.length - 1) * lineWidth; //總長度
        //記錄顯示站點的長度
        let totalLength = 0;
        station.forEach((f: any, index) => {
            let siteLength = getSiteNameWidth(f, true);
            let bigHalf =siteLength / 2;//獲取比較大的半寬
            let bigHalfPre = 0;
            //計算上一項的文字半寬
            if (index >= 1) {
                let siteLengthPre = getSiteNameWidth(station[index - 1], true);
                bigHalfPre =siteLengthPre / 2;
            }
            f.left = toDecimal(lineWidth * index);
            f.show =index==0?true: f.left >=toDecimal(totalLength);
            if(index >= 1&&station[index-1].show&&bigHalf+bigHalfPre>lineWidth){
                f.show=false;
            }
            if (f.show) {
                let times = getDivisor(siteLength, lineWidth);
                totalLength += times * lineWidth;
            }
        })
    }
/**
 * 兩個數相除有餘數時結果加1
 * @param all 被除數 站點寬度
 * @param num 除數  線寬
 * @returns 
 */
export const getDivisor=( all:number,item:number)=>{     if(all<=item) return 1;     let diff:number=0;     if(item<=20){         diff=2.5;     }     if(item<=30){         diff=2;     }     else if(item<=40){         diff=1.5;     }     else if(item<=46){         diff=1.05;     }     else if(item<=50){         diff=1;     }     return all%item==0?(all/item):(Math.ceil(all/item)+diff); }

完!

博客地址:http://www.cnblogs.com/jiekzou/
博客版權:本文以學習、研究和分享為主,歡迎轉載,但必須在文章頁面明顯位置給出原文連接。
如果文中有不妥或者錯誤的地方還望高手的你指出,以免誤人子弟。如果覺得本文對你有所幫助不如【推薦】一下!如果你有更好的建議,不如留言一起討論,共同進步!
再次感謝您耐心的讀完本篇文章。
其它: .net-QQ群4:612347965 java-QQ群:805741535 H5-QQ群:773766020
我的拙作 《Vue3.x+TypeScript實踐指南》 《ASP.NET MVC企業級實戰》 《H5+移動應用實戰開發》 《Vue.js 2.x實踐指南》 《JavaScript實用教程 》 《Node+MongoDB+React 項目實戰開發》 已經出版,希望大家多多支持!


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

-Advertisement-
Play Games
更多相關文章
  • ‍ 寫在開頭 點贊 + 收藏 學會 如題,慣性思路很簡單,就是直接擼上一個空內容的html。 註:以下都是在現代瀏覽器中執行,主要為**Chrome 版本 120.0.6099.217(正式版本) (64 位)和Firefox123.0.1 (64 位) ** <!DOCTYPE ...
  • 本文概述了Nuxt 3框架的升級特點,對比Nuxt 2,詳細解析中間件應用、配置策略與實戰示例,涵蓋功能、錯誤管理、優化技巧,並探討與Nuxt 3核心組件集成方法,給出最佳實踐和問題解決方案,強調利用Vue 3和Serverless Functions提升中間件效能。 ...
  • 前言 v-bind指令想必大家都不陌生,並且都知道他支持各種寫法,比如<div v-bind:title="title">、<div :title="title">、<div :title>(vue3.4中引入的新的寫法)。這三種寫法的作用都是一樣的,將title變數綁定到div標簽的title屬性 ...
  • RFC規範並沒有指明HTTP協議的GET方法是否不能攜帶body數據,但是對於瀏覽器環境下的XHR和fetch API的規範來說,它們的規範限制了它們不能在GET中攜帶body,而postman或apifox等介面測試工具則由於不遵循這些規範而可以攜帶body。 ...
  • 摘要:“本文深入探討了Nuxt3 Composables,重點介紹了其目錄架構和內置API的高效應用。通過學習本文,讀者將能夠更好地理解和利用Nuxt3 Composables來構建高效的應用程式。” ...
  • 依賴分類 依賴根據開發環境需要和實際運行環境需要,分為dependencies和devDependencies。 例如:typescript和eslint屬於devDependencies,而vue和axios等屬於dependencies。 版本號組成 版本由兩部分組成,一是前面的首碼符號,二是版 ...
  • 摘要:該文指南詳述了Nuxt 3的概況與安裝,聚焦於在Nuxt 3框架下運用Vuex進行高效的狀態管理,涵蓋基礎配置、模塊化實踐至高階策略,助力開發者構建高性能前後端分離應用。 ...
  • ‍ 寫在開頭 點贊 + 收藏 學會 如何解決uniapp H5本地代理實現跨域訪問? 1.第一種解決方法: 直接創建一個vue.config.js文件,併在裡面配置devServer,直接上代碼,重啟跑項目 親測有效 // vue.config.js module.exports ...
一周排行
    -Advertisement-
    Play Games
  • 前言 推薦一款基於.NET 8、WPF、Prism.DryIoc、MVVM設計模式、Blazor以及MySQL資料庫構建的企業級工作流系統的WPF客戶端框架-AIStudio.Wpf.AClient 6.0。 項目介紹 框架採用了 Prism 框架來實現 MVVM 模式,不僅簡化了 MVVM 的典型 ...
  • 先看一下效果吧: 我們直接通過改造一下原版的TreeView來實現上面這個效果 我們先創建一個普通的TreeView 代碼很簡單: <TreeView> <TreeViewItem Header="人事部"/> <TreeViewItem Header="技術部"> <TreeViewItem He ...
  • 1. 生成式 AI 簡介 https://imp.i384100.net/LXYmq3 2. Python 語言 https://imp.i384100.net/5gmXXo 3. 統計和 R https://youtu.be/ANMuuq502rE?si=hw9GT6JVzMhRvBbF 4. 數 ...
  • 本文為大家介紹下.NET解壓/壓縮zip文件。雖然解壓縮不是啥核心技術,但壓縮性能以及進度處理還是需要關註下,針對使用較多的zip開源組件驗證,給大家提供個技術選型參考 之前在《.NET WebSocket高併發通信阻塞問題 - 唐宋元明清2188 - 博客園 (cnblogs.com)》講過,團隊 ...
  • 之前寫過兩篇關於Roslyn源生成器生成源代碼的用例,今天使用Roslyn的代碼修複器CodeFixProvider實現一個cs文件頭部註釋的功能, 代碼修複器會同時涉及到CodeFixProvider和DiagnosticAnalyzer, 實現FileHeaderAnalyzer 首先我們知道修 ...
  • 在軟體行業,經常會聽到一句話“文不如表,表不如圖”說明瞭圖形在軟體應用中的重要性。同樣在WPF開發中,為了程式美觀或者業務需要,經常會用到各種個樣的圖形。今天以一些簡單的小例子,簡述WPF開發中幾何圖形(Geometry)相關內容,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 在 C# 中使用 RabbitMQ 通過簡訊發送重置後的密碼到用戶的手機號上,你可以按照以下步驟進行 1.安裝 RabbitMQ 客戶端庫 首先,確保你已經安裝了 RabbitMQ 客戶端庫。你可以通過 NuGet 包管理器來安裝: dotnet add package RabbitMQ.Clien ...
  • 1.下載 Protocol Buffers 編譯器(protoc) 前往 Protocol Buffers GitHub Releases 頁面。在 "Assets" 下找到適合您系統的壓縮文件,通常為 protoc-{version}-win32.zip 或 protoc-{version}-wi ...
  • 簡介 在現代微服務架構中,服務發現(Service Discovery)是一項關鍵功能。它允許微服務動態地找到彼此,而無需依賴硬編碼的地址。以前如果你搜 .NET Service Discovery,大概率會搜到一大堆 Eureka,Consul 等的文章。現在微軟為我們帶來了一個官方的包:Micr ...
  • ZY樹洞 前言 ZY樹洞是一個基於.NET Core開發的簡單的評論系統,主要用於大家分享自己心中的感悟、經驗、心得、想法等。 好了,不賣關子了,這個項目其實是上班無聊的時候寫的,為什麼要寫這個項目呢?因為我單純的想吐槽一下工作中的不滿而已。 項目介紹 項目很簡單,主要功能就是提供一個簡單的評論系統 ...