HarmonyOS 分布式之仿抖音应用

开发 分布式 OpenHarmony
使用Java UI开发分布式仿抖音应用,上下滑动切换视频,评论功能,设备迁移功能:记录播放的视频页和进度、评论数据。

[[430075]]

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

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

https://harmonyos.51cto.com

项目介绍

使用Java UI开发分布式仿抖音应用,上下滑动切换视频,评论功能,设备迁移功能:记录播放的视频页和进度、评论数据。

效果演示

1.上下滑动切换视频、点击迁移图标,弹框选择在线的设备,完成视频数据的迁移。

2.点击评论图标查看评论,编辑评论内容并发送。点击迁移图标,弹框选择在线的设备,完成评论数据的迁移。

项目结构

主要代码

1、上下滑动页面

页面切换用到系统组件PageSlider,默认左右切换,设置为上下方向:setOrientation(Component.VERTICAL);

  1. import ohos.aafwk.ability.AbilitySlice; 
  2. import ohos.aafwk.content.Intent; 
  3. import ohos.agp.components.*; 
  4.  
  5. import java.util.ArrayList; 
  6. import java.util.List; 
  7.  
  8. public class MainAbilitySlice extends AbilitySlice { 
  9.     @Override 
  10.     public void onStart(Intent intent) { 
  11.         super.onStart(intent); 
  12.         super.setUIContent(ResourceTable.Layout_ability_main); 
  13.         // 查找滑动页面组件 
  14.         PageSlider pageSlider = (PageSlider) findComponentById(ResourceTable.Id_pageSlider); 
  15.         // 设置滑动方向为上下滑动 
  16.         pageSlider.setOrientation(Component.VERTICAL); 
  17.         // 集合测试数据 
  18.         List<String> listData=new ArrayList<>(); 
  19.         listData.add("第一页"); 
  20.         listData.add("第二页"); 
  21.         listData.add("第三页"); 
  22.          
  23.         // 设置页面适配器 
  24.         pageSlider.setProvider(new PageSliderProvider() { 
  25.             /** 
  26.              * 获取当前适配器中可用视图的数量 
  27.              */ 
  28.             @Override 
  29.             public int getCount() { 
  30.                 return listData.size(); 
  31.             } 
  32.             /** 
  33.              * 创建页面 
  34.              */ 
  35.             @Override 
  36.             public Object createPageInContainer(ComponentContainer container, int position) { 
  37.                 // 查找布局 
  38.                 Component component = LayoutScatter.getInstance(getContext()).parse(ResourceTable.Layout_item_page, nullfalse); 
  39.                 Text textContent = (Text) component.findComponentById(ResourceTable.Id_text_item_page_content); 
  40.                 // 设置数据 
  41.                 textContent.setText(listData.get(position)); 
  42.                 // 添加到容器中 
  43.                 container.addComponent(component); 
  44.                 return component; 
  45.             } 
  46.             /** 
  47.              * 销毁页面 
  48.              */ 
  49.             @Override 
  50.             public void destroyPageFromContainer(ComponentContainer container, int position, Object object) { 
  51.                 // 从容器中移除 
  52.                 container.removeComponent((Component) object); 
  53.             } 
  54.             /** 
  55.              * 检查页面是否与对象匹配 
  56.              */ 
  57.             @Override 
  58.             public boolean isPageMatchToObject(Component page, Object object) { 
  59.                 return true
  60.             } 
  61.         }); 
  62.  
  63.         // 添加页面改变监听器 
  64.         pageSlider.addPageChangedListener(new PageSlider.PageChangedListener() { 
  65.             /** 
  66.              * 页面滑动时调用 
  67.              */ 
  68.             @Override 
  69.             public void onPageSliding(int itemPos, float itemPosOffset, int itemPosOffsetPixels) {} 
  70.             /** 
  71.              * 当页面滑动状态改变时调用 
  72.              */ 
  73.             @Override 
  74.             public void onPageSlideStateChanged(int state) {} 
  75.             /** 
  76.              * 选择新页面时回调 
  77.              */ 
  78.             @Override 
  79.             public void onPageChosen(int itemPos) { 
  80.                 // 在此方法下,切换页面获取当前页面的视频源,进行播放 
  81.                 String data = listData.get(itemPos); 
  82.             } 
  83.         }); 
  84.     } 

2、播放视频

视频播放使用Player,视频画面窗口显示使用SurfaceProvider。

  1. import ohos.aafwk.ability.AbilitySlice; 
  2. import ohos.aafwk.content.Intent; 
  3. import ohos.agp.components.surfaceprovider.SurfaceProvider; 
  4. import ohos.agp.graphics.SurfaceOps; 
  5. import ohos.global.resource.RawFileDescriptor; 
  6. import ohos.media.common.Source; 
  7. import ohos.media.player.Player; 
  8.  
  9. import java.io.IOException; 
  10.  
  11. public class MainAbilitySlice extends AbilitySlice { 
  12.     // 视频路径 
  13.     private final String videoPath = "resources/rawfile/HarmonyOS.mp4"
  14.     // 播放器 
  15.     private Player mPlayer; 
  16.  
  17.     @Override 
  18.     public void onStart(Intent intent) { 
  19.         super.onStart(intent); 
  20.         super.setUIContent(ResourceTable.Layout_ability_main); 
  21.         // 初始化播放器 
  22.         mPlayer = new Player(getContext()); 
  23.         // 查找视频窗口组件 
  24.         SurfaceProvider surfaceProvider = (SurfaceProvider) findComponentById(ResourceTable.Id_surfaceProvider); 
  25.         // 设置视频窗口在顶层 
  26.         surfaceProvider.pinToZTop(true); 
  27.         // 设置视频窗口操作监听 
  28.         if (surfaceProvider.getSurfaceOps().isPresent()) { 
  29.             surfaceProvider.getSurfaceOps().get().addCallback(new SurfaceOps.Callback() { 
  30.                 /** 
  31.                  * 创建视频窗口 
  32.                  */ 
  33.                 @Override 
  34.                 public void surfaceCreated(SurfaceOps holder) { 
  35.                     try { 
  36.                         RawFileDescriptor fileDescriptor = getResourceManager().getRawFileEntry(videoPath).openRawFileDescriptor(); 
  37.                         Source source = new Source(fileDescriptor.getFileDescriptor(), 
  38.                                 fileDescriptor.getStartPosition(), 
  39.                                 fileDescriptor.getFileSize() 
  40.                         ); 
  41.                         // 设置媒体文件 
  42.                         mPlayer.setSource(source); 
  43.                         // 设置播放窗口 
  44.                         mPlayer.setVideoSurface(holder.getSurface()); 
  45.                         // 循环播放 
  46.                         mPlayer.enableSingleLooping(true); 
  47.                         // 准备播放环境并缓冲媒体数据 
  48.                         mPlayer.prepare(); 
  49.                         // 开始播放 
  50.                         mPlayer.play(); 
  51.                     } catch (IOException e) { 
  52.                         e.printStackTrace(); 
  53.                     } 
  54.  
  55.                 } 
  56.                 /** 
  57.                  * 视频窗口改变 
  58.                  */ 
  59.                 @Override 
  60.                 public void surfaceChanged(SurfaceOps holder, int format, int width, int height) {} 
  61.                 /** 
  62.                  * 视频窗口销毁 
  63.                  */ 
  64.                 @Override 
  65.                 public void surfaceDestroyed(SurfaceOps holder) {} 
  66.             }); 
  67.         } 
  68.     } 
  69.  
  70.     @Override 
  71.     protected void onStop() { 
  72.         super.onStop(); 
  73.         // 页面销毁,释放播放器 
  74.         if (mPlayer != null) { 
  75.             mPlayer.stop(); 
  76.             mPlayer.release(); 
  77.         } 
  78.     } 

3、跨设备迁移示例

跨设备迁移使用IAbilityContinuation接口。

1、在entry下的config.json配置权限

  1. "reqPermissions": [ 
  2.       { 
  3.         "name""ohos.permission.DISTRIBUTED_DATASYNC" 
  4.       }, 
  5.       { 
  6.         "name""ohos.permission.GET_DISTRIBUTED_DEVICE_INFO" 
  7.       }, 
  8.       { 
  9.         "name""ohos.permission.DISTRIBUTED_DEVICE_STATE_CHANGE" 
  10.       } 
  11.     ] 

2、实现IAbilityContinuation接口,说明:一个应用可能包含多个Page,仅需要在支持迁移的Page中通过以下方法实现IAbilityContinuation接口。同时,此Page所包含的所有AbilitySlice也需要实现此接口。

  1. import ohos.aafwk.ability.AbilitySlice; 
  2. import ohos.aafwk.ability.IAbilityContinuation; 
  3. import ohos.aafwk.content.Intent; 
  4. import ohos.aafwk.content.IntentParams; 
  5. import ohos.agp.components.Button; 
  6. import ohos.agp.components.Text; 
  7. import ohos.bundle.IBundleManager; 
  8. import ohos.distributedschedule.interwork.DeviceInfo; 
  9. import ohos.distributedschedule.interwork.DeviceManager; 
  10.  
  11. import java.util.List; 
  12.  
  13. public class MainAbilitySlice extends AbilitySlice implements IAbilityContinuation { 
  14.     private String data = ""
  15.     String PERMISSION = "ohos.permission.DISTRIBUTED_DATASYNC"
  16.  
  17.     @Override 
  18.     public void onStart(Intent intent) { 
  19.         super.onStart(intent); 
  20.         super.setUIContent(ResourceTable.Layout_ability_main); 
  21.         // 申请权限 
  22.         if (verifySelfPermission(PERMISSION) != IBundleManager.PERMISSION_GRANTED) { 
  23.             requestPermissionsFromUser(new String[]{PERMISSION}, 0); 
  24.         } 
  25.         Button button = (Button)findComponentById(ResourceTable.Id_button); 
  26.         Text text = (Text)findComponentById(ResourceTable.Id_text); 
  27.          
  28.         // 点击迁移 
  29.         button.setClickedListener(component -> { 
  30.             // 查询分布式网络中所有在线设备(不包括本地设备)的信息。 
  31.             List<DeviceInfo> deviceList = DeviceManager.getDeviceList(DeviceInfo.FLAG_GET_ONLINE_DEVICE); 
  32.             if (deviceList.size()>0) { 
  33.                 // 启动迁移,指定的设备ID 
  34.                 continueAbility(deviceList.get(0).getDeviceId()); 
  35.             } 
  36.         }); 
  37.         // 显示迁移的数据 
  38.         text.setText("迁移的数据:"+data); 
  39.     } 
  40.     /** 
  41.      * 启动迁移时首次调用此方法 
  42.      * @return 是否进行迁移 
  43.      */ 
  44.     @Override 
  45.     public boolean onStartContinuation() { 
  46.         return true
  47.     } 
  48.     /** 
  49.      * 迁移时存入数据 
  50.      */ 
  51.     @Override 
  52.     public boolean onSaveData(IntentParams intentParams) { 
  53.         intentParams.setParam("data","测试数据"); 
  54.         return true
  55.     } 
  56.     /** 
  57.      * 获取迁移存入的数据,在生命周期的onStart之前执行 
  58.      */ 
  59.     @Override 
  60.     public boolean onRestoreData(IntentParams intentParams) { 
  61.         data= (String) intentParams.getParam("data"); 
  62.         return true
  63.     } 
  64.     /** 
  65.      * 迁移完成 
  66.      */ 
  67.     @Override 
  68.     public void onCompleteContinuation(int i) {} 

根据上面的核心代码示例,了解实现原理,接下来便可以结合实际需求完善功能了。

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

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

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2021-11-16 09:38:10

鸿蒙HarmonyOS应用

2018-07-17 08:14:22

分布式分布式锁方位

2021-01-21 09:45:36

鸿蒙HarmonyOS分布式

2021-12-13 11:07:10

鸿蒙HarmonyOS应用

2020-11-06 12:12:35

HarmonyOS

2022-03-06 21:43:05

Citus架构PostgreSQL

2019-09-26 15:43:52

Hadoop集群防火墙

2019-02-13 13:41:07

MemCache分布式HashMap

2018-12-14 10:06:22

缓存分布式系统

2021-07-23 08:57:32

鸿蒙HarmonyOS应用

2019-10-10 09:16:34

Zookeeper架构分布式

2023-01-09 11:23:03

系统

2017-09-01 05:35:58

分布式计算存储

2019-06-19 15:40:06

分布式锁RedisJava

2023-05-29 14:07:00

Zuul网关系统

2021-12-10 15:06:56

鸿蒙HarmonyOS应用

2021-12-02 10:11:44

鸿蒙HarmonyOS应用

2021-12-28 17:03:29

数据质量分布式

2019-12-26 08:59:20

Redis主从架构

2019-07-04 15:13:16

分布式缓存Redis
点赞
收藏

51CTO技术栈公众号