DistributedVideoPlayer 分散ビデオプレーヤー (I)

DistributedVideoPlayer 分散ビデオプレーヤー (I)

[[429810]]

詳細については、以下をご覧ください。

51CTOとHuaweiが共同で構築したHongmengテクノロジーコミュニティ

https://harmonyos..com

導入

この例は、基本的なビデオ再生機能を提供し、携帯電話とテレビ間でビデオを転送できるようにする公式のビデオ再生機能テンプレートの拡張です。

アプリケーションは、モバイル側 (entry)、TV 側 (entrytv)、および依存モジュール (commonlib) に分かれています。

この例に基づいて、携帯電話にはビデオのプレイリスト機能、再生詳細ページ、コメント機能が追加されます。携帯電話で再生した動画をテレビ側にストリーミングし、リモコン機能を実現できます。

内容が盛りだくさんなので、2回に分けてご説明します。この記事では主にモバイル部分について説明します。

1. ビデオ プレーヤーを実装します。2. プレイリストを実装します。3. コメント機能を実装します。

【この記事は優秀クリエイター奨励金に参加しています】

エフェクト表示


環境を構築する

DevEco Studio をインストールします。詳細については、DevEco Studio のダウンロードを参照してください。

DevEco Studio 開発環境をセットアップします。 DevEco Studio 開発環境はネットワーク環境に依存する必要があります。ツールを正常に使用するには、ネットワークに接続する必要があります。次の 2 つの状況に応じて開発環境を構成できます。

インターネットに直接アクセスできる場合は、HarmonyOS SDK をダウンロードするだけで済みます。

ネットワークがインターネットに直接アクセスできず、プロキシ サーバー経由でアクセスする必要がある場合は、「開発環境の構成」を参照してください。

ソースコードをダウンロードしたら、DevEco を使用してプロジェクトを開きます。

コード構造

Javaコード

  1. │ config.json
  2. ├─ジャワ
  3. │ └─com
  4. │ └─ブティ
  5. │ └─分散ビデオプレイヤー
  6. │ │ メインアビリティ.java
  7. │ │MyApplication.java
  8. │ │
  9. │ ├─能力
  10. │ │ DevicesSelectAbility.java
  11. │ │ MainAbilitySlice.java #ビデオ再生リストページ
  12. │ │ SyncControlServiceAbility.java
  13. │ │ VideoPlayAbility.java #ビデオ再生機能
  14. │ │ VideoPlayAbilitySlice.java #ビデオ再生の詳細とコメントページ
  15. │ │
  16. │ ├─コンポーネント
  17. │ │ EpisodesSelectionDialog.java
  18. │ │リモートコントローラ.java
  19. │ │ VideoPlayerPlaybackButton.java #再生ボタンコンポーネント
  20. │ │ VideoPlayerSlider.java #再生時間プログレスバー
  21. │ │
  22. │ ├─定数
  23. │ │ 定数.java
  24. │ │ ResolutionEnum.java #解像度列挙
  25. │ │ RouteRegister.java #カスタムルーティング
  26. │ │
  27. │ ├─データ
  28. │ │ VideoInfo.java #基本的なビデオ情報
  29. │ │ VideoInfoService.java #シミュレーションデータに使用されるビデオ情報サービス
  30. │ │ Videos.java #ビデオリスト
  31. │ │
  32. │ ├─モデル
  33. │ │ CommentModel.java #コメントモデル
  34. │ │デバイスモデル.java
  35. │ │ ResolutionModel.java #解像度モデル
  36. │ │ VideoModel.java #ビデオモデル
  37. │ │
  38. │ ├─プロバイダー
  39. │ │ CommentItemProvider.java #コメントデータプロバイダ
  40. │ │デバイスアイテムプロバイダー.java
  41. │ │ ResolutionItemProvider.java #解像度データプロバイダー
  42. │ │ VideoItemProvider.java #ビデオデータプロバイダ
  43. │ │
  44. │ └─ユーティリティ
  45. │ AppUtil.java #Toolクラス
  46. │ DateUtils.java

リソースファイル

  1. └─リソース
  2. ├─ベース
  3. │ ├─要素
  4. │ │ カラー.json
  5. │ │フロート.json
  6. │ │ strarray.json
  7. │ │ 文字列.json
  8. │ │
  9. │ ├─グラフィック
  10. │ │ background_ability_control_bg.xml
  11. │ │ background_ability_control_middle.xml
  12. │ │ background_ability_control_ok.xml
  13. │ │ background_ability_devices.xml
  14. │ │ background_ability_episodes.xml
  15. │ │ background_button_click.xml
  16. │ │ background_button_clicked.xml
  17. │ │ 背景エピソードアイテム.xml
  18. │ │ 背景エピソード品質.xml
  19. │ │ 背景エピソードトレーラー.xml
  20. │ │ background_slide_thumb.xml
  21. │ │ background_switch_checked.xml
  22. │ │ background_switch_empty.xml
  23. │ │ background_switch_thumb.xml
  24. │ │ 背景スイッチトラック.xml
  25. │ │ リスト区切り線.xml
  26. │ │ シェイプスライダーサム.xml
  27. │ │
  28. │ ├─レイアウト
  29. │ │ ability_main.xml #プレイリストレイアウト
  30. │ │ comments_item.xml #単一コメントレイアウト
  31. │ │ ダイアログプレイリスト.xml
  32. │ │ ダイアログ解像度リスト.xml
  33. │ │ ダイアログテーブルレイアウト.xml
  34. │ │ hm_sample_ability_video_box.xml #ビデオ再生コンポーネントページ
  35. │ │ hm_sample_ability_video_comments.xml #再生詳細レイアウトページ
  36. │ │ hm_sample_view_video_box_seek_bar_style1.xml #再生進捗バーのレイアウト
  37. │ │ hm_sample_view_video_box_seek_bar_style2.xml
  38. │ │ リモートアビリティコントロール.xml
  39. │ │ リモートアビリティエピソード.xml
  40. │ │ リモート機能選択デバイス.xml
  41. │ │ リモートアビリティサウンド機器.xml
  42. │ │ リモートデバイスアイテム.xml
  43. │ │ リモートエピソードアイテム.xml
  44. │ │ リモートビデオ品質アイテム.xml
  45. │ │
  46. │ ├─メディア
  47. │ │ コメント.png
  48. │ │ 素晴らしい.png
  49. │ │ アイコン.png
  50. │ │ ic_anthology.png
  51. │ │
  52. │ └─プロフィール
  53. ├─ja
  54. │ └─要素
  55. │ 文字列.json
  56. ├─rawファイル
  57. │ videos.json #シミュレーションデータJSONファイル
  58. └─zh
  59. └─要素
  60. 文字列.json

実装手順

1. ビデオプレーヤーを実装する

commonlib への依存関係を導入すると、ビデオ プレーヤーを簡単に実装できます。

エントリのbuild.gradleにcommonlibへの依存関係を追加する

  1. 依存関係 {
  2. 実装 fileTree(dir: 'libs' 、 include: [ '*.jar' '*.har' ])
  3. テスト実装'junit:junit:4.13'  
  4. ohosTestImplementation 'com.huawei.ohos.testkit:runner:1.0.0.200'  
  5. //commonlib依存関係を参照
  6. 実装プロジェクト(パス: ':commonlib' )
  7. }

1.1.ページレイアウト hm_sample_ability_video_comments.xml

VideoPlayerViewコンポーネントを追加する

  1. <?xml バージョン = "1.0"エンコーディング = "utf-8" ?>
  2. <スタックレイアウト
  3. xmlns:ohos= "http://schemas.huawei.com/res/ohos"  
  4. ohos:id= "$+id:root_layout"  
  5. ohos:height= "match_parent"  
  6. ohos:width= "match_parent"  
  7. ohos:background_element= "#FFFFFFFF"  
  8. ohos:orientation= "垂直" >
  9.  
  10. <com.buty.distributedvideoplayer.player.ui.widget.media.VideoPlayerView
  11. ohos:id= "$+id:video_view"  
  12. ohos:height= "250vp"  
  13. ohos:width= "match_parent" />
  14. ...
  15. </スタックレイアウト>

1.2.Javaコード

ビデオ コンポーネント オブジェクトを取得したら、次の 2 つの手順だけが必要です。1. ビデオ リソースのパスと説明を設定します。 2. プレーヤーのコアコンポーネントとカスタムコンポーネントを設定します。数行のキーコード VideoPlayAbilitySlice.java

  1. /**
  2. * プレーヤーを初期化する
  3. */
  4. プライベートvoid initPlayer() {
  5. HiLog.debug(LABEL、 "initPlayer" );
  6. //カスタムビデオ再生ビューコンポーネント
  7. プレーヤー = (VideoPlayerView) findComponentById(ResourceTable.Id_video_view);
  8.  
  9. if (プレイヤー != null ) {
  10. //ビデオ情報サービスを取得する
  11. ビデオサービス = 新しいビデオ情報サービス (getContext());
  12. //現在再生中のビデオのパスを取得します
  13. 文字列パス = videoService
  14. .getVideoInfoByIndex(現在の再生インデックス)
  15. .getResolutions()
  16. .get(現在の再生解像度インデックス)
  17. .getUrl();
  18. //ビデオの説明
  19. 文字列 videoDesc = videoService.getVideoInfoByIndex(currentPlayingIndex).getVideoDesc();
  20. HiLog.debug(LABEL、 "videoDesc = " + videoDesc + " パス = " + パス);
  21.  
  22. パスがnull場合
  23. //パスと説明を設定する
  24. player.setVideoPathAndTitle(パス、ビデオの説明);
  25. //準備ができたらビデオを再生し、再生進行状況バーを設定します
  26. プレイヤー.setPlayerOnPreparedListener()
  27. () -> {
  28. プレイヤーを起動します。
  29. //再生位置を設定する
  30. player.seekTo(現在の再生位置);
  31. });
  32. //ダブルクリックして再生または一時停止します
  33. プレイヤー.setDoubleClickedListener(
  34. コンポーネント -> {
  35. //テレビ側を制御しているかどうか
  36. リモートコントローラがnullの場合、リモートコントローラ.isShown() は次のように動作します。
  37. 戻る;
  38. }
  39. HiLog.debug(LABEL, "VideoPlayView ダブルクリック イベント" );
  40. (プレイヤーが再生中の場合){
  41. プレーヤーを一時停止します。
  42. }それ以外{
  43. プレイヤーを起動します。
  44. }
  45. });
  46. //再生エラーを監視する
  47. プレイヤー.setErrorListener()
  48. (エラータイプ、エラーコード) -> {
  49. ToastDialog トースト = 新しい ToastDialog(getContext());
  50. スイッチ (エラータイプ) {
  51. HmPlayerAdapter.ERROR_LOADING_RESOURCEの場合:
  52. トースト.setText()
  53. AppUtil.getStringResource() 関数は、
  54. getContext()、ResourceTable.String_media_file_loading_error));
  55. 壊す;
  56. HmPlayerAdapter.ERROR_INVALID_OPERATIONの場合:
  57. トースト.setText()
  58. AppUtil.getStringResource() 関数は、
  59. getContext()、ResourceTable.String_invalid_operation));
  60. 壊す;
  61. デフォルト
  62. トースト.setText()
  63. AppUtil.getStringResource() 関数は、
  64. getContext()、ResourceTable.String_undefined_error_type));
  65. 壊す;
  66. }
  67. getUITaskDispatcher().asyncDispatch(toast::show);
  68. });
  69. }
  70. //コアコンポーネント、再生時間進行スライダーを追加
  71. コアコンポーネントを追加します。
  72. //カスタムコンポーネントを追加する
  73. カスタムコンポーネントを追加します。
  74. HiLog.debug(LABEL、 「initPlayer 終了」 );
  75. }
  76. }

2. ビデオプレイリスト機能を実装する

2.1.ページレイアウト ability_main.xml

DirectionalLayoutレイアウトコンポーネント、ScrollViewコンポーネント、ListContainerコンポーネントが使用されています。

  1. <?xml バージョン = "1.0"エンコーディング = "utf-8" ?>
  2. <方向レイアウト
  3. xmlns:ohos= "http://schemas.huawei.com/res/ohos"  
  4. ohos:id= "$+id:ability_main_root"  
  5. ohos:height= "match_parent"  
  6. ohos:width= "match_parent"  
  7. ohos:orientation= "垂直" >
  8.  
  9. <方向レイアウト
  10. ohos:id= "$+id:ability_main_titlebar"  
  11. ohos:height= "match_content"  
  12. ohos:width= "match_parent"  
  13. ohos:background_element= "#B0B0B0"  
  14. ohos:orientation= "水平"  
  15. ohos:padding= "10vp" >
  16.  
  17. <方向レイアウト
  18. ohos:height= "match_parent"  
  19. ohos:width= "match_content"  
  20. ohos:weight= "1" >
  21.  
  22. <テキスト
  23. ohos:id= "$+id:tag_favorite"  
  24. ohos:height= "match_content"  
  25. ohos:width= "match_parent"  
  26. ohos:text= "フォロー"  
  27. ohos:text_size= "$float:normal_text_size_15"  
  28. />
  29. </方向レイアウト>
  30.  
  31. <方向レイアウト
  32. ohos:height= "match_content"  
  33. ohos:width= "match_content"  
  34. ohos:weight= "1" >
  35.  
  36. <テキスト
  37. ohos:id= "$+id:tag_support"  
  38. ohos:height= "match_content"  
  39. ohos:width= "match_content"  
  40. ohos:text= "推奨"  
  41. ohos:text_size= "$float:normal_text_size_15"  
  42. />
  43. </方向レイアウト>
  44.  
  45. <方向レイアウト
  46. ohos:height= "match_content"  
  47. ohos:width= "match_content"  
  48. ohos:weight= "1" >
  49.  
  50. <テキスト
  51. ohos:id= "$+id:tag_movie"  
  52. ohos:height= "match_content"  
  53. ohos:width= "match_content"  
  54. ohos:element_end= "$id:favorite"  
  55. ohos:text= "映画"  
  56. ohos:text_color= "#1C6AE9"  
  57. ohos:text_size= "$float:normal_text_size_15"  
  58. ohos:text_weight= "600" />
  59.  
  60. <コンポーネント
  61. ohos:id= "$+id:デバイスアイテム区切り線"  
  62. ohos:height= "2vp"  
  63. ohos:width= "30vp"  
  64. ohos:background_element= "$graphic:list_divider" />
  65. </方向レイアウト>
  66.  
  67. <方向レイアウト
  68. ohos:height= "match_content"  
  69. ohos:width= "match_content"  
  70. ohos:weight= "1" >
  71.  
  72. <テキスト
  73. ohos:id= "$+id:tag_live"  
  74. ohos:height= "match_content"  
  75. ohos:width= "match_parent"  
  76. ohos:text= "ライブ放送"  
  77. ohos:text_size= "$float:normal_text_size_15"  
  78. />
  79. </方向レイアウト>
  80.  
  81. <方向レイアウト
  82. ohos:height= "match_content"  
  83. ohos:width= "match_content"  
  84. ohos:weight= "1" >
  85.  
  86. <テキスト
  87. ohos:id= "$+id:tag_tv"  
  88. ohos:height= "match_content"  
  89. ohos:width= "match_parent"  
  90. ohos:text= "テレビ"  
  91. ohos:text_size= "$float:normal_text_size_15"  
  92. />
  93. </方向レイアウト>
  94.  
  95. </方向レイアウト>
  96.  
  97. <スクロールビュー
  98. ohos:height= "match_parent"  
  99. ohos:width= "match_parent"  
  100. ohos:id= "$+id:video_list_scroll" >
  101.  
  102. <リストコンテナ
  103. ohos:id= "$+id:videos_container"  
  104. ohos:height= "match_parent"  
  105. ohos:width= "match_parent" >
  106.  
  107. </リストコンテナ>
  108. </スクロールビュー>
  109. </方向レイアウト>

2.2.Javaコード

ユーザー固有の権限承認を申請する MainAbility.java

  1. /**
  2. *プログラムメインインターフェースへのエントリ
  3. */
  4. パブリッククラスMainAbilityはAbilityを拡張します{
  5. プライベート静的最終int REQUEST_CODE = 1;
  6.  
  7. // メディアの読み取りと書き込みの権限
  8. プライベート最終String[]権限リスト
  9. = 新しい文字列[]{ "ohos.permission.READ_MEDIA" "ohos.permission.WRITE_MEDIA" };
  10.  
  11. @オーバーライド
  12. パブリックvoid onStart(インテント インテント) {
  13. super.onStart(インテント);
  14. UIContent をスーパーに設定します。
  15. super.setMainRoute(MainAbilitySlice.class.getName());
  16.  
  17. // 承認を申請する
  18. 権限を確認します();
  19. }
  20. プライベートvoid verifyPermissions() {
  21. (文字列permissionList:permissionLists) {
  22. int結果 = verifySelfPermission(permissionList);
  23. (結果がIBundleManager.PERMISSION_GRANTEDではない場合){
  24. ユーザーからの権限の要求(権限リスト、REQUEST_CODE)。
  25. }
  26. }
  27. }
  28. ...

VideoInfoService を通じて videos.json ファイルを読み取り、ビデオ コンテナ リスト MainAbilitySlice.java を初期化します。

  1. パブリッククラス MainAbilitySlice は AbilitySlice を拡張します {
  2. 公共 静的最終 HiLogLabel LABEL = new HiLogLabel(0, 0, "=>MainAbilitySlice" );
  3. プライベート VideoInfoService ビデオサービス;
  4.  
  5. @オーバーライド
  6. 保護されたvoid onStart(インテントインテント) {
  7. super.onStart(インテント);
  8. //ビデオプレーヤーページを読み込む
  9. UIContent をスーパーに設定します。
  10. ビデオコンテナを初期化します。
  11.  
  12. }
  13.  
  14. /**
  15. * シミュレーションデータ
  16. * ビデオコンテナリストを初期化する
  17. */
  18. プライベートvoid initVideoContainer() {
  19. HiLog.debug(LABEL、 "initVideoContainer" );
  20. List<VideoModel> videos = 新しい ArrayList<>();
  21. //ビデオ情報サービスを取得する
  22. ビデオサービス = 新しいビデオ情報サービス (getContext());
  23.  
  24. ( int i = 0; i < 7; i++) {
  25. ビデオモデルビデオ = 新しいビデオモデル();
  26. ビデオ.setComments(新しいRandom().nextInt(1000));
  27. ビデオをお気に入りに設定する(新しい Random().nextInt(1000));
  28. ビデオ.setGreats(新しい Random().nextInt(10000));
  29. ビデオ情報 videoInfo = videoService.getVideoInfoByIndex(i);
  30. ビデオ情報.setIndex(i);
  31. ビデオ情報を設定します。
  32. videos.add (ビデオ);
  33. }
  34.  
  35. ListContainer listContainer = (ListContainer) findComponentById(ResourceTable.Id_videos_container);
  36. //コンテナはデータプロバイダをバインドします
  37. VideoItemProvider プロバイダー = 新しい VideoItemProvider(this, videos, this);
  38. listContainer.setItemProvider(プロバイダー);
  39. }
  40. }

ビデオ コンテナ リスト データ プロバイダー VideoItemProvider.java

  1. パブリッククラス VideoItemProvider は BaseItemProvider を拡張します {
  2. プライベート静的最終 HiLogLabel LABEL = new HiLogLabel(0, 0, "=>VideoItemProvider" );
  3. プライベート最終コンテキストコンテキスト;
  4. プライベート最終リスト<VideoModel> リスト;
  5. プライベートAbilitySlice abilitySlice;
  6. //現在の再生ビデオ解像度インデックス
  7. プライベートint現在の再生解像度インデックス = 0;
  8.  
  9. /**
  10. * 初期化
  11. */
  12. パブリックVideoItemProvider(コンテキスト context、List<VideoModel> list、AbilitySlice abilitySlice) {
  13. HiLog.debug(LABEL、 「VideoItemProvider」 );
  14. this.context = コンテキスト;
  15. this.list = リスト;
  16. 能力スライス = 能力スライス;
  17. }
  18.  
  19. パブリックコンテキスト getContext() {
  20. コンテキストを返します
  21. }
  22.  
  23. @オーバーライド
  24. 公共 整数getCount() {
  25. 戻りリスト == null ? 0 : リスト。サイズ();
  26. }
  27.  
  28. @オーバーライド
  29. パブリックオブジェクトgetItem( int位置){
  30. if (list != null && position >= 0 && position < list.size ( )) {
  31. リストを返します。get(position);
  32. }
  33. 新しい VideoModel()を返します
  34. }
  35.  
  36. @オーバーライド
  37. パブリックlong getItemId( int位置) {
  38. 位置を戻す
  39. }
  40.  
  41. @オーバーライド
  42. パブリックコンポーネント getComponent( int position, Component convertComponent, ComponentContainer componentContainer) {
  43. HiLog.debug(LABEL、 "getComponent:" + convertComponent + "," + componentContainer);
  44. 最終コンポーネント cpt;
  45. 最終的な VideoPlayerView プレーヤー。
  46.  
  47. ビデオモデル項目 = list.get(位置);
  48.  
  49. // 初めてコンポーネントを取得する
  50. (convertComponent == null )の場合{
  51. cpt = LayoutScatter.getInstance(context).parse(ResourceTable.Layout_hm_sample_ability_video_box, componentContainer, false );
  52. プレーヤー = (VideoPlayerView) cpt.findComponentById(ResourceTable.Id_video_view);
  53. // プレーヤーコンポーネントを初期化する
  54. プレーヤーを初期化します(プレーヤー、アイテム.getVideoInfo());
  55. }それ以外{
  56. //ScrollView がスライドするときにも getComponent メソッドが呼び出されます。
  57. cpt = convertComponent;
  58. プレーヤー = (VideoPlayerView) cpt.findComponentById(ResourceTable.Id_video_view);
  59. }
  60.  
  61. //他のコンポーネントの値を設定する
  62. テキスト text_favorites = (テキスト) cpt.findComponentById(ResourceTable.Id_text_favorites);
  63. テキスト text_comments = (テキスト) cpt.findComponentById(ResourceTable.Id_text_comments);
  64. テキスト text_greats = (テキスト) cpt.findComponentById(ResourceTable.Id_text_greats);
  65. //setTextのパラメータ型は文字列であることに注意してください
  66. text_favorites.setText(item.getFavorites() + "" );
  67. text_comments.setText(item.getComments() + "" );
  68. text_greats.setText(item.getGreats() + "" );
  69. //コメントアイコンをクリックすると、現在のプレーヤーが停止し、VideoPlayAbility のスライス ページが開きます。
  70. 画像コメントImage = (画像) cpt.findComponentById(ResourceTable.Id_image_comment);
  71. commentImage.setClickedListener(コンポーネント -> {
  72. //再生を停止、リストページのプレーヤー
  73. プレーヤーの再生を停止します。
  74. //コメント付きの再生ページを開く
  75. ビデオ再生の詳細を開始します(アイテム、プレーヤー)。
  76. });
  77. cptを返します
  78. }
  79.  
  80. /**
  81. * プレーヤーを初期化する
  82. */
  83. プライベート void initPlayer(VideoPlayerView プレーヤー、VideoInfo ビデオ情報) {
  84. HiLog.debug(LABEL、 "initPlayer" );
  85. if (プレイヤー != null ) {
  86. //ビデオパス
  87. 文字列パス = videoInfo
  88. .getResolutions()
  89. .get(現在の再生解像度インデックス)
  90. .getUrl();
  91. HiLog.debug(LABEL, "path:" + パス);
  92.  
  93. //ビデオの説明
  94. 文字列 videoDesc = videoInfo.getVideoDesc();
  95. HiLog.debug(LABEL、 "videoDesc = " + videoDesc + " パス = " + パス);
  96.  
  97. パスがnull場合
  98. //パスと名前を設定する
  99. player.setVideoPathAndTitle(パス、ビデオの説明);
  100. //ダブルクリックして再生または一時停止します
  101. プレイヤー.setDoubleClickedListener(
  102. コンポーネント -> {
  103. HiLog.debug(LABEL, "VideoPlayView ダブルクリック イベント" );
  104. (プレイヤーが再生中の場合){
  105. プレーヤーを一時停止します。
  106. }それ以外{
  107. プレイヤーを起動します。
  108. }
  109. });
  110. //再生エラーを監視する
  111. プレイヤー.setErrorListener()
  112. (エラータイプ、エラーコード) -> {
  113. ToastDialog トースト = 新しい ToastDialog(getContext());
  114. スイッチ (エラータイプ) {
  115. HmPlayerAdapter.ERROR_LOADING_RESOURCEの場合:
  116. トースト.setText()
  117. AppUtil.getStringResource() 関数は、
  118. getContext()、ResourceTable.String_media_file_loading_error));
  119. 壊す;
  120. HmPlayerAdapter.ERROR_INVALID_OPERATIONの場合:
  121. トースト.setText()
  122. AppUtil.getStringResource() 関数は、
  123. getContext()、ResourceTable.String_invalid_operation));
  124. 壊す;
  125. デフォルト
  126. トースト.setText()
  127. AppUtil.getStringResource() 関数は、
  128. getContext()、ResourceTable.String_undefined_error_type));
  129. 壊す;
  130. }
  131. 能力スライス.getUITaskDispatcher().asyncDispatch(toast::show);
  132. });
  133. }
  134. //コアコンポーネント、再生時間進行スライダーを追加
  135. コアコンポーネントを追加します(プレーヤー)。
  136. //カスタムコンポーネントを追加する
  137. HiLog.debug(LABEL、 「initPlayer 終了」 );
  138. }
  139. }
  140.  
  141. /**
  142. * 再生ボタン、エピソード、進行状況バーなどのコア コンポーネントを追加します。
  143. *再生ボタンシークバーなどのコアコンポーネントの追加
  144. */
  145. プライベートvoid addCoreComponent(VideoPlayerViewプレーヤー) {
  146. HiLog.debug(LABEL、 「CoreComponent を追加」 );
  147. //再生ボタンコンポーネントを追加する
  148. player.addPlaybackButton(新しいVideoPlayerPlaybackButton(getContext()), VideoBoxArea.BOTTOM);
  149. //再生進行状況バーコンポーネントを追加する
  150. プレイヤー.addSeekBar(
  151. 新しいビデオプレーヤースライダー(getContext())、
  152. ビデオボックスエリア.BOTTOM、
  153. ( int ) AppUtil.getFloatResource(getContext(), ResourceTable.Float_normal_margin_24));
  154. }
  155.  
  156. /**
  157. * ビデオ再生の詳細ページを開く
  158. */
  159. プライベート void startVideoPlayDetail(ビデオモデル項目、ビデオプレーヤービュープレーヤー) {
  160. HiLog.debug(LABEL、 「startVideoPlayDetail」 );
  161. //コメント再生ページを開始する
  162. インテントのintentService = 新しいIntent();
  163. //ビデオコメントを再生する
  164. intentService.setParam( "コメント" ,String.valueOf(item.getComments()));
  165.  
  166. ビデオ情報 videoInfo=item.getVideoInfo();
  167.  
  168. ビデオ情報! = nullの場合{
  169. // プレイリストで再生するビデオのインデックス
  170. intentService.setParam( "現在の再生インデックス" 、 videoInfo.getIndex());
  171. //ビデオ再生時の解像度インデックス
  172. intentService.setParam( "現在の再生解像度インデックス" , 現在の再生解像度インデックス);
  173. //現在のビデオ再生位置
  174. intentService.setParam(RemoteConstant.INTENT_PARAM_REMOTE_START_POSITION,( int )player.getCurrentPosition());
  175. HiLog.debug(LABEL、RemoteConstant.INTENT_PARAM_REMOTE_START_POSITION+ ":" +player.getCurrentPosition());
  176. }
  177. 操作操作 =
  178. 新しい Intent.OperationBuilder()
  179. .withDeviceId( "" )
  180. .withBundleName(abilitySlice.getBundleName())
  181. .withAbilityName(ビデオ再生能力.クラス)
  182. 。建てる();
  183. 操作を実装します。
  184. 能力スライスを開始します。(intentService);
  185. }
  186. }

3. コメント機能を実装する

3.1.ページレイアウト、コメントリストレイアウトページ hm_sample_ability_video_comments.xml

StackLayout、DependentLayoutレイアウトコンポーネントとListContainer、TextField、Textコンポーネントを使用する

  1. ...
  2. <! -- ツールバー -->  
  3. <依存レイアウト
  4. ohos:id= "$+id:comments_bar"  
  5. ohos:height= "40vp"  
  6. ohos:width= "match_parent"  
  7. ohos:background_element= "#FFDDDADA"  
  8. ohos:layout_alignment= "上"  
  9. ohos:top_margin= "250vp" >
  10.  
  11. <方向レイアウト
  12. ohos:id= "$+id:お気に入り"  
  13. ohos:height= "match_parent"  
  14. ohos:width= "match_content"  
  15. ohos:align_parent_left= "true"  
  16. ohos:left_margin= "10vp"  
  17. ohos:orientation= "水平"  
  18. ohos:padding= "4vp" >
  19.  
  20. <テキスト
  21. ohos:id= "$+id:text_favorites"  
  22. ohos:height= "match_parent"  
  23. ohos:width= "match_parent"  
  24. ohos:left_padding= "5vp"  
  25. ohos:text= "はじめに"  
  26. ohos:text_size= "16fp"  
  27. ohos:text_weight= "600" >
  28.  
  29. </テキスト>
  30. </方向レイアウト>
  31.  
  32. <方向レイアウト
  33. ohos:id= "$+id:コメント"  
  34. ohos:height= "match_parent"  
  35. ohos:width= "match_content"  
  36. ohos:end_of= "$id:favorite"  
  37. ohos:left_margin= "20vp"  
  38. ohos:orientation= "水平"  
  39. ohos:padding= "4vp" >
  40.  
  41. <テキスト
  42. ohos:id= "$+id:テキストコメント"  
  43. ohos:height= "match_parent"  
  44. ohos:width= "match_content"  
  45. ohos:left_padding= "5vp"  
  46. ohos:text= "コメント"  
  47. ohos:テキストの重み= "600"  
  48. ohos:text_size= "16fp" >
  49. </テキスト>
  50.  
  51. <テキスト
  52. ohos:id= "$+id:text_comments"  
  53. ohos:height= "match_parent"  
  54. ohos:width= "match_content"  
  55. ohos:left_padding= "5vp"  
  56. ohos:text= "1161"  
  57. ohos:テキストの重み= "600"  
  58. ohos:text_size= "16fp" >
  59. </テキスト>
  60. </方向レイアウト>
  61.  
  62. </依存レイアウト>
  63.  
  64. <! -- コメント一覧 -->  
  65. <依存レイアウト
  66. ohos:id= "$+id:comments_view"  
  67. ohos:height= "match_parent"  
  68. ohos:width= "match_parent"  
  69. ohos:background_element= "#FFFDFFFF"  
  70. ohos:top_margin= "290vp" >
  71.  
  72. <リストコンテナ
  73. ohos:id= "$+id:comments_container"  
  74. ohos:height= "match_parent"  
  75. ohos:width= "match_parent"  
  76. ohos:layout_alignment= "水平中央"  
  77. ohos:orientation= "垂直" />
  78.  
  79. </依存レイアウト>
  80.  
  81. <! -- コメントダイアログ -->  
  82. <依存レイアウト
  83. ohos:id= "$+id:comments_dialog"  
  84. ohos:height= "70vp"  
  85. ohos:width= "match_parent"  
  86. ohos:background_element= "#FFF4F4F8"  
  87. ohos:layout_alignment= "下"  
  88. ohos:padding= "5vp" >
  89.  
  90. <テキストフィールド
  91. ohos:id= "$+id:comment_tf"  
  92. ohos:height= "match_parent"  
  93. ohos:width= "match_parent"  
  94. ohos:hint= 「フレンドリーなコメントを送信」  
  95. ohos:right_padding= "60vp"  
  96. ohos:text_size= "16vp" ></テキストフィールド>
  97.  
  98. <テキスト
  99. ohos:id= "$+id:sent_comment"  
  100. ohos:height= "match_parent"  
  101. ohos:width= "60vp"  
  102. ohos:align_parent_right= "true"  
  103. ohos:background_element= "#FF7ECCCCF"  
  104. ohos:text= "送信"  
  105. ohos:input_enter_key_type= "enter_key_type_send"  
  106. ohos:text_alignment= "中央"  
  107. ohos:text_size= "16vp" >
  108. </テキスト>
  109. </依存レイアウト>

3.2.ページレイアウト、単一コメントコンポーネントレイアウト comments_item.xml

DependentLayout、DirectionalLayoutレイアウトコンポーネントとImage、Component、Textコンポーネントを使用する

  1. <?xml バージョン = "1.0"エンコーディング = "utf-8" ?>
  2. <依存レイアウト
  3. xmlns:ohos= "http://schemas.huawei.com/res/ohos"  
  4. ohos:height= "match_content"  
  5. ohos:width= "match_parent" >
  6.  
  7. <画像
  8. ohos:id= "$+id:header_item_icon"  
  9. ohos:height= "50vp"  
  10. ohos:width= "50vp"  
  11. ohos:end_margin= "13vp"  
  12. ohos:left_margin= "5vp"  
  13. ohos:image_src= "$media:ic_header"  
  14. ohos:scale_mode= "ストレッチ"  
  15. ohos:top_margin= "13vp" />
  16.  
  17. <方向レイアウト
  18. ohos:id= "$+id:コメント項目コンテンツ"  
  19. ohos:height= "match_content"  
  20. ohos:width= "250vp"  
  21. ohos:bottom_padding= "5vp"  
  22. ohos:orientation= "垂直"  
  23. ohos:right_of= "$id:header_item_icon" >
  24.  
  25. <テキスト
  26. ohos:id= "$+id:comment_uname"  
  27. ohos:height= "match_content"  
  28. ohos:width= "160vp"  
  29. ohos:layout_alignment= "垂直中央"  
  30. ohos:padding= "5vp"  
  31. ohos:text= 「私は魚です」  
  32. ohos:text_color= "#FFA09E9E"  
  33. ohos:text_size= "16vp" />
  34.  
  35. <テキスト
  36. ohos:id= "$+id:comment_content"  
  37. ohos:height= "match_content"  
  38. ohos:width= "match_parent"  
  39. ohos:layout_alignment= "垂直中央"  
  40. ohos:multiple_lines = "true"  
  41. ohos:padding= "5vp"  
  42. ohos:text= "これはコメントです"  
  43. ohos:text_color= "$color:default_black_color"  
  44. ohos:text_size= "16vp" />
  45.  
  46. <テキスト
  47. ohos:id= "$+id:comment_date"  
  48. ohos:height= "match_content"  
  49. ohos:width= "match_content"  
  50. ohos:layout_alignment= "垂直中央"  
  51. ohos:padding= "5vp"  
  52. ohos:text= "2日前"  
  53. ohos:text_color= "#FFA09E9E"  
  54. ohos:text_size= "16vp" />
  55.  
  56. </方向レイアウト>
  57.  
  58. <方向レイアウト
  59. ohos:height= "match_content"  
  60. ohos:width= "match_content"  
  61. ohos:id= "$+id:goods_view"  
  62. ohos:align_parent_right= "true"  
  63. ohos:padding= "5vp"  
  64. ohos:right_margin= "5vp"  
  65. ohos:right_of= "$id:comment_item_content" >
  66.  
  67. <画像
  68. ohos:id= "$+id:good_icon"  
  69. ohos:height= "25vp"  
  70. ohos:width= "match_parent"  
  71. ohos:image_src= "$media:ic_great"  
  72. ohos:scale_mode= "ストレッチ"  
  73. ohos:top_margin= "13vp"  
  74. />
  75.  
  76. <テキスト
  77. ohos:id= "$+id:comment_goods"  
  78. ohos:height= "match_content"  
  79. ohos:width= "match_parent"  
  80. ohos:text= "42"  
  81. ohos:text_alignment= "中央"  
  82. ohos:text_color= "#FFA09E9E"  
  83. ohos:text_size= "12vp" ></Text>
  84.  
  85. </方向レイアウト>
  86.  
  87. <方向レイアウト
  88. ohos:height= "match_content"  
  89. ohos:width= "match_parent"  
  90. ohos:align_bottom= "$id:comment_item_content" >
  91. <コンポーネント
  92. ohos:id= "$+id:デバイスアイテム区切り線"  
  93. ohos:height= "1vp"  
  94. ohos:width= "match_parent"  
  95. ohos:background_element= "$graphic:list_divider" />
  96. </方向レイアウト>
  97. </依存レイアウト>

3.3. Javaコード

コメントリストプロバイダクラス CommentItemProvider.java を作成する

  1. パブリッククラスCommentItemProviderはBaseItemProviderを拡張します。
  2. プライベート静的最終 HiLogLabel LABEL = new HiLogLabel(0, 0, "=>CommentsItemProvider" );
  3. プライベート最終コンテキストコンテキスト;
  4. プライベート最終リスト<CommentModel> リスト;
  5. プライベートAbilitySlice abilitySlice;
  6.  
  7. /**
  8. * 初期化
  9. */
  10. パブリックCommentItemProvider(コンテキスト コンテキスト、List<CommentModel> リスト、AbilitySlice スライス) {
  11. this.context = コンテキスト;
  12. this.list = リスト;
  13. this.abilitySlice = スライス;
  14. }
  15.  
  16. @オーバーライド
  17. 公共 整数getCount() {
  18. 戻りリスト == null ? 0 : リスト。サイズ();
  19. }
  20.  
  21. @オーバーライド
  22. パブリックオブジェクトgetItem( int位置){
  23. if (list != null && position >= 0 && position < list.size ( )) {
  24. リストを返します。get(position);
  25. }
  26. 新しい CommentModel()を返します
  27. }
  28.  
  29. パブリックvoid addComment(コメントモデルコメント) {
  30. if (コメント == null )戻り値;
  31.  
  32. リストに0、コメントを追加します
  33. DataSetItemInserted を通知します(0);
  34. }
  35.  
  36. @オーバーライド
  37. パブリックlong getItemId( int位置) {
  38. 位置を戻す
  39. }
  40.  
  41. @オーバーライド
  42. パブリックコンポーネント getComponent( int position, Component convertComponent, ComponentContainer componentContainer) {
  43. 最終コンポーネント cpt;
  44. (convertComponent == null )の場合{
  45. cpt = LayoutScatter.getInstance(context).parse(ResourceTable.Layout_comments_item, null false );
  46. }それ以外{
  47. cpt = convertComponent;
  48. }
  49.  
  50.  
  51. コメントモデル item = list.get(position);
  52. //コメント投稿者のアバター
  53. 画像ヘッダーアイコン = (画像) cpt.findComponentById(ResourceTable.Id_header_item_icon);
  54. headerIcon.setPixelMap(ResourceTable.Media_ic_header);
  55.  
  56. // 解説者の名前
  57. テキスト uName = (テキスト) cpt.findComponentById(ResourceTable.Id_comment_uname);
  58. uName.setText(item.getuName());
  59.  
  60. //コメント内容
  61. テキスト uContent = (テキスト) cpt.findComponentById(ResourceTable.Id_comment_content);
  62. uContent.setText(item.getCommentContent());
  63. //クリックイベント
  64. uContent.setClickedListener(コンポーネント -> {
  65. DependentLayout の commentDialog を (DependentLayout) abilitySlice.findComponentById(ResourceTable.Id_comments_dialog);
  66. コメントダイアログの getVisibility() が Component.VISIBLE の場合 {
  67. コメントダイアログの可視性を設定します(Component.VISIBLE);
  68. }
  69. テキストフィールドコメント = (テキストフィールド) abilitySlice.findComponentById(ResourceTable.Id_comment_tf);
  70. コメント.setText( "" );
  71. comment.setHint( "返信 " + item.getuName() + ":" );
  72. コメントをリクエストフォーカスします。
  73. });
  74.  
  75.  
  76. // コメントの日付
  77. テキスト uDate = (テキスト) cpt.findComponentById(ResourceTable.Id_comment_date);
  78. uDate.setText(item.getCommentDate());
  79.  
  80. //いいねの数
  81. テキスト商品 = (テキスト) cpt.findComponentById(ResourceTable.Id_comment_goods);
  82. HiLog.debug(LABEL, "goods:" + item.getCommentGoods());
  83. goods.setText(item.getCommentGoods() + "" );
  84.  
  85. //のように
  86. 画像 greatImage = (画像) cpt.findComponentById(ResourceTable.Id_good_icon);
  87. //いいねして追加する
  88. greatImage.setClickedListener(component1 -> {
  89. greatImage.setPixelMap(ResourceTable.Media_ic_great_red);
  90. goods.setTextColor(Color.RED);
  91. goods.setText((item.getCommentGoods() + 1) + "" );
  92. });
  93.  
  94. if (位置 == リスト.サイズ() - 1) {
  95. コンポーネント ディバイダ = cpt.findComponentById(ResourceTable.Id_device_item_divider);
  96. 区切り線.setVisibility(Component.INVISIBLE);
  97. }
  98.  
  99. cptを返します
  100. }
  101. }

コメントリストとコメント関数VideoPlayAbilitySlice.javaを初期化する

  1. /**
  2. * コメントリストを初期化する
  3. */
  4. プライベートvoid initComments(AbilitySliceスライス) {
  5. //コメントリスト、クリックリスナーイベントの設定、データの受け渡し
  6. ListContainer listContainer = (ListContainer) findComponentById(ResourceTable.Id_comments_container);
  7. リスト<CommentModel> リスト = 新しい ArrayList<>();
  8. ( int i = 0; i < 10; i++) {
  9. コメントモデル obj = 新しいコメントモデル();
  10. obj.setuId(i + "" );
  11. obj.setuName(getRandomUname());
  12. obj.setCommentContent(getRandomText());
  13. obj.setCommentDate( "たった今" );
  14. obj.setCommentGoods(新しいRandom().nextInt(100));
  15. リストを追加します
  16. }
  17.  
  18. //コンテナはデータプロバイダをバインドします
  19. プロバイダー = 新しい CommentItemProvider(this、リスト、スライス);
  20. listContainer.setItemProvider(プロバイダー);
  21.  
  22. コンポーネント sendComment = findComponentById(ResourceTable.Id_sent_comment);
  23. テキストフィールドコメント = (テキストフィールド) findComponentById(ResourceTable.Id_comment_tf);
  24.  
  25. //コメント関数
  26. sendComment.setClickedListener(コンポーネント -> {
  27. if (comment.getText().isEmpty()) {
  28. new ToastDialog(getContext()).setText( "コメントは空にできません" ).show();
  29. 戻る;
  30. }
  31. コメントモデル モデル = 新しいコメントモデル();
  32. モデルにコメントコンテンツを設定します。
  33. モデル.setuId(UUID.randomUUID().toString());
  34. model.setuName( "ロボット" );
  35. model.setCommentDate( "たった今" );
  36.  
  37. //データプロバイダに追加
  38. プロバイダー.addComment(モデル);
  39.  
  40. コメント.setText( "" );
  41. //コメント +1
  42. commentsText.setText(String.valueOf( Integer.valueOf (comments) + 1));
  43.  
  44. new ToastDialog(getContext()).setText( "コメント成功" ).show();
  45.  
  46. });
  47. }

問題の概要

1. プレーヤーリストをスライドすると、再生コンポーネントが繰り返し追加されます

2. コメント機能、入力方法に応じて入力ボックスの位置を調整(改善予定)

3. プレイリストがスライドする際、ScrollViewの位置に応じて動画の再生を制御する(改善予定)

4. プレイリスト内の動画の再生進行を再生詳細ページに同期する問題(改善予定)

記事に関連する添付ファイルをダウンロードするには、以下のリンクをクリックしてください。

https://harmonyos..com/resource/1338

詳細については、以下をご覧ください。

51CTOとHuaweiが共同で構築したHongmengテクノロジーコミュニティ

https://harmonyos..com

<<:  アリババクラウド社長張建鋒氏:新たなコンピューティングアーキテクチャが形になりつつある

>>:  クラウド通話ソリューションは本当に優れた柔軟性と制御性を提供するのでしょうか?

推薦する

分析データの氾濫を克服するクラウドコンピューティングの役割

情報インフラストラクチャを近代化する戦略の一環として、企業はクラウド コンピューティングをより有効に...

老米農家の考え:ドメイン名投資にはスキルが必要

友人の中には、ドメイン名投資を一種の投機産業とみなす人もいます。私は個人的にこの言葉を否定しません。...

映画ウェブサイトを最適化する方法を教える実践的な経験

最近、多くの人が自分の映画のウェブサイトを作りたいのですが、それをどのように宣伝したらよいか分かりま...

Moments 広告の新しい @friend 機能はどのような効果をもたらすのでしょうか?

モーメントの@Friendsの新機能が話題になりました。 WeChat広告アシスタント公式アカウント...

地雷原に潜む電子商取引の5つの主要な違法リスク 電子商取引に関わるウェブマスターがしてはいけないこと

電子商取引は金鉱ですが、探索すべきではない地雷原もいくつかあります。この記事では、電子商取引における...

百度の重みを高めるために新しいサイトが注意すべきいくつかのポイントを簡単に分析する

現在、中国の検索エンジン業界は百度が独占しています。特にGoogleが中国市場から撤退した後、百度は...

企業がプラットフォーム・アズ・ア・サービス (PaaS) を選択すべき理由

Platform as a Service (PaaS) とは、アプリケーションの開発、実行、管理の...

ぜひ受け取ってください! Android マーケットの ASO 最適化チュートリアルの包括的なガイド!

今日は簡単なまとめ、つまり 8 つの主要な Android マーケットの ASO 最適化について簡単...

あなたのウェブサイトをインターネット上の他のウェブサイトよりも目立たせるための3つの黄金律

近年の SEO の人気は誰の目にも明らかです。大規模な Web サイトはもちろんのこと、雨後の筍のよ...

ウェブデザインでタイポグラフィを使用して情報を伝達する

この記事の著者@刘明はWeibo UDCデザインセンターのメンバーです。この記事から、優れた組版デザ...

shockhosting: 高性能 VPS と大容量ハードディスク VPS が 50% オフ、オプションのデータセンター 5 つ

shockhosting は正式にデータセンターを 5 か所に拡大しました。公式は VPS サービス...

360 百度検索戦争は SEO に影響を与えるか?

この間、Baidu と 360 Search の間での検索戦争は静かに進行してきました。Baidu ...

Youpin.com の視点から見た SEO の考え方と戦略

待望の Youpin.com がついにオンラインになりました。ほぼ 1 か月間、黙々と作業してきまし...

cloudsilk: 日本の VPS を販売中、必須の 3 ネットワークの日本の CMI、360 元/年 - 512M メモリ/1 コア/10g SSD/800G トラフィック

Cloudsilk は、中国本土向けに最適化された BGP ネットワーク、具体的には 3 つのネット...

テンセントクラウドデータベースが国信証券に上陸、1日平均取引量が数十億に上る

3月7日、テンセントクラウドデータベースTDSQLが国森証券の業務システムに導入され、システムが3ヶ...