小程式裡經常會碰到需要實現多行文字」全文收起「功能,在掘金上有搜尋到用純css實現。親測:ios很完美,andriod上的無效。
小程式社群有很多方案,目前在社群有看到一位大佬使用js動態計算告訴去實現,親測大致有效果,測試後,在一些特殊情況下,計算會有誤差,所以有更改些許程式碼。
主要用到用到 line-clamp,關鍵樣式如下
.text-clamp3 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 3; }
編寫兩段文字,一段展示完整的文字A,一段展示使用 line-clamp省略後的文字B,因為B有被擷取,因此B的高度相對較小。對比兩段文字的高度,便可以知道文字是否超出兩行
在小程式裡,可以使用wx.createSelectorQuery()
獲取文字高度
js
const query = wx.createSelectorQuery().in(this); query.selectAll(".showArea, .hideArea").boundingClientRect(res => { console.log(res, 'res') }).exec()
根據設計思路,立馬上手程式碼
foldable.wxml
<view class="content"> <view class="contentInner content-inner-class showArea {{!onFold ? 'text-clamp' + maxLine : ''}}">{{content}}</view> <view class="contentInner content-inner-class hideArea" style="width: {{width}}px">{{content}}</view> <view class="foldInner fold-class {{position === 'right' ? 'flex-end' : 'flex'}}" wx:if="{{showFold}}"> <text class="fold" catchtap="handleFold">{{onFold ? unFoldText : onFoldText}}</text> </view> </view>
foldable.js
/** * 長文字內容展開與收起 * @param {String} content 長文字內容 * @param {Number} maxLine 最多展示行數[只允許 1-5 的正整數] * @param {String} position 展開收起按鈕位置[可選值為 left right] * @param {Boolean} foldable 點選長文字是否展開收起 * @param { String } onFoldText 收縮時文字 * @param { String } unFoldText 展開時文字 * */ Component({ externalClasses: ['content-inner-class', 'fold-class'], properties: { content: { type: String, observer(val) { if (this.data.onReady) { this.getNodeClientReact() } } }, maxLine: { type: Number, value: 1, observer(value) { if (!(/^[1-5]$/).test(value)) { throw new Error(`maxLine field value can only be digits (1-5), Error value: ${value}`) } else if (this.data.onReady) { this.getNodeClientReact() } } }, position: { type: String, value: "left" }, foldable: { type: Boolean, value: true }, // 收縮時文字 onFoldText: { type: String, value: "全文" }, // 展開時文字 unFoldText: { type: String, value: "收起" }, }, data: { width: null, onFold: false, showFold: false, onReady: false }, lifetimes: { attached() { this.getNodeClientReact() this.setData({ onReady: true }) }, }, methods: { getNodeClientReact() { setTimeout(() => this.checkFold(), 10) }, checkFold() { const query = this.createSelectorQuery(); query.selectAll(".showArea, .hideArea").boundingClientRect(res => { let showFold = res[0].height < res[1].height; this.setData({ width: res[0].width, showFold, }) }).exec() }, handleFold() { this.setData({ onFold: !this.data.onFold }) } } })
foldable.wxss
.content { width: 100%; position: relative; overflow: hidden; } .contentInner { word-break: break-all; width: 100%; color: #2f3033; font-size: 30rpx; line-height: 1.35; } .hideArea { display: -webkit-box; overflow: hidden; position: fixed; top: 100vh; left: -100vw; } .foldInner { padding-top: 10rpx; color: #6676bd; font-size: 32rpx; } .foldInner .fold { cursor: pointer; } .text-clamp1 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 1; } .text-clamp2 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 2; } .text-clamp3 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 3; } .text-clamp4 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 4; } .text-clamp5 { overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 5; }
正常情況下,此方法可行,但是在級別文字下,會計算錯誤。經過測試,可將 節點是.hideArea
的內容定位在.showArea
節點下可解決
foldable.wxss
.hideArea { display: -webkit-box; overflow: hidden; /* position: fixed; top: 100vh; left: -100vw; */ position: absolute; top: 0; left: 0; z-index: -1; color: #fff; }
經過修復之後,本來是可以完美實現了,但是在測試過程中,第一次正常渲染是沒有問題。但如果文字資料更新,會發現如果原來的文字從一行增加到兩行時,使用wx.createSelectorQuery()
計算的高度會有存在是實際高低的兩倍的現象。導致會錯誤出現【全文】
文字。然後文字從兩行增加到三行或者多行都沒問題,不太理解為什麼會出現這個錯誤計算的現象。(期待大神能留言告知 ? )
為了彌補這個坑,我引入了lineHieght
這個屬性。
// foldable.js Component({ properties: { lineHieght: { type: Number, observer(value) { if (!(/^[0-9]*$/).test(value)) { throw new Error(`lineHieght field value can only be digits`) } } } } })
通過lineHieght
和最多可展示行數maxLine
可以計算出,可在介面展示的最大高度。
// 文字可見的最大高度 const maxHeight = this.data.lineHieght * this.data.maxLine;
當然了,我們也需要適配不同的裝置,而且通過wx.createSelectorQuery()
計算出來的結果是以px
為單位的。
所以,行高需要根據裝置尺寸去改變。因為我們是以寬度是750px
尺寸為設計稿的,所以根據wx.getSystemInfoSync()
可以獲取裝置資訊,進而轉換成px
的尺寸。
// foldable.js changeRpxToPx(rpxInteger) { return wx.getSystemInfoSync().windowWidth / 750 * rpxInteger },
因此,更新checkFold
方法
checkFold() { const query = this.createSelectorQuery(); query.selectAll(".showArea, .hideArea").boundingClientRect(res => { let showFold = res[0].height < res[1].height; const lineHeightToPx = this.changeRpxToPx(this.data.LineHeight); // 展示區域高度(即是可能會被擷取的可見文字) const showAreaHeight = res[0].height; // 隱藏區域的高度(即是完整文字高度,偶然事件會計算錯誤) const hideAreaHeight = res[1].height; // 文字可見的最大高度 const maxHeight = lineHeightToPx * this.data.maxLine; // 如果是一行文字,偶然計算錯誤,用行高判斷 if (this.data.LineHeight && showAreaHeight <= maxHeight) { showFold = hideAreaHeight > maxHeight } this.setData({ width: res[0].width, showFold, }) }).exec() },
經過上一個版本,基本功能都已經實現。但是,如果文字超過最大行數,並且在展開全文的情況下,更新了文字,此時,全文/展開
按鈕會展示錯誤。
通過分析程式碼可知,在展開全文的狀態下更新了文字,此時.showArea
節點和.hideArea
節點的高度一致,執行程式碼let showFold = res[0].height < res[1].height;
,會返回false
,因此按鈕會消失。
因此解決方案為:
// 如果文字超出最大行數,並且是顯示全文的狀態下,再次更新了文字 let onFold = false if (showAreaHeight == hideAreaHeight && showAreaHeight > maxHeight) { showFold = true onFold = true }
所以最終版本的checkFold
方法是:
checkFold() { const query = this.createSelectorQuery(); query.selectAll(".showArea, .hideArea").boundingClientRect(res => { let showFold = res[0].height < res[1].height; const lineHeightToPx = this.changeRpxToPx(this.data.LineHeight); // 展示區域高度(即是可能會被擷取的可見文字) const showAreaHeight = res[0].height; // 隱藏區域的高度(即是完整文字高度,偶然事件會計算錯誤) const hideAreaHeight = res[1].height; // 文字可見的最大高度 const maxHeight = lineHeightToPx * this.data.maxLine; // 如果是一行文字,偶然計算錯誤,用行高判斷 if (this.data.LineHeight && showAreaHeight <= maxHeight) { showFold = hideAreaHeight > maxHeight } // 如果文字超出最大行數,並且是顯示全文的狀態下,再次更新了文字 let onFold = false if (showAreaHeight == hideAreaHeight && showAreaHeight > maxHeight) { showFold = true onFold = true } this.setData({ width: res[0].width, showFold, onFold, }) }).exec() },
經過多次測試,修改,最後附上程式碼片段:
https://developers.weixin.qq.com/s/GWj19vmC7oxp
各位大神如果有更好的建議,可留言哦~~~
【相關學習推薦:】
以上就是聊聊小程式怎麼實現「全文收起」功能的詳細內容,更多請關注TW511.COM其它相關文章!