欢迎来到思维库

思维库

组件开发ScrollView嵌套ListContainer滑动问题详解

时间:2025-11-05 16:02:16 出处:娱乐阅读(143)

想了解更多内容,组件请访问:

和华为官方合作共建的嵌套鸿蒙技术社区

https://harmonyos.51cto.com

ScrollView嵌套ListContainer

就ScrollView嵌套ListContainer的滑动问题,社区问答中也是动问遇见了两次提问的小伙伴。在帮助第一个小伙解决这个问题的题详时候,我提供了一个思路和以前在写Android ScrollView嵌套ListView滑动问题的组件解决方法。经过方法的嵌套修改也是解决了他的问题,后续没有再把这个问题解决的动问全过程记录下来,直到发现有第二个小伙伴也遇到了同样的题详问题,准备把这个小问题写成一篇帖子,组件希望后面再遇到“ScrollView嵌套ListContainer 滑动问题”的嵌套同学可以帮助到你们。

思路

一、动问ScrollView嵌套ListContainer 想让ListContainer不滑动,题详只滑动ScrollView。组件在Android中有个东西叫做拦截器,嵌套ScrollView的动问拦截器,通过对拦截器的赋值达到只滑动ScrollView,不滑动ListView。

调用方式

因为ScrollView继承自ViewGroup,在ViewGroup中有有dispatchTouchEvent()这个方法,

但是服务器托管在HarmonyOS中,ScrollView继承自ComponentContainer,而且在ComponentContainer中没有类似于dispatchTouchEvent的拦截器方法,那么拦截器不能搞就得换方法。

二、这时第二个思路也成型了,因为ScrollView的高度是根据它内部的组件的高度变化的,当内部的组件高度大于手机屏幕的高度时会出现ScrollView的滚动,反之不会出现。

那么就只能从ScrollView的高度入手了,要改变ScrollView的高度就必须去改变它内部组件的高度,那么问题来了ScrollView嵌套ListContainer,ListContainer的高度最大只能到屏幕大小或者是固定于屏幕内部,一旦高度达到所设置的高度,ListContainer就会出现自动滚动此时ScrollView的滚动也会失效,这里是焦点的IT技术网关系滑动动作取到的焦点会在它当前组件上。

思路到这里也就清晰了,ListContainer的高度大于原始设置的高度时会发生滑动,ScrollView在内部组件高度大于手机屏幕时才会滑动。那么如果把ListContainer的高度设置成一个动态的固定值,ListContainer的数据永远不会被填充满,ListContainer就不会出现滑动。随即ListContainer的高度如果大于了屏幕的高度ScrollView就会滑动。

OK,问题找到了,解决ListContainer的动态高度就解决的滑动冲突。

解决问题

首先我找到了当初写Android时动态Listview高度的方法。这里就粘一下图

思路没有变,将每次listview的Item高度相加作为listview的整体高度,listview的高度就是动态的变化,listview的高度会根据数据的云服务器提供商增加而变化。

根据参考

下面开始写代码

首先整体布局文件,很简单ScrollView嵌套ListContainer

为了效果明显加入了一个图片

<?xml version="1.0" encoding="utf-8"?> <DirectionalLayout     xmlns:ohos="http://schemas.huawei.com/res/ohos"     ohos:height="match_parent"     ohos:width="match_parent">     <ScrollView         ohos:id="$+id:view"         ohos:height="match_parent"         ohos:width="match_parent">         <!--设置DirectionalLayout的高度为match_parent-->         <DirectionalLayout             ohos:height="match_parent"             ohos:width="match_parent"             ohos:alignment="center"             ohos:orientation="vertical">             <Image                 ohos:id="$+id:img"                 ohos:height="100vp"                 ohos:width="100vp"                 ohos:image_src="$media:icon"></Image>             <ListContainer                 ohos:id="$+id:list"                 ohos:height="match_parent"                 ohos:width="match_parent"                 ></ListContainer>         </DirectionalLayout>     </ScrollView> </DirectionalLayout> 

ListContainer的Item 布局,这里很简单就放一个文本

<?xml version="1.0" encoding="utf-8"?> <DirectionalLayout     xmlns:ohos="http://schemas.huawei.com/res/ohos"     ohos:height="match_content"     ohos:width="match_parent"     ohos:left_margin="16vp"     ohos:right_margin="16vp"     ohos:orientation="vertical">     <Text         ohos:id="$+id:item_index"         ohos:height="match_content"         ohos:width="match_content"         ohos:padding="4vp"         ohos:text="Item0"         ohos:text_size="20fp"         ohos:layout_alignment="center"/> </DirectionalLayout> 

 创建SampleItem.java,作为ListContainer的数据包装类。

public class SampleItem {     private String name;     public SampleItem(String name) {         this.name = name;     }     public String getName() {         return name;     }     public void setName(String name) {         this.name = name;     } } 

 写一个ListContainer的适配器用于放数据

public class SampleItemProvider extends BaseItemProvider {      private List<SampleItem> list;     private AbilitySlice slice;      public SampleItemProvider(List<SampleItem> list, AbilitySlice slice) {         this.list = list;         this.slice = slice;     }      @Override     public int getCount() {         return list == null ? 0 : list.size();     }      @Override     public Object getItem(int position) {         if (list != null && position >= 0 && position < list.size()){             return list.get(position);         }         return null;     }      @Override     public long getItemId(int position) {         return position;     }      @Override     public Component getComponent(int position, Component component, ComponentContainer componentContainer) {         final Component cpt;         if (component == null) {             cpt = LayoutScatter.getInstance(slice).parse(ResourceTable.Layout_item_sample, null, false);         } else {             cpt = component;         }         SampleItem sampleItem = list.get(position);         Text text = (Text) cpt.findComponentById(ResourceTable.Id_item_index);         text.setText(sampleItem.getName());         return cpt;     } } 

在Java代码中添加ListContainer的数据,并适配其数据结构。

还没有加动态高度的方法。

public class MainAbilitySlice extends AbilitySlice {      @Override     public void onStart(Intent intent) {         super.onStart(intent);         super.setUIContent(ResourceTable.Layout_ability_main);         initListContainer();     }      private void initListContainer() {         ListContainer listContainer = (ListContainer) findComponentById(ResourceTable.Id_list);         List<SampleItem> list = getData();         listContainer.setBoundarySwitch(true);  //添加分界线         SampleItemProvider sampleItemProvider = new SampleItemProvider(list, this);         listContainer.setItemProvider(sampleItemProvider);     }      private ArrayList<SampleItem> getData() {         ArrayList<SampleItem> list = new ArrayList<>();         for (int i = 0; i < 30; i++) {             list.add(new SampleItem("Item" + i));         }         return list;     } } 

查看效果

编写自定义高度方法:

private void setListContainerHeight(ListContainer listContainer) {         //获取当前listContainer的适配器         BaseItemProvider BaseItemProvider = listContainer.getItemProvider();        if (BaseItemProvider == null){            return;        }         int itemHeight = 0;         for (int i = 0; i < BaseItemProvider.getCount(); i++) {             //循环将listContainer适配器的Item数据进行累加             Component listItem = BaseItemProvider.getComponent(i, null, listContainer);             itemHeight += listItem.getHeight();         }         //对当前listContainer进行高度赋值         ComponentContainer.LayoutConfig config = listContainer.getLayoutConfig();         //这边加上(listContainer.getBoundaryThickness() * (BaseItemProvider.getCount()+1))         //listContainer.getBoundaryThickness() 就是分界线的高度         //(BaseItemProvider.getCount()+1) 是Item的数量  加1  是因为顶部还有一条分界线         config.height = itemHeight                 + (listContainer.getBoundaryThickness() * (BaseItemProvider.getCount()+1));         //赋值         listContainer.setLayoutConfig(config);     } 

调用方法:

实现效果:

出问题了,不能滑动!!!!!!!

找到了,问题在布局中。

重新运行,查看结果:

OK了,以达到了最终的效果。

代码放在了下面的资源链接里,大家可以进行参考。

FirstDemo.zip

想了解更多内容,请访问:

和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

-->

分享到:

上一篇:技巧一、用命令行往文件的顶部添加文字每次我都会重新寻找这个命令的写法。下面就是如何使用sed往一个文件顶部添加一行的方法:复制代码代码如下: sed -i 1s/^/line to insertn/ path/to/file/you/want/to/change.txt技巧二、用命令行往配置文件里插入多行文本这种方法非常简单,很多人都知道,下面就是如何用命令行将(>>)多行文本插入一个文件中。这里使用的是“here document”语法,它能让你通过块文本符号来将段落插入文件中,通常用的符合是EOF(意思是 “End Of File”):复制代码代码如下:cat >>path/to/file/to/append-to.txt << EOF export PATH=$HOME/jdk1.8.0_31/bin:$PATH export JAVA_HOME=$HOME/jdk1.8.0_31/ EOF两个”EOF“之间的所有内容都会被添加到文件中。技巧三、用命令行递归方式全局搜索目录文件和替换假如你使用Eclipse,ItelliJ或其它IDE,这些工具的强大重构能力也许会让你轻松实现很多事情。但我估计很多时候你的开发环境中没有这样的集成工具。如何使用命令行对一个目录进行递归搜索和替换?别想Perl语言,你可以使用find and sed。复制代码代码如下:# OSX version find . -type f -name *.txt -exec sed -i s/this/that/g {} +使用了一段时间后,我总结写出了一个函数,添加入了 .bashrc ,就像下面这样:复制代码代码如下:function sr { find . -type f -exec sed -i s/$1/$2/g {} +}你可以像这样使用它:复制代码代码如下:sr wrong_word correct_word技巧四、用命令行在vim和Dropbox里开启一个临时文件我过去喜欢用Emacs里的scratch facility功能。也经常用Vim快速创建临时文件。下面这两个函数是使用openssl生成随机的字符串作为文件名:复制代码代码如下:function sc { gvim ~/Dropbox/$(openssl rand -base64 10 | tr -dc a-zA-Z).txt } function scratch { gvim ~/Dropbox/$(openssl rand -base64 10 | tr -dc a-zA-Z).txt }在命令行窗口输入sc或scratch,一个新的gvim或macvim窗口就会弹出来,里面会加载一个随机文件名的临时文件。技巧五、用命令行下载文件,支持链接转向、HTTPS和安全加密等情况下载一个页面输出到终端,跟随链接转向,忽略安全异常:复制代码代码如下:curl -Lks 下载一个链接,跟随链接转向,忽略安全异常: [/code]curl -OLks 这里用了很多参数,你可以阅读这个简单的curl文档来了解它们。技巧六、Bashmarks你还没有在.bashrc里使用bashmarks吗?还在等待什么?它真的非常有用。它能帮你保持历史操作,跳回到你经常使用的目录。下面是我的配置文件里脚本,但我想上面的链接能提供你更多技巧:复制代码代码如下: # USAGE: # s bookmarkname - saves the curr dir as bookmarkname # g bookmarkname - jumps to the that bookmark # g b[TAB] - tab completion is available # l - list all bookmarks # save current directory to bookmarks touch ~/.sdirs function s { cat ~/.sdirs | grep -v export DIR_$1= >~/.sdirs1 mv ~/.sdirs1 ~/.sdirs echo export DIR_$1=$PWD >>~/.sdirs } # jump to bookmark function g { source ~/.sdirs cd $(eval $(echo echo $(echo $DIR_$1))) } # list bookmarks with dirnam function l { source ~/.sdirs env | grep ^DIR_ | cut -c5- | grep ^.*= } # list bookmarks without dirname function _l { source ~/.sdirs env | grep ^DIR_ | cut -c5- | grep ^.*= | cut -f1 -d = } # completion command for g function _gcomp { local curw COMPREPLY=() curw=${COMP_WORDS[COMP_CWORD]} COMPREPLY=($(compgen -W `_l` -- $curw)) return 0 } # bind completion command for g to _gcomp complete -F _gcomp g技巧七、从格式化输出里提取一列(我最常使用的awk技巧)我几乎天天都会使用它。真的。经常会有一些输出,我只需要其中的第二列,或第三列,下面这个命令就能做到这些:复制代码代码如下:#Sample output of git status -s command: $ git status -s M .bashrc .vim/bundle/extempore/ # Remove status code from git status and just get the file names $ git status -s | awk {print $2} .bashrc .vim/bundle/extempore/为什么不写个函数,让我们随时都可以用呢?复制代码代码如下: function col { awk -v col=$1 {print $col} }这使得提取列非常容易,比如,你不想要第一列?简单:复制代码代码如下:$ git status -s | col 2 .bashrc .vim/bundle/extempore/技巧八、忽略头x个词我对xargs很着迷,我感觉它就像一把快刀。但有时候用它获得的结果需要调整一下,也许需要取得一些值。例如,你想去掉下面文件影像里的一些信息:复制代码代码如下:function skip { n=$(($1 + 1)) cut -d -f$n- }下面是如何使用它: 使用 docker images 得到下面的输出:复制代码代码如下:$ docker images REPOSITORY TAG IMAGE ID CREATED VIRTUAL SIZE 65a9e3ef7171 3 weeks ago 1.592 GB 7c01ca6c30f2 3 weeks ago 11.1 MB 9518620e6a0e 3 weeks ago 7.426 MB 430707ee7fe8 3 weeks ago 7.426 MB boot2docker/boot2docker latest 1dbd7ebffe31 3 weeks ago 1.592 GB spaceghost/tinycore-x86_64 5.4 f47686df00df 7 weeks ago 11.1 MB durdn/bithub latest df1e39df8dbf 8 weeks ago 100.9 MB c5e6cf38d985 8 weeks ago 100.9 MB nginx latest e426f6ef897e 12 weeks ago 100.2 MB zoobab/tinycore-x64 latest 8cdd417ec611 8 months ago 7.426 MB scratch latest 511136ea3c5a 20 months ago 0 B 使用上面的函数,你可以获取所有的IDs:复制代码代码如下:$ docker images | col 3 IMAGE 65a9e3ef7171 7c01ca6c30f2 9518620e6a0e 430707ee7fe8 1dbd7ebffe31 f47686df00df df1e39df8dbf c5e6cf38d985 e426f6ef897e 8cdd417ec611 511136ea3c5a 进一步处理:复制代码代码如下:docker images | col 3 | xargs IMAGE 65a9e3ef7171 7c01ca6c30f2 9518620e6a0e 430707ee7fe8 1dbd7ebffe31 f47686df00df df1e39df8dbf c5e6cf38d985 e426f6ef897e 8cdd417ec611 511136ea3c5a 但前面的”IMAGE”字符我也想去掉:复制代码代码如下:docker images | col 3 | xargs | skip 1 65a9e3ef7171 7c01ca6c30f2 9518620e6a0e 430707ee7fe8 1dbd7ebffe31 f47686df00df df1e39df8dbf c5e6cf38d985 e426f6ef897e 8cdd417ec611 511136ea3c5a 完整的写下来就是这样:复制代码代码如下:docker rmi $(docker images | col 3 | xargs | skip 1)技巧九、创建自己的命令包在bash里,你可以很容易的创建自己的命令组件,你可以看一下下面我写的:复制代码代码如下: function dur { case $1 in clone|cl) git clone git@bitbucket.org:nicolapaolucci/$2.git ;; move|mv) git remote add bitbucket git@bitbucket.org:nicolapaolucci/$(basename $(pwd)).git git push --all bitbucket ;; trackall|tr) #track all remote branches of a project for remote in $(git branch -r | grep -v master ); do git checkout --track $remote ; done ;; key|k) #track all remote branches of a project ssh $2 mkdir -p .ssh && cat >>.ssh/authorized_keys < ~/.ssh/id_rsa.pub ;; fun|f) #list all custom bash functions defined typeset -F | col 3 | grep -v _ | xargs | fold -sw 60 ;; def|d) #show definition of function $1 typeset -f $2 ;; help|h|*) echo [dur]dn shell automation tools echo commands available: echo [cl]one, [mv|move] echo [f]fun lists all bash functions defined in .bashrc echo [def] lists definition of function defined in .bashrc echo [k]ey copies ssh key to target host echo [tr]ackall], [h]elp ;; esac }通过上面的脚本,我可以将ssh key拷贝到任何网站服务器——只需要键入 dur key user@somehost.总结你可以试一下我的这个.bashrc文件,或你自己也可以写一个。

下一篇:360N4A电池续航能力全面评测(为什么360N4A电池是你的最佳选择)

温馨提示:以上内容和图片整理于网络,仅供参考,希望对您有帮助!如有侵权行为请联系删除!

猜你喜欢

友情链接: