偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

HarmonyOS列表組件-ListContainer

開(kāi)發(fā) OpenHarmony
我們?cè)赼pp開(kāi)發(fā)中,列表組件絕對(duì)是使用場(chǎng)景最高的組件之一,鴻蒙為我們提供了ListContainer列表組件,它是一個(gè)是用來(lái)呈現(xiàn)連續(xù)、多行數(shù)據(jù)的組件。

[[417156]]

想了解更多內(nèi)容,請(qǐng)?jiān)L問(wèn):

51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)

https://harmonyos.51cto.com

前言

我們?cè)赼pp開(kāi)發(fā)中,列表組件絕對(duì)是使用場(chǎng)景最高的組件之一,鴻蒙為我們提供了ListContainer列表組件,它是一個(gè)是用來(lái)呈現(xiàn)連續(xù)、多行數(shù)據(jù)的組件,繼承自ComponentContainer,因此它是一個(gè)容器組件,使用BaseItemProvider來(lái)存儲(chǔ)對(duì)象。

正文

這里先簡(jiǎn)單介紹下ListContainer的基本用法:

1.在layout文件中聲明ListContainer控件;

2.定義列表控件的適配器ListItemProvider;

3.在Ability中給ListContainer設(shè)置數(shù)據(jù);

只需要三步就可以實(shí)現(xiàn)最基本的列表效果,這里就不貼代碼了,官方文檔有比較詳細(xì)的說(shuō)明,本文重點(diǎn)分析下如何通過(guò)自定義ListContainer來(lái)

實(shí)現(xiàn)子組件弧形排布的效果,并且隨著半徑和鏡像距離的改變子組件的排布也不斷變化,效果如下:

因?yàn)長(zhǎng)istContainer的子組件默認(rèn)是直線排列,可以通過(guò)設(shè)置LayoutManager(布局管理器)來(lái)改變子組件排列方式,但是官方只提供了TableLayoutManager(網(wǎng)格)和DirectionalLayoutManager(線性)兩種布局管理器,很顯然無(wú)法滿足需求,于是設(shè)想自定義一個(gè)TurnLayoutManager繼承DirectionalLayoutManager,然后重寫(xiě)相關(guān)方法對(duì)子組件重新排列:

然而事情并非如預(yù)想一般簡(jiǎn)單,DirectionalLayoutManager并沒(méi)有對(duì)應(yīng)的方法,它的父類(lèi)LayoutManager也沒(méi)有,驚不驚喜,意不意外?!

  1. public abstract class LayoutManager { 
  2.     public LayoutManager() { 
  3.         throw new RuntimeException("Stub!"); 
  4.     } 
  5.  
  6.     public void setOrientation(int orientation) { 
  7.         throw new RuntimeException("Stub!"); 
  8.     } 
  9.  
  10.     public int getOrientation() { 
  11.         throw new RuntimeException("Stub!"); 
  12.     } 

但是令人欣慰的是ListContainer并不是必須設(shè)置布局管理器子組件才能顯示出來(lái),于是一個(gè)大膽的念頭在我的腦海中閃現(xiàn):何不從ListContainer本身入手,自定義TurnListContainer類(lèi)繼承ListContainer,因?yàn)長(zhǎng)istContainer繼承自ComponentContainer,可以在onArrange()回調(diào)方法中修改子組件的位置以達(dá)到預(yù)期效果,事不宜遲,說(shuō)干就干:

1.實(shí)現(xiàn)ComponentContainer.ArrangeListener接口,重寫(xiě)onArrange()方法,在該方法中計(jì)算圓心,及x,y坐標(biāo)偏移量(列表是垂直方向時(shí)計(jì)算x軸偏移量,水平方向時(shí)計(jì)算y軸偏移量)

  1. @Override 
  2. public void onArrange() { 
  3.     //計(jì)算圓心 
  4.     this.center = deriveCenter(gravity, getOrientation(), radius, peekDistance, center); 
  5.     //設(shè)置子組件偏移     
  6.     setChildOffsets(); 

2.調(diào)用child.arrange()方法修改子組件位置(因?yàn)楸疚闹攸c(diǎn)講解自定義ListContainer中遇到的問(wèn)題,因此圓心、子組件的坐標(biāo)計(jì)算過(guò)程就不贅述了,熟悉三角函數(shù)就很容易看懂)

  1. public void setChildOffsetsVertical() { 
  2.     //遍利修改每一個(gè)子組件的位置 
  3.     for (int ii = 0; ii < getChildCount(); ii++) { 
  4.         Component child = getComponentAt(ii); 
  5.         if (child == null) { 
  6.             continue
  7.         } 
  8.         LayoutConfig layoutParams = child.getLayoutConfig(); 
  9.         //計(jì)算x軸偏移量 
  10.         final int offsetX = (int) resolveOffsetX(radius, child.getContentPositionY() +child.getHeight() / 2.0f, 
  11.                 center, peekDistance); 
  12.         final int x = gravity == Gravity.START ? offsetX + layoutParams.getMarginLeft() 
  13.                 : getWidth() - offsetX - child.getWidth() - getMarginStart(layoutParams); 
  14.         //調(diào)用子組件的arrange方法修改自身位置 
  15.         child.arrange(x, child.getTop(), child.getWidth(), child.getHeight()); 
  16.     } 

3.在修改半徑、鏡像距離、方向、文字旋轉(zhuǎn)時(shí),調(diào)用Component的postLayout()方法請(qǐng)求重新進(jìn)行測(cè)量、布局、繪制這三個(gè)流程來(lái)更新位置,因?yàn)槲业淖咏M件是provider提供的,不牽扯測(cè)量、和繪制過(guò)程,調(diào)用postLayout()的目的只是觸發(fā)onArrange回調(diào)對(duì)子組件位置修改。

  1. /** 
  2.     * 設(shè)置半徑 
  3.     * 
  4.     * @param radius 半徑 
  5.     */ 
  6.    public void setRadius(int radius) { 
  7.        this.radius = Math.max(radius, MIN_RADIUS); 
  8.        postLayout(); 
  9.    } 
  10.  
  11.    /** 
  12.     * 設(shè)置鏡像距離 
  13.     * 
  14.     * @param peekDistance 鏡像距離 
  15.     */ 
  16.    public void setPeekDistance(int peekDistance) { 
  17.        this.peekDistance = Math.min(Math.max(peekDistance, MIN_PEEK), radius); 
  18.        postLayout(); 
  19.    } 
  20.  
  21.    /** 
  22.     * 設(shè)置水平方向 
  23.     * 
  24.     * @param gravity 水平方向 
  25.     */ 
  26.    public void setGravity(@Gravity int gravity) { 
  27.        this.gravity = gravity; 
  28.        postLayout(); 
  29.    } 
  30.  
  31.    /** 
  32.     * 設(shè)置文字旋轉(zhuǎn) 
  33.     * 
  34.     * @param isRotate 文字是否旋轉(zhuǎn) 
  35.     */ 
  36.    public void setRotate(boolean isRotate) { 
  37.        this.isRotate = isRotate; 
  38.        postLayout(); 
  39.    } 

準(zhǔn)備工作告一段落,開(kāi)始測(cè)試, what? 滿心期待的結(jié)果并沒(méi)有出現(xiàn),除了設(shè)置文字旋轉(zhuǎn)有效果,修改半徑,鏡像距離,水平方向都沒(méi)效果。。。。。。這翻車(chē)來(lái)得太快就像龍卷風(fēng).

我開(kāi)始陷入漫長(zhǎng)的沉思中。。。。。。,嘗試了N多種方法后依然無(wú)果,最后分析認(rèn)為:我是在ListContainer的onArrange()回調(diào)中調(diào)用了子組件的onArrange()方法,有可能這兩個(gè)onArrange()方法存在沖突導(dǎo)致子組件本身的onArrange()失效,帶著些許疑問(wèn)我修改了代碼,設(shè)置半徑、鏡像距離時(shí)不用調(diào)用postLayout()來(lái)請(qǐng)求重新布局,直接調(diào)用child.arrange()更新子組件位置,代碼修改后效果如下:

效果還可以,修改半徑、鏡像距離,方向都能達(dá)到預(yù)期效果,但是細(xì)心的小伙伴一定觀察到了異常,。。。,靜止?fàn)顟B(tài)下是沒(méi)有問(wèn)題的,一旦開(kāi)始滾動(dòng)就出現(xiàn)原始位置和修改后位置交替出現(xiàn)的情況,為什么呢??,因?yàn)榭床坏皆创a我也不知道listContainer滾動(dòng)中的刷新邏輯,只能推測(cè)滾動(dòng)事件過(guò)程中肯定是觸發(fā)了重新布局的方法,導(dǎo)致子組件位置被反復(fù)重置。既然只有滾動(dòng)時(shí)才有問(wèn)題,那就從滾動(dòng)事件開(kāi)始入手吧,我的思路是監(jiān)聽(tīng)滾動(dòng)狀態(tài),如果已經(jīng)開(kāi)始滑動(dòng)了,改變滾動(dòng)狀態(tài)跳過(guò)慣性滾動(dòng)直接停止?jié)L動(dòng):

方法1:ListContainer.ScrolledListener監(jiān)聽(tīng)滾動(dòng),慣性滾動(dòng)時(shí)設(shè)置setEnabled(false)

  1. @Override 
  2. public void scrolledStageUpdate(Component component, int newStage) { 
  3.     switch (newStage) { 
  4.         case Component.SCROLL_IDLE_STAGE: 
  5.             //觸摸滾動(dòng) 
  6.             break; 
  7.         case Component.SCROLL_AUTO_STAGE: 
  8.             //慣性滾動(dòng) 
  9.             break; 
  10.         case Component.SCROLL_NORMAL_STAGE: 
  11.             //停止?jié)L動(dòng) 
  12.             break; 
  13.     } 

方法2:Component.TouchEventListener監(jiān)聽(tīng)滾動(dòng),手指抬起時(shí)設(shè)置listContainer.setEnabled(false)

  1. @Override 
  2. public boolean onTouchEvent(Component component, TouchEvent touchEvent) { 
  3.     switch (touchEvent.getAction()){ 
  4.         case TouchEvent.PRIMARY_POINT_DOWN: 
  5.             //按下時(shí)設(shè)置禁止滑動(dòng) 
  6.             setEnabled(false); 
  7.             break; 
  8.         case TouchEvent.PRIMARY_POINT_UP: 
  9.             //抬起時(shí)設(shè)置可以滑動(dòng) 
  10.             setEnabled(true); 
  11.             break; 
  12.         default
  13.             return true
  14.     } 
  15.     return false

但是經(jīng)過(guò)測(cè)試,兩種方法都沒(méi)法立即停止慣性滾動(dòng),也就是說(shuō)沒(méi)有辦法來(lái)干預(yù)ListContainer的滾動(dòng)狀態(tài),至少目前我沒(méi)有找到阻止慣性滾動(dòng)的相關(guān)API,那么,只能再嘗試其他方法了,。。。。。。。。。。。。。。又一次我陷入漫長(zhǎng)的沉思中。。。。。。,在嘗試了各種方法都以失敗告終后,最終在我鍥而不舍的努力下終于得以解決,這是這個(gè)項(xiàng)目中我遇到的最大的坑沒(méi)有之一,耗費(fèi)了太多時(shí)間和精力,鴨梨好大呀,罷了罷了。。。,話不多說(shuō),直接看正解吧:

  1. public void setChildOffsetsVertical() { 
  2.     for (int ii = 0; ii < getChildCount(); ii++) { 
  3.         Component child = getComponentAt(ii); 
  4.         if (child == null) { 
  5.             continue
  6.         } 
  7.         LayoutConfig layoutParams = child.getLayoutConfig(); 
  8.         //計(jì)算x軸偏移量 
  9.         final int offsetX = (int) resolveOffsetX(radius, child.getContentPositionY() + child.getHeight() / 2.0f, 
  10.                 center, peekDistance); 
  11.         final int xx = gravity == Gravity.START ? offsetX + layoutParams.getMarginLeft() 
  12.                 : getWidth() - offsetX - child.getWidth() - getMarginStart(layoutParams); 
  13.        //調(diào)用子組件的setTranslationX方法修改自身x軸偏移量 
  14.         child.setTranslationX(xx); 
  15.         //設(shè)置子組件旋轉(zhuǎn) 
  16.         setChildRotationVertical(gravity, child, radius, center); 
  17.     } 

對(duì),沒(méi)有錯(cuò),就只是修改了一行代碼,用child.setTranslationX()替換child.arrange(),就這么簡(jiǎn)單,不管你相不相信它就是這么神奇,之所以說(shuō)神奇是因?yàn)榭床坏皆创a不知道ListContainer的內(nèi)部滾動(dòng)機(jī)制:

經(jīng)過(guò)許多波折最終達(dá)到了預(yù)期的效果,肝都要爆了, 其實(shí)一開(kāi)始并不覺(jué)得項(xiàng)目本身有多復(fù)雜,計(jì)算量也不大,直到開(kāi)始做的時(shí)候問(wèn)題才一一顯現(xiàn)出來(lái),不得不感慨,人生路上哪有那么多的順風(fēng)順?biāo)樞氖?總會(huì)有一些波折和苦難不合時(shí)宜的出現(xiàn),磕磕絆絆的人生才是完整的。。。。。。,寫(xiě)這個(gè)文章主要是分享下開(kāi)發(fā)中我遇到的坑(主要還是想抒發(fā)下被代碼虐了千百遍的爆炸心態(tài)),避免后面再有人誤入歧途,浪費(fèi)寶貴的時(shí)間。

結(jié)束

下面是技術(shù)總結(jié):

1.使用postLayout()請(qǐng)求重新布局后再調(diào)用child.arrange(),會(huì)導(dǎo)致child.arrange()失效;

  1. @Override 
  2. public boolean onArrange(int i, int i1, int i2, int i3) { 
  3.     child.arrange();//此時(shí)設(shè)置子組件位置無(wú)效 
  4.     return false

2.child.arrange()會(huì)觸發(fā)listContainer的滾動(dòng)刷新機(jī)制,反復(fù)重置位置,鴻蒙調(diào)用child.arrange()修改子組件位置一切正常,但是listContainer滾動(dòng)中位置會(huì)被頻繁重置,如果涉及到修改子組件位置的,出現(xiàn)滾動(dòng)中位置被反復(fù)重置的,可以嘗試用child.setTranslationX(x)和child.setTranslationX(y)來(lái)代替;

3.監(jiān)聽(tīng)滾動(dòng)事件

android中有scrollVerticallyBy和scrollHorizontallyBy回調(diào)來(lái)監(jiān)聽(tīng)橫向滾動(dòng)和垂直滾動(dòng),鴻蒙可以實(shí)現(xiàn)ListContainer.ScrolledListener接口或者Component.TouchEventListener接口監(jiān)聽(tīng),我這里只所以選擇實(shí)現(xiàn)ListContainer.ScrolledListener是因?yàn)榭梢灾貙?xiě)它的兩個(gè)方法,onContentScrolled監(jiān)聽(tīng)滾動(dòng)中變化和scrolledStageUpdate監(jiān)聽(tīng)滾動(dòng)狀態(tài)變化,會(huì)比TouchEventListener方便些;

4.setEnable(false)

這個(gè)方法可以禁止listContainer滾動(dòng),但是如果listContainer已經(jīng)開(kāi)始滾動(dòng)了再設(shè)置setEnable(false)并不會(huì)阻止listContainer慣性滾動(dòng),禁止慣性滾動(dòng)的方法目前還沒(méi)有找到。

想了解更多內(nèi)容,請(qǐng)?jiān)L問(wèn):

51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)

https://harmonyos.51cto.com

 

責(zé)任編輯:jianghua 來(lái)源: 鴻蒙社區(qū)
相關(guān)推薦

2020-12-28 10:15:18

鴻蒙HarmonyOSListContain

2021-08-25 09:49:48

鴻蒙HarmonyOS應(yīng)用

2022-04-24 15:17:56

鴻蒙操作系統(tǒng)

2021-12-20 20:51:44

鴻蒙HarmonyOS應(yīng)用

2025-02-06 03:15:48

2021-01-08 09:55:17

鴻蒙HarmonyOS組裝列表

2021-10-18 10:14:26

鴻蒙HarmonyOS應(yīng)用

2021-08-02 14:54:50

鴻蒙HarmonyOS應(yīng)用

2022-10-26 15:54:46

canvas組件鴻蒙

2022-10-25 15:12:24

自定義組件鴻蒙

2021-03-03 09:42:26

鴻蒙HarmonyOS圖片裁剪

2021-08-24 14:57:27

鴻蒙HarmonyOS應(yīng)用

2021-06-22 09:44:56

鴻蒙HarmonyOS應(yīng)用

2021-11-01 10:21:36

鴻蒙HarmonyOS應(yīng)用

2021-08-09 10:24:49

鴻蒙HarmonyOS應(yīng)用

2021-01-21 09:45:36

鴻蒙HarmonyOS分布式

2022-07-06 20:24:08

ArkUI計(jì)時(shí)組件

2021-03-26 09:35:35

鴻蒙HarmonyOS應(yīng)用開(kāi)發(fā)

2021-03-31 15:49:34

鴻蒙HarmonyOS應(yīng)用

2022-05-19 15:59:23

組件焦點(diǎn)鴻蒙
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)