@quicktvui/ai 1.0.9 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (345) hide show
  1. package/package.json +1 -1
  2. package/rules/.clinerules +1 -0
  3. package/rules/.cursorrules +1 -0
  4. package/rules/.github/copilot-instructions.md +1 -0
  5. package/rules/.source/hellotv/PROJECT-README.md +52 -0
  6. package/rules/.source/hellotv/app.d.ts +11 -0
  7. package/rules/.source/hellotv/package.json +146 -0
  8. package/rules/.source/hellotv/scripts/build-apk.ts +12 -0
  9. package/rules/.source/hellotv/scripts/build.ts +35 -0
  10. package/rules/.source/hellotv/scripts/dev.ts +12 -0
  11. package/rules/.source/hellotv/scripts/pack.ts +24 -0
  12. package/rules/.source/hellotv/scripts/process.ts +37 -0
  13. package/rules/.source/hellotv/scripts/quicktvui-webpack.android.ts +187 -0
  14. package/rules/.source/hellotv/scripts/quicktvui-webpack.dev.ts +147 -0
  15. package/rules/.source/hellotv/scripts/webpack-watch.ts +24 -0
  16. package/rules/.source/hellotv/src/App.vue +192 -0
  17. package/rules/.source/hellotv/src/api/my/index.ts +198 -0
  18. package/rules/.source/hellotv/src/api/user/impl-user.ts +52 -0
  19. package/rules/.source/hellotv/src/api/user/qr-code-mock.ts +2 -0
  20. package/rules/.source/hellotv/src/api/user/request-url.ts +11 -0
  21. package/rules/.source/hellotv/src/api/user/user-manager.ts +258 -0
  22. package/rules/.source/hellotv/src/app.scss +8 -0
  23. package/rules/.source/hellotv/src/assets/component-media/ic_media_btn_pause.png +0 -0
  24. package/rules/.source/hellotv/src/assets/component-media/ic_media_btn_play.png +0 -0
  25. package/rules/.source/hellotv/src/assets/component-media/ic_media_menu_setting_focused.png +0 -0
  26. package/rules/.source/hellotv/src/assets/component-media/ic_media_menu_setting_normal.png +0 -0
  27. package/rules/.source/hellotv/src/assets/component-media/ic_media_menu_xuanji_focused.png +0 -0
  28. package/rules/.source/hellotv/src/assets/component-media/ic_media_menu_xuanji_normal.png +0 -0
  29. package/rules/.source/hellotv/src/assets/component-media/ic_media_ok_focused.png +0 -0
  30. package/rules/.source/hellotv/src/assets/component-media/ic_media_ok_selected.png +0 -0
  31. package/rules/.source/hellotv/src/assets/detail/ic_collect_focused.png +0 -0
  32. package/rules/.source/hellotv/src/assets/detail/ic_collect_normal.png +0 -0
  33. package/rules/.source/hellotv/src/assets/detail/ic_full_focused.png +0 -0
  34. package/rules/.source/hellotv/src/assets/detail/ic_full_normal.png +0 -0
  35. package/rules/.source/hellotv/src/assets/detail/ic_info_focused.png +0 -0
  36. package/rules/.source/hellotv/src/assets/detail/ic_info_normal.png +0 -0
  37. package/rules/.source/hellotv/src/assets/detail/ic_introduction_bg.png +0 -0
  38. package/rules/.source/hellotv/src/assets/detail/ic_media_list_item_normal.png +0 -0
  39. package/rules/.source/hellotv/src/assets/detail/ic_media_list_item_selected.png +0 -0
  40. package/rules/.source/hellotv/src/assets/detail/ic_media_player_pause.png +0 -0
  41. package/rules/.source/hellotv/src/assets/detail/ic_media_player_placeholder.png +0 -0
  42. package/rules/.source/hellotv/src/assets/detail/ic_media_player_play.png +0 -0
  43. package/rules/.source/hellotv/src/assets/detail/ic_vip_focused.png +0 -0
  44. package/rules/.source/hellotv/src/assets/detail/ic_vip_normal.png +0 -0
  45. package/rules/.source/hellotv/src/assets/detail/skeleton.png +0 -0
  46. package/rules/.source/hellotv/src/assets/filter/ic_empty.png +0 -0
  47. package/rules/.source/hellotv/src/assets/filter/ic_filter_focused.png +0 -0
  48. package/rules/.source/hellotv/src/assets/filter/ic_filter_normal.png +0 -0
  49. package/rules/.source/hellotv/src/assets/filter/ic_filter_selected.png +0 -0
  50. package/rules/.source/hellotv/src/assets/filter/ic_left_focused.png +0 -0
  51. package/rules/.source/hellotv/src/assets/filter/ic_left_normal.png +0 -0
  52. package/rules/.source/hellotv/src/assets/filter/ic_left_selected.png +0 -0
  53. package/rules/.source/hellotv/src/assets/history/ic_delete.png +0 -0
  54. package/rules/.source/hellotv/src/assets/history/ic_empty.png +0 -0
  55. package/rules/.source/hellotv/src/assets/home/bg_history_item.png +0 -0
  56. package/rules/.source/hellotv/src/assets/home/bg_shadow.png +0 -0
  57. package/rules/.source/hellotv/src/assets/home/ic_4k_logo.png +0 -0
  58. package/rules/.source/hellotv/src/assets/home/ic_4k_play_focused.png +0 -0
  59. package/rules/.source/hellotv/src/assets/home/ic_4k_subtitle_normal.png +0 -0
  60. package/rules/.source/hellotv/src/assets/home/ic_arrow_focused.png +0 -0
  61. package/rules/.source/hellotv/src/assets/home/ic_arrow_normal.png +0 -0
  62. package/rules/.source/hellotv/src/assets/home/ic_exit_corner.png +0 -0
  63. package/rules/.source/hellotv/src/assets/home/ic_exit_header.png +0 -0
  64. package/rules/.source/hellotv/src/assets/home/ic_play.png +0 -0
  65. package/rules/.source/hellotv/src/assets/live/bg-error.png +0 -0
  66. package/rules/.source/hellotv/src/assets/live/bg-play-info.png +0 -0
  67. package/rules/.source/hellotv/src/assets/live/ic-arrow-left.png +0 -0
  68. package/rules/.source/hellotv/src/assets/live/ic-arrow-right.png +0 -0
  69. package/rules/.source/hellotv/src/assets/live/ic-back.png +0 -0
  70. package/rules/.source/hellotv/src/assets/live/ic-change.png +0 -0
  71. package/rules/.source/hellotv/src/assets/live/ic-corner-vip-tips.png +0 -0
  72. package/rules/.source/hellotv/src/assets/live/ic-corner-vip.png +0 -0
  73. package/rules/.source/hellotv/src/assets/live/ic-key.png +0 -0
  74. package/rules/.source/hellotv/src/assets/live/ic-loading-error.png +0 -0
  75. package/rules/.source/hellotv/src/assets/live/ic-menu-ext-arrow.png +0 -0
  76. package/rules/.source/hellotv/src/assets/live/ic-menu-ext.png +0 -0
  77. package/rules/.source/hellotv/src/assets/live/ic-menu.png +0 -0
  78. package/rules/.source/hellotv/src/assets/live/ic-mine.png +0 -0
  79. package/rules/.source/hellotv/src/assets/live/ic-network-error.png +0 -0
  80. package/rules/.source/hellotv/src/assets/live/ic-ok.png +0 -0
  81. package/rules/.source/hellotv/src/assets/live/ic-playing.png +0 -0
  82. package/rules/.source/hellotv/src/assets/live/ic-playing2.png +0 -0
  83. package/rules/.source/hellotv/src/assets/live/ic-playing3.png +0 -0
  84. package/rules/.source/hellotv/src/assets/live/ic-service.png +0 -0
  85. package/rules/.source/hellotv/src/assets/live/ic_live_broadcast.png +0 -0
  86. package/rules/.source/hellotv/src/assets/live/ic_live_broadcast_focused.png +0 -0
  87. package/rules/.source/hellotv/src/assets/my/ic_collect_focus.png +0 -0
  88. package/rules/.source/hellotv/src/assets/my/ic_collect_normal.png +0 -0
  89. package/rules/.source/hellotv/src/assets/my/ic_order_focus.png +0 -0
  90. package/rules/.source/hellotv/src/assets/my/ic_order_normal.png +0 -0
  91. package/rules/.source/hellotv/src/assets/my/ic_record_focus.png +0 -0
  92. package/rules/.source/hellotv/src/assets/my/ic_record_normal.png +0 -0
  93. package/rules/.source/hellotv/src/assets/my/ic_user.png +0 -0
  94. package/rules/.source/hellotv/src/assets/my/ic_vip_normal.png +0 -0
  95. package/rules/.source/hellotv/src/assets/search/ic_back.png +0 -0
  96. package/rules/.source/hellotv/src/assets/search/ic_back_focused.png +0 -0
  97. package/rules/.source/hellotv/src/assets/search/ic_clear.png +0 -0
  98. package/rules/.source/hellotv/src/assets/search/ic_clear_dark.png +0 -0
  99. package/rules/.source/hellotv/src/assets/search/ic_clear_focused.png +0 -0
  100. package/rules/.source/hellotv/src/assets/search/ic_search.png +0 -0
  101. package/rules/.source/hellotv/src/assets/top-view/ic_logo.png +0 -0
  102. package/rules/.source/hellotv/src/assets/top-view/ic_top_search_focused.png +0 -0
  103. package/rules/.source/hellotv/src/assets/top-view/ic_top_search_normal.png +0 -0
  104. package/rules/.source/hellotv/src/assets/top-view/ic_top_user_focused.png +0 -0
  105. package/rules/.source/hellotv/src/assets/top-view/ic_top_user_normal.png +0 -0
  106. package/rules/.source/hellotv/src/components/bg-animation.scss +14 -0
  107. package/rules/.source/hellotv/src/components/bg-animation.vue +176 -0
  108. package/rules/.source/hellotv/src/components/btn-pack-view.vue +172 -0
  109. package/rules/.source/hellotv/src/components/btn-pack.scss +53 -0
  110. package/rules/.source/hellotv/src/components/grid-item-horizontal.vue +203 -0
  111. package/rules/.source/hellotv/src/components/grid-item-vertical.vue +201 -0
  112. package/rules/.source/hellotv/src/components/media/build-data/media-config.ts +4 -0
  113. package/rules/.source/hellotv/src/components/media/build-data/media-control-adapter.ts +378 -0
  114. package/rules/.source/hellotv/src/components/media/build-data/media-imp.ts +44 -0
  115. package/rules/.source/hellotv/src/components/media/view/media-loading-component.vue +50 -0
  116. package/rules/.source/hellotv/src/components/media/view/media-loading-view.vue +154 -0
  117. package/rules/.source/hellotv/src/components/media/view/media-manager-view.vue +931 -0
  118. package/rules/.source/hellotv/src/components/media/view/media-menu-detail-item.vue +42 -0
  119. package/rules/.source/hellotv/src/components/media/view/media-menu-detail-view.vue +109 -0
  120. package/rules/.source/hellotv/src/components/media/view/media-menu-icon-item.vue +43 -0
  121. package/rules/.source/hellotv/src/components/media/view/media-menu-text-item.vue +27 -0
  122. package/rules/.source/hellotv/src/components/media/view/media-menu-view.vue +88 -0
  123. package/rules/.source/hellotv/src/components/media/view/media-player-view.vue +408 -0
  124. package/rules/.source/hellotv/src/components/media/view/scss/media-loading.scss +51 -0
  125. package/rules/.source/hellotv/src/components/media/view/scss/media-manager.scss +79 -0
  126. package/rules/.source/hellotv/src/components/media/view/scss/media-menu.scss +114 -0
  127. package/rules/.source/hellotv/src/components/media/view/scss/media-player.scss +6 -0
  128. package/rules/.source/hellotv/src/components/qt-tabs-component.ts +268 -0
  129. package/rules/.source/hellotv/src/components/qt-ul-component.ts +792 -0
  130. package/rules/.source/hellotv/src/components/top-view.scss +38 -0
  131. package/rules/.source/hellotv/src/components/top-view.vue +114 -0
  132. package/rules/.source/hellotv/src/config/README.md +22 -0
  133. package/rules/.source/hellotv/src/config/build-config.ts +22 -0
  134. package/rules/.source/hellotv/src/config/private-theme-config.ts +21 -0
  135. package/rules/.source/hellotv/src/config/public-config.scss +16 -0
  136. package/rules/.source/hellotv/src/config/test.scss +45 -0
  137. package/rules/.source/hellotv/src/config/theme-config.ts +51 -0
  138. package/rules/.source/hellotv/src/main.ts +45 -0
  139. package/rules/.source/hellotv/src/pages/activity/adapter/config.ts +37 -0
  140. package/rules/.source/hellotv/src/pages/activity/adapter/index.ts +849 -0
  141. package/rules/.source/hellotv/src/pages/activity/adapter/interface.ts +173 -0
  142. package/rules/.source/hellotv/src/pages/activity/api/index.ts +36 -0
  143. package/rules/.source/hellotv/src/pages/activity/api/interface.ts +11 -0
  144. package/rules/.source/hellotv/src/pages/activity/api/request-url.ts +10 -0
  145. package/rules/.source/hellotv/src/pages/activity/components/item/focus-change-img-item.vue +40 -0
  146. package/rules/.source/hellotv/src/pages/activity/components/item/inner-out-title-item.vue +59 -0
  147. package/rules/.source/hellotv/src/pages/activity/components/item/no-title-item.vue +45 -0
  148. package/rules/.source/hellotv/src/pages/activity/components/item/placeholder-item.vue +25 -0
  149. package/rules/.source/hellotv/src/pages/activity/index.vue +219 -0
  150. package/rules/.source/hellotv/src/pages/activity/scss/index.scss +46 -0
  151. package/rules/.source/hellotv/src/pages/activity/scss/item.scss +69 -0
  152. package/rules/.source/hellotv/src/pages/detail/adapter/index.ts +317 -0
  153. package/rules/.source/hellotv/src/pages/detail/adapter/interface.ts +150 -0
  154. package/rules/.source/hellotv/src/pages/detail/adapter/media-player.ts +378 -0
  155. package/rules/.source/hellotv/src/pages/detail/api/index.ts +77 -0
  156. package/rules/.source/hellotv/src/pages/detail/api/request-url.ts +8 -0
  157. package/rules/.source/hellotv/src/pages/detail/components/media-player/collapse/media-collapse-definition.vue +192 -0
  158. package/rules/.source/hellotv/src/pages/detail/components/media-player/collapse/media-collapse-list-item.vue +108 -0
  159. package/rules/.source/hellotv/src/pages/detail/components/media-player/collapse/media-collapse-media-series.vue +349 -0
  160. package/rules/.source/hellotv/src/pages/detail/components/media-player/collapse/media-collapse-order.vue +194 -0
  161. package/rules/.source/hellotv/src/pages/detail/components/media-player/collapse/media-collapse-speed.vue +192 -0
  162. package/rules/.source/hellotv/src/pages/detail/components/media-player/index.vue +404 -0
  163. package/rules/.source/hellotv/src/pages/detail/components/media-player/media-player-small-view.vue +382 -0
  164. package/rules/.source/hellotv/src/pages/detail/components/media-player/media-player-view.vue +1099 -0
  165. package/rules/.source/hellotv/src/pages/detail/components/recommend-item.vue +167 -0
  166. package/rules/.source/hellotv/src/pages/detail/components/section/basic-section.vue +218 -0
  167. package/rules/.source/hellotv/src/pages/detail/components/section/config.ts +6 -0
  168. package/rules/.source/hellotv/src/pages/detail/components/section/header-section.vue +15 -0
  169. package/rules/.source/hellotv/src/pages/detail/components/section/item/button-menu.vue +259 -0
  170. package/rules/.source/hellotv/src/pages/detail/components/section/item/media-introduction.vue +238 -0
  171. package/rules/.source/hellotv/src/pages/detail/components/section/item/media-series.vue +308 -0
  172. package/rules/.source/hellotv/src/pages/detail/components/section/item/player-placeholder.vue +88 -0
  173. package/rules/.source/hellotv/src/pages/detail/index.vue +571 -0
  174. package/rules/.source/hellotv/src/pages/detail/scss/index.scss +22 -0
  175. package/rules/.source/hellotv/src/pages/detail-full-player/components/media-player/full-player-loading.vue +78 -0
  176. package/rules/.source/hellotv/src/pages/detail-full-player/components/media-player/full-player-menu-view.vue +803 -0
  177. package/rules/.source/hellotv/src/pages/detail-full-player/components/media-player/full-player-view.vue +219 -0
  178. package/rules/.source/hellotv/src/pages/detail-full-player/index.vue +101 -0
  179. package/rules/.source/hellotv/src/pages/detail-full-player/scss/full-player-view.scss +6 -0
  180. package/rules/.source/hellotv/src/pages/detail-full-screen/adapter/index.ts +0 -0
  181. package/rules/.source/hellotv/src/pages/detail-full-screen/adapter/interface.ts +133 -0
  182. package/rules/.source/hellotv/src/pages/detail-full-screen/api/index.ts +77 -0
  183. package/rules/.source/hellotv/src/pages/detail-full-screen/api/request-url.ts +8 -0
  184. package/rules/.source/hellotv/src/pages/detail-full-screen/components/media-info.vue +263 -0
  185. package/rules/.source/hellotv/src/pages/detail-full-screen/components/media-player.vue +42 -0
  186. package/rules/.source/hellotv/src/pages/detail-full-screen/components/media-series-section.vue +26 -0
  187. package/rules/.source/hellotv/src/pages/detail-full-screen/components/recommend-item.vue +167 -0
  188. package/rules/.source/hellotv/src/pages/detail-full-screen/index.vue +163 -0
  189. package/rules/.source/hellotv/src/pages/detail-full-screen/scss/index.scss +41 -0
  190. package/rules/.source/hellotv/src/pages/device-info/index.vue +106 -0
  191. package/rules/.source/hellotv/src/pages/device-info/scss/device-info.scss +49 -0
  192. package/rules/.source/hellotv/src/pages/filter/README.md +66 -0
  193. package/rules/.source/hellotv/src/pages/filter/adapter/index.ts +180 -0
  194. package/rules/.source/hellotv/src/pages/filter/adapter/interface.ts +111 -0
  195. package/rules/.source/hellotv/src/pages/filter/api/index.ts +38 -0
  196. package/rules/.source/hellotv/src/pages/filter/api/interface.ts +44 -0
  197. package/rules/.source/hellotv/src/pages/filter/api/request-url.ts +8 -0
  198. package/rules/.source/hellotv/src/pages/filter/components/content/index.vue +469 -0
  199. package/rules/.source/hellotv/src/pages/filter/components/content/list-item-record.vue +16 -0
  200. package/rules/.source/hellotv/src/pages/filter/components/content/list-item.vue +78 -0
  201. package/rules/.source/hellotv/src/pages/filter/components/expand/index.vue +83 -0
  202. package/rules/.source/hellotv/src/pages/filter/components/sidebar/index.vue +81 -0
  203. package/rules/.source/hellotv/src/pages/filter/components/sidebar/list-item-filter-title.vue +78 -0
  204. package/rules/.source/hellotv/src/pages/filter/components/sidebar/list-item-filter.vue +68 -0
  205. package/rules/.source/hellotv/src/pages/filter/components/sidebar/list-item-line.vue +11 -0
  206. package/rules/.source/hellotv/src/pages/filter/components/sidebar/list-item-text.vue +43 -0
  207. package/rules/.source/hellotv/src/pages/filter/components/sidebar/list-item-title.vue +14 -0
  208. package/rules/.source/hellotv/src/pages/filter/config.ts +10 -0
  209. package/rules/.source/hellotv/src/pages/filter/expand-sidebar-contents.vue +203 -0
  210. package/rules/.source/hellotv/src/pages/filter/scss/filter-content-list-item.scss +42 -0
  211. package/rules/.source/hellotv/src/pages/filter/scss/filter-content.scss +60 -0
  212. package/rules/.source/hellotv/src/pages/filter/scss/filter-expand.scss +30 -0
  213. package/rules/.source/hellotv/src/pages/filter/scss/filter-sidebar.scss +45 -0
  214. package/rules/.source/hellotv/src/pages/filter/scss/filter.scss +21 -0
  215. package/rules/.source/hellotv/src/pages/filter/sidebar-contents.vue +145 -0
  216. package/rules/.source/hellotv/src/pages/filter/single-contents.vue +61 -0
  217. package/rules/.source/hellotv/src/pages/history/adapter/index.ts +51 -0
  218. package/rules/.source/hellotv/src/pages/history/adapter/interface.ts +22 -0
  219. package/rules/.source/hellotv/src/pages/history/api/index.ts +54 -0
  220. package/rules/.source/hellotv/src/pages/history/api/interface.ts +41 -0
  221. package/rules/.source/hellotv/src/pages/history/api/request-url.ts +10 -0
  222. package/rules/.source/hellotv/src/pages/history/components/confirm-dialog.vue +155 -0
  223. package/rules/.source/hellotv/src/pages/history/config.ts +8 -0
  224. package/rules/.source/hellotv/src/pages/history/index-raw.vue +540 -0
  225. package/rules/.source/hellotv/src/pages/history/index.vue +431 -0
  226. package/rules/.source/hellotv/src/pages/history/mock.ts +32 -0
  227. package/rules/.source/hellotv/src/pages/history/scss/history-raw.scss +181 -0
  228. package/rules/.source/hellotv/src/pages/history/scss/history.scss +148 -0
  229. package/rules/.source/hellotv/src/pages/home/adapter/exit/home-exit-adapter.ts +20 -0
  230. package/rules/.source/hellotv/src/pages/home/adapter/exit/home-exit-imp.ts +13 -0
  231. package/rules/.source/hellotv/src/pages/home/adapter/media/create-home-player-interceptor.ts +50 -0
  232. package/rules/.source/hellotv/src/pages/home/adapter/media/home-media-imp.ts +74 -0
  233. package/rules/.source/hellotv/src/pages/home/adapter/tab-bar/tab-bar-adapter.ts +151 -0
  234. package/rules/.source/hellotv/src/pages/home/adapter/tab-bar/tab-bar-config.ts +17 -0
  235. package/rules/.source/hellotv/src/pages/home/adapter/tab-bar/tab-bar-imp.ts +53 -0
  236. package/rules/.source/hellotv/src/pages/home/adapter/tab-bar/tab-bar-item-type.ts +6 -0
  237. package/rules/.source/hellotv/src/pages/home/adapter/tab-content/tab-content-adapter.ts +1304 -0
  238. package/rules/.source/hellotv/src/pages/home/adapter/tab-content/tab-content-config.ts +30 -0
  239. package/rules/.source/hellotv/src/pages/home/adapter/tab-content/tab-content-imp.ts +173 -0
  240. package/rules/.source/hellotv/src/pages/home/adapter/tab-content/tab-content-item-type.ts +17 -0
  241. package/rules/.source/hellotv/src/pages/home/api/index.ts +157 -0
  242. package/rules/.source/hellotv/src/pages/home/api/interface.ts +56 -0
  243. package/rules/.source/hellotv/src/pages/home/api/request-url.ts +22 -0
  244. package/rules/.source/hellotv/src/pages/home/components/exit-dialog.vue +148 -0
  245. package/rules/.source/hellotv/src/pages/home/components/media/bg-player.vue +440 -0
  246. package/rules/.source/hellotv/src/pages/home/components/nav-bar/bar-img-item.vue +39 -0
  247. package/rules/.source/hellotv/src/pages/home/components/nav-bar/bar-text-item.vue +100 -0
  248. package/rules/.source/hellotv/src/pages/home/components/tab-content/bg-player-cell-list-item-img.vue +54 -0
  249. package/rules/.source/hellotv/src/pages/home/components/tab-content/bg-player-cell-list-item-text.vue +67 -0
  250. package/rules/.source/hellotv/src/pages/home/components/tab-content/cell-list-player-item.vue +74 -0
  251. package/rules/.source/hellotv/src/pages/home/components/tab-content/cell-player-item.vue +40 -0
  252. package/rules/.source/hellotv/src/pages/home/components/tab-content/focus-change-img-item.vue +41 -0
  253. package/rules/.source/hellotv/src/pages/home/components/tab-content/history-item.vue +362 -0
  254. package/rules/.source/hellotv/src/pages/home/components/tab-content/inner-out-title-item.vue +60 -0
  255. package/rules/.source/hellotv/src/pages/home/components/tab-content/no-title-item.vue +46 -0
  256. package/rules/.source/hellotv/src/pages/home/components/tab-content/placeholder-item.vue +25 -0
  257. package/rules/.source/hellotv/src/pages/home/components/tab-content/short-video/short-video-section-item.vue +80 -0
  258. package/rules/.source/hellotv/src/pages/home/components/tab-content/short-video/short-video-section.vue +70 -0
  259. package/rules/.source/hellotv/src/pages/home/components/tab-content/small-4k/small-four-section.vue +76 -0
  260. package/rules/.source/hellotv/src/pages/home/components/tab-content/world-4k/world-four-section-item.vue +90 -0
  261. package/rules/.source/hellotv/src/pages/home/components/tab-content/world-4k/world-four-section.vue +57 -0
  262. package/rules/.source/hellotv/src/pages/home/components/waterfall-tabs.vue +1307 -0
  263. package/rules/.source/hellotv/src/pages/home/index.vue +117 -0
  264. package/rules/.source/hellotv/src/pages/home/scss/bar-img-item.scss +11 -0
  265. package/rules/.source/hellotv/src/pages/home/scss/bar-text-item.scss +80 -0
  266. package/rules/.source/hellotv/src/pages/home/scss/base.scss +8 -0
  267. package/rules/.source/hellotv/src/pages/home/scss/bg-player-cell-list-item.scss +24 -0
  268. package/rules/.source/hellotv/src/pages/home/scss/bg-player.scss +50 -0
  269. package/rules/.source/hellotv/src/pages/home/scss/cell-player-item.scss +24 -0
  270. package/rules/.source/hellotv/src/pages/home/scss/exit-dialog.scss +89 -0
  271. package/rules/.source/hellotv/src/pages/home/scss/focus-change-img-item.scss +12 -0
  272. package/rules/.source/hellotv/src/pages/home/scss/home.scss +32 -0
  273. package/rules/.source/hellotv/src/pages/home/scss/inner-out-title-item.scss +24 -0
  274. package/rules/.source/hellotv/src/pages/home/scss/no-title-item.scss +15 -0
  275. package/rules/.source/hellotv/src/pages/home/scss/placeholder-item.scss +14 -0
  276. package/rules/.source/hellotv/src/pages/home/scss/short-video-section.scss +59 -0
  277. package/rules/.source/hellotv/src/pages/home/scss/small-four-section.scss +14 -0
  278. package/rules/.source/hellotv/src/pages/home/scss/waterfall-tabs.scss +50 -0
  279. package/rules/.source/hellotv/src/pages/home/scss/world-four-section.scss +106 -0
  280. package/rules/.source/hellotv/src/pages/introduction/index.vue +227 -0
  281. package/rules/.source/hellotv/src/pages/live/components/menu/first-list-item-icon-text.vue +35 -0
  282. package/rules/.source/hellotv/src/pages/live/components/menu/first-list-item-img.vue +16 -0
  283. package/rules/.source/hellotv/src/pages/live/components/menu/first-list-item-text.vue +16 -0
  284. package/rules/.source/hellotv/src/pages/live/components/menu/index.vue +266 -0
  285. package/rules/.source/hellotv/src/pages/live/components/menu/second-list-item.vue +80 -0
  286. package/rules/.source/hellotv/src/pages/live/components/menu/third-list-item.vue +68 -0
  287. package/rules/.source/hellotv/src/pages/live/components/player/index.vue +168 -0
  288. package/rules/.source/hellotv/src/pages/live/components/player/player-error.vue +48 -0
  289. package/rules/.source/hellotv/src/pages/live/components/player/player-loading.vue +50 -0
  290. package/rules/.source/hellotv/src/pages/live/components/player/player-tips.vue +125 -0
  291. package/rules/.source/hellotv/src/pages/live/components/player/tips-icon-text.vue +27 -0
  292. package/rules/.source/hellotv/src/pages/live/css/menu.css +109 -0
  293. package/rules/.source/hellotv/src/pages/live/css/player.css +56 -0
  294. package/rules/.source/hellotv/src/pages/live/index.vue +128 -0
  295. package/rules/.source/hellotv/src/pages/live/mock/index.ts +263 -0
  296. package/rules/.source/hellotv/src/pages/live/types/index.ts +42 -0
  297. package/rules/.source/hellotv/src/pages/login/index.vue +116 -0
  298. package/rules/.source/hellotv/src/pages/login/scss/login.scss +61 -0
  299. package/rules/.source/hellotv/src/pages/m-test.vue +84 -0
  300. package/rules/.source/hellotv/src/pages/my/README.md +87 -0
  301. package/rules/.source/hellotv/src/pages/my/index.vue +177 -0
  302. package/rules/.source/hellotv/src/pages/my/my-data-manager.ts +606 -0
  303. package/rules/.source/hellotv/src/pages/my/my-templates.vue +24 -0
  304. package/rules/.source/hellotv/src/pages/my/poster/index.vue +98 -0
  305. package/rules/.source/hellotv/src/pages/my/poster/my-card2.vue +90 -0
  306. package/rules/.source/hellotv/src/pages/my/poster/my-icon-title-col.vue +72 -0
  307. package/rules/.source/hellotv/src/pages/my/poster/my-icon-title-row.vue +72 -0
  308. package/rules/.source/hellotv/src/pages/my/poster/my-info.vue +50 -0
  309. package/rules/.source/hellotv/src/pages/my/poster/my-user.vue +126 -0
  310. package/rules/.source/hellotv/src/pages/play-test.vue +95 -0
  311. package/rules/.source/hellotv/src/pages/search/README.md +57 -0
  312. package/rules/.source/hellotv/src/pages/search/adapter/index.ts +285 -0
  313. package/rules/.source/hellotv/src/pages/search/adapter/interface.ts +60 -0
  314. package/rules/.source/hellotv/src/pages/search/api/index.ts +100 -0
  315. package/rules/.source/hellotv/src/pages/search/api/interface.ts +64 -0
  316. package/rules/.source/hellotv/src/pages/search/api/request-url.ts +20 -0
  317. package/rules/.source/hellotv/src/pages/search/components/search-content-tabs.vue +153 -0
  318. package/rules/.source/hellotv/src/pages/search/components/search-content.vue +282 -0
  319. package/rules/.source/hellotv/src/pages/search/components/search-keyboard.vue +252 -0
  320. package/rules/.source/hellotv/src/pages/search/components/search-keyword-grid.vue +85 -0
  321. package/rules/.source/hellotv/src/pages/search/components/search-keyword.vue +346 -0
  322. package/rules/.source/hellotv/src/pages/search/config.ts +10 -0
  323. package/rules/.source/hellotv/src/pages/search/scss/search-content.scss +130 -0
  324. package/rules/.source/hellotv/src/pages/search/scss/search-keyboard.scss +91 -0
  325. package/rules/.source/hellotv/src/pages/search/scss/search-keyword-grid.scss +30 -0
  326. package/rules/.source/hellotv/src/pages/search/scss/search-keyword.scss +104 -0
  327. package/rules/.source/hellotv/src/pages/search/scss/search.scss +18 -0
  328. package/rules/.source/hellotv/src/pages/search/three-columns.vue +164 -0
  329. package/rules/.source/hellotv/src/pages/search/two-columns.vue +128 -0
  330. package/rules/.source/hellotv/src/pages/tabs-test.vue +339 -0
  331. package/rules/.source/hellotv/src/pages/webview/load-web-view.vue +44 -0
  332. package/rules/.source/hellotv/src/routes.ts +142 -0
  333. package/rules/.source/hellotv/src/tools/common.ts +18 -0
  334. package/rules/.source/hellotv/src/tools/format-date.ts +16 -0
  335. package/rules/.source/hellotv/src/tools/index.ts +58 -0
  336. package/rules/.source/hellotv/src/tools/launch.ts +339 -0
  337. package/rules/.source/hellotv/src/tools/request/index.ts +206 -0
  338. package/rules/.source/hellotv/src/tools/request/interface.ts +64 -0
  339. package/rules/.source/hellotv/src/typings/shims-vue.d.ts +6 -0
  340. package/rules/.source/hellotv/src/vue.config.js +3 -0
  341. package/rules/.source/hellotv/tsconfig.json +34 -0
  342. package/rules/.windsurfrules +1 -0
  343. package/rules/AGENTS.md +1 -0
  344. package/rules/CLAUDE.md +1 -0
  345. package/rules/GEMINI.md +1 -0
@@ -0,0 +1,282 @@
1
+ <template>
2
+ <qt-view class="search-content">
3
+ <!-- 提示 -->
4
+ <qt-text
5
+ v-if="showTips && !lockTips"
6
+ class="search-content-tips"
7
+ :text="tips"
8
+ gravity="center|start"
9
+ typeface="bold"
10
+ ></qt-text>
11
+ <!-- 内容 -->
12
+ <qt-tabs
13
+ ref="tabRef"
14
+ tabNavBarClass="search-content-tab"
15
+ tabPageClass="search-content-tab-page"
16
+ :focusMemory="true"
17
+ :autoHandleBackKey="true"
18
+ :enablePlaceholder="themeConfig.placeHolderEnable"
19
+ :contentNextFocus="{ left: 'keywordList' }"
20
+ :tabContentBlockFocusDirections="['up', 'down']"
21
+ @onTabPageLoadData="onTabPageLoadData"
22
+ @onTabPageItemFocused="onTabPageItemFocused"
23
+ @onTabPageItemClick="onTabPageItemClick"
24
+ @onTabMoveToTopStart="onTabMoveToTopStart"
25
+ @onTabMoveToBottomEnd="onTabMoveToBottomEnd"
26
+ >
27
+ <!-- Tab -->
28
+ <template v-slot:tab-item>
29
+ <qt-view
30
+ class="search-content-tab-item"
31
+ :type="TabItemType.TEXT"
32
+ :focusable="true"
33
+ >
34
+ <qt-text
35
+ class="search-content-tab-item-text"
36
+ autoWidth
37
+ text="${text}"
38
+ gravity="center|start"
39
+ :focusable="false"
40
+ :duplicateParentState="true"
41
+ ></qt-text>
42
+ </qt-view>
43
+ </template>
44
+ <!-- Content -->
45
+ <template v-slot:waterfall-item>
46
+ <!-- 横图 -->
47
+ <grid-item-horizontal
48
+ :type="ContentType.HORIZONTAL"
49
+ :style="{ width: `410px`, height: `276px` }"
50
+ :imageStyle="{
51
+ width: `410px`,
52
+ height: `230px`,
53
+ borderRadius: `${themeConfig.focusBorderCorner}px`,
54
+ }"
55
+ layout="${layout}"
56
+ />
57
+ <!-- 竖图 -->
58
+ <grid-item-vertical
59
+ :type="ContentType.VERTICAL"
60
+ name="gridItem"
61
+ layout="${layout}"
62
+ />
63
+ </template>
64
+ <!-- 分页加载中 -->
65
+ <template v-slot:waterfall-section>
66
+ <qt-view
67
+ style="
68
+ width: 1920px;
69
+ height: 100px;
70
+ background-color: transparent;
71
+ align-items: center;
72
+ justify-content: center;
73
+ "
74
+ :type="-10008"
75
+ :focusable="false"
76
+ >
77
+ <qt-loading-view
78
+ style="height: 40px; width: 40px"
79
+ name="loading"
80
+ color="rgba(255,255,255,0.3)"
81
+ :focusable="false"
82
+ />
83
+ </qt-view>
84
+ </template>
85
+ </qt-tabs>
86
+ </qt-view>
87
+ </template>
88
+
89
+ <script setup lang="ts">
90
+ import { ref, watch } from "vue";
91
+ import {
92
+ QTITab,
93
+ QTTabPageData,
94
+ QTTabPageState,
95
+ QTWaterfallItem,
96
+ } from "@quicktvui/quicktvui3";
97
+ import {
98
+ buildTab,
99
+ buildContentSections,
100
+ buildRecommendSections,
101
+ buildLoadingSection,
102
+ buildEndSection,
103
+ } from "../adapter/index";
104
+ import { TabItemType, ContentType } from "../adapter/interface";
105
+ import themeConfig from "../../../config/theme-config";
106
+ import gridItemHorizontal from "../../../components/grid-item-horizontal.vue";
107
+ import gridItemVertical from "../../../components/grid-item-vertical.vue";
108
+ import launch from "../../../tools/launch";
109
+ import config from "../config";
110
+ import searchManager from "../api/index";
111
+
112
+ const props = defineProps({
113
+ keyword: {
114
+ type: String,
115
+ default: "",
116
+ },
117
+ });
118
+ const emits = defineEmits(["setLoading", "updateFocusName", "updateFocusDeny"]);
119
+ // 顶部提示
120
+ const showTips = ref<boolean>(true);
121
+ const lockTips = ref<boolean>(false);
122
+ const tips = ref<string>("");
123
+ // 内容组件
124
+ const tabRef = ref<QTITab>();
125
+ // 关键词
126
+ const rawKeyword = ref<string>();
127
+
128
+ watch(
129
+ () => props.keyword,
130
+ () => {
131
+ // 设置loading状态
132
+ emits("setLoading", true);
133
+ // 重置状态
134
+ showTips.value = true;
135
+ lockTips.value = false;
136
+ // 提示词
137
+ tips.value =
138
+ props.keyword.length > 0 ? `全部“${props.keyword}”结果` : "热门推荐";
139
+ // 初始化组件
140
+ init(props.keyword);
141
+ },
142
+ );
143
+
144
+ function init(keyword: string) {
145
+ rawKeyword.value = keyword;
146
+ // 初始化组件
147
+ tabRef.value?.initTab(buildTab());
148
+ tabRef.value?.initPage({ width: 1920, height: 1080 });
149
+ }
150
+
151
+ // 加载关键词搜索和大家都在搜
152
+ async function loadSearchData(pageIndex: number, page: number) {
153
+ let tabPage: QTTabPageData = { useDiff: true, data: [] };
154
+ let stopPage = false;
155
+
156
+ try {
157
+ const contentsResult = await searchManager.getContents(
158
+ rawKeyword.value,
159
+ page,
160
+ config.gridContentsLimit,
161
+ );
162
+ const sections = buildContentSections(contentsResult, 4);
163
+
164
+ if (contentsResult.items.length > 0) {
165
+ tabPage.data.push(...sections);
166
+ } else {
167
+ // 没有搜索结果时, 不展示顶部提示词
168
+ showTips.value = false;
169
+ lockTips.value = true;
170
+ }
171
+
172
+ if (contentsResult.items.length < config.gridContentsLimit) {
173
+ stopPage = true;
174
+ // 请求大家都在搜
175
+ const recommends = await searchManager.getHotRecommends(
176
+ 1,
177
+ config.gridHotRecommendsLimit,
178
+ );
179
+ tabPage.data.push(
180
+ ...buildRecommendSections(recommends, 6, true, showTips.value),
181
+ );
182
+ tabPage.data.push(buildEndSection());
183
+ // 停止分页
184
+ tabRef.value?.setPageState(
185
+ pageIndex,
186
+ QTTabPageState.QT_TAB_PAGE_STATE_COMPLETE,
187
+ );
188
+ }
189
+
190
+ // 添加加载中
191
+ if (!stopPage) {
192
+ tabPage.data.push(buildLoadingSection());
193
+ }
194
+
195
+ if (page === 1) {
196
+ // 列表第一个Item添加特殊标识
197
+ tabPage.data[0].itemList[0]._id = "--search-grid-first-item--";
198
+ tabRef.value?.setPageData(pageIndex, tabPage);
199
+ } else {
200
+ tabRef.value?.addPageData(pageIndex, tabPage, 1);
201
+ }
202
+ } catch (error) {
203
+ qt.log.e("error->loadSearchData", error);
204
+ } finally {
205
+ delayedUpdate();
206
+ }
207
+ }
208
+
209
+ // 延迟更新上层状态
210
+ function delayedUpdate() {
211
+ clearTimeout(timer);
212
+ timer = setTimeout(() => {
213
+ emits("setLoading", false);
214
+ emits("updateFocusDeny", false);
215
+ }, 500);
216
+ }
217
+
218
+ let timer: any = -1;
219
+ function onTabPageLoadData(pageIndex: number, pageNo: number) {
220
+ if (rawKeyword.value?.length === 0) {
221
+ searchManager
222
+ .getHotRecommends(1, config.gridHotRecommendsLimit)
223
+ .then((recommends) => {
224
+ let tabPage: QTTabPageData = { useDiff: true, data: [] };
225
+ if (recommends.items.length > 0) {
226
+ tabPage.data.push(
227
+ ...buildRecommendSections(recommends, 6, false, showTips.value),
228
+ );
229
+ }
230
+ tabPage.data.push(buildEndSection());
231
+ tabRef.value?.setPageData(pageIndex, tabPage);
232
+ tabRef.value?.setPageState(
233
+ pageIndex,
234
+ QTTabPageState.QT_TAB_PAGE_STATE_COMPLETE,
235
+ );
236
+ })
237
+ .catch((error) => {
238
+ qt.log.e("error->getContents for recommends", error);
239
+ })
240
+ .finally(() => {
241
+ delayedUpdate();
242
+ });
243
+ } else {
244
+ loadSearchData(pageIndex, ++pageNo);
245
+ }
246
+ }
247
+
248
+ function onTabPageItemFocused(
249
+ pageIndex: number,
250
+ sectionIndex: number,
251
+ itemIndex: number,
252
+ isFocused: boolean,
253
+ item: QTWaterfallItem,
254
+ ) {
255
+ if (isFocused) {
256
+ qt.eventBus.emit("updateFocusRightSid", "");
257
+ emits("updateFocusName", "searchContentItem");
258
+ }
259
+ }
260
+
261
+ function onTabPageItemClick(
262
+ pageIndex: number,
263
+ sectionIndex: number,
264
+ itemIndex: number,
265
+ item: QTWaterfallItem,
266
+ ) {
267
+ // 跳转详情页
268
+ launch.launchDetail(item.id);
269
+ // 上报搜索历史
270
+ searchManager.addHistory(rawKeyword.value || "");
271
+ }
272
+
273
+ function onTabMoveToTopStart() {
274
+ showTips.value = false;
275
+ }
276
+
277
+ function onTabMoveToBottomEnd() {
278
+ showTips.value = true;
279
+ }
280
+ </script>
281
+
282
+ <style lang="scss" src="../scss/search-content.scss"></style>
@@ -0,0 +1,252 @@
1
+ <template>
2
+ <qt-view class="search-keyboard">
3
+ <!-- 输入框 -->
4
+ <qt-view class="search-keyboard-input">
5
+ <qt-image
6
+ class="search-keyboard-input-icon"
7
+ :src="icSearch"
8
+ :focusable="false"
9
+ ></qt-image>
10
+ <qt-text
11
+ v-if="inputText.length > 0"
12
+ class="search-keyboard-input-text"
13
+ style="color: white; font-size: 40px; margin-top: -3px"
14
+ :text="inputText"
15
+ gravity="center|start"
16
+ ></qt-text>
17
+ <qt-text
18
+ v-else
19
+ class="search-keyboard-input-text"
20
+ :textSpan="defaultText"
21
+ ></qt-text>
22
+ </qt-view>
23
+ <!-- 输入框底部横线 -->
24
+ <qt-view class="search-keyboard-input-bottom"></qt-view>
25
+ <!-- 按钮区域 -->
26
+ <qt-view
27
+ class="search-keyboard-btns"
28
+ :blockFocusDirections="['up', 'left']"
29
+ >
30
+ <qt-view
31
+ class="search-keyboard-btn"
32
+ :focusable="true"
33
+ @click="onBtnClick('clear')"
34
+ >
35
+ <qt-image
36
+ style="
37
+ width: 30px;
38
+ height: 30px;
39
+ background-color: transparent;
40
+ position: absolute;
41
+ left: 30px;
42
+ "
43
+ :src="icClear"
44
+ :showOnState="['normal', 'selected']"
45
+ :focusable="false"
46
+ :duplicateParentState="true"
47
+ ></qt-image>
48
+ <qt-image
49
+ style="
50
+ width: 30px;
51
+ height: 30px;
52
+ background-color: transparent;
53
+ position: absolute;
54
+ left: 30px;
55
+ "
56
+ :src="icClearFocused"
57
+ :showOnState="'focused'"
58
+ :focusable="false"
59
+ :duplicateParentState="true"
60
+ ></qt-image>
61
+ <qt-text
62
+ class="search-keyboard-btn-text"
63
+ text="清空"
64
+ :showOnState="['normal', 'selected']"
65
+ :focusable="false"
66
+ :duplicateParentState="true"
67
+ ></qt-text>
68
+ <qt-text
69
+ class="search-keyboard-btn-text"
70
+ text="清空"
71
+ typeface="bold"
72
+ :showOnState="'focused'"
73
+ :focusable="false"
74
+ :duplicateParentState="true"
75
+ ></qt-text>
76
+ </qt-view>
77
+ <qt-view
78
+ class="search-keyboard-btn"
79
+ :nextFocusName="{ right: 'keywordList' }"
80
+ :focusable="true"
81
+ @click="onBtnClick('back')"
82
+ >
83
+ <qt-image
84
+ style="
85
+ width: 30px;
86
+ height: 30px;
87
+ background-color: transparent;
88
+ position: absolute;
89
+ left: 30px;
90
+ "
91
+ :src="icBack"
92
+ :showOnState="['normal', 'selected']"
93
+ :focusable="false"
94
+ :duplicateParentState="true"
95
+ ></qt-image>
96
+ <qt-image
97
+ style="
98
+ width: 30px;
99
+ height: 30px;
100
+ background-color: transparent;
101
+ position: absolute;
102
+ left: 30px;
103
+ "
104
+ :src="icBackFocused"
105
+ :showOnState="'focused'"
106
+ :focusable="false"
107
+ :duplicateParentState="true"
108
+ ></qt-image>
109
+ <qt-text
110
+ class="search-keyboard-btn-text"
111
+ text="退格"
112
+ :showOnState="['normal', 'selected']"
113
+ :focusable="false"
114
+ :duplicateParentState="true"
115
+ ></qt-text>
116
+ <qt-text
117
+ class="search-keyboard-btn-text"
118
+ text="退格"
119
+ typeface="bold"
120
+ :showOnState="'focused'"
121
+ :focusable="false"
122
+ :duplicateParentState="true"
123
+ ></qt-text>
124
+ </qt-view>
125
+ </qt-view>
126
+ <!-- 键盘区域 -->
127
+ <qt-grid-view
128
+ class="search-keyboard-grid"
129
+ ref="gridRef"
130
+ name="keyboardGrid"
131
+ :focusMemory="false"
132
+ :spanCount="6"
133
+ :autofocusPosition="14"
134
+ :nextFocusName="{ right: gridFocusNameRight }"
135
+ :blockFocusDirections="['down']"
136
+ @item-click="onGridItemClick"
137
+ >
138
+ <qt-view
139
+ :type="1"
140
+ class="search-keyboard-grid-item"
141
+ :focusable="true"
142
+ eventFocus
143
+ eventClick
144
+ >
145
+ <qt-text
146
+ class="search-keyboard-grid-item-text"
147
+ autoWidth
148
+ autoHeight
149
+ text="${text}"
150
+ gravity="center"
151
+ :showOnState="['normal', 'selected']"
152
+ :focusable="false"
153
+ :duplicateParentState="true"
154
+ ></qt-text>
155
+ <qt-text
156
+ class="search-keyboard-grid-item-text"
157
+ autoWidth
158
+ autoHeight
159
+ text="${text}"
160
+ typeface="bold"
161
+ gravity="center"
162
+ :showOnState="'focused'"
163
+ :focusable="false"
164
+ :duplicateParentState="true"
165
+ ></qt-text>
166
+ </qt-view>
167
+ </qt-grid-view>
168
+ </qt-view>
169
+ </template>
170
+
171
+ <script setup lang="ts">
172
+ import { ref, watch, onMounted } from "vue";
173
+ import { QTIGridView, QTListViewItem } from "@quicktvui/quicktvui3";
174
+ import ThemeConfig from "../../../config/theme-config";
175
+ import icSearch from "../../../assets/search/ic_search.png";
176
+ import icClear from "../../../assets/search/ic_clear.png";
177
+ import icClearFocused from "../../../assets/search/ic_clear_focused.png";
178
+ import icBack from "../../../assets/search/ic_back.png";
179
+ import icBackFocused from "../../../assets/search/ic_back_focused.png";
180
+
181
+ const emits = defineEmits(["updateInput"]);
182
+
183
+ const gridRef = ref<QTIGridView>();
184
+ const gridFocusNameRight = ref<string>("");
185
+ const inputText = ref<string>("");
186
+ const defaultText = {
187
+ text: "输入片名的首字母或全拼搜索",
188
+ spanAttr: [
189
+ { type: "color", value: ["#FFFFFF", 5, 8] },
190
+ { type: "color", value: ["#FFFFFF", 9, 11] },
191
+ ],
192
+ };
193
+
194
+ onMounted(() => {
195
+ const keyboardItems: QTListViewItem[] = [];
196
+ // 输出 A-Z
197
+ for (let i = 65; i <= 90; i++) {
198
+ keyboardItems.push({
199
+ type: 1,
200
+ text: String.fromCharCode(i),
201
+ decoration: { top: 2, bottom: 2 },
202
+ });
203
+ }
204
+ // 输出 0-9
205
+ for (let i = 48; i <= 57; i++) {
206
+ keyboardItems.push({
207
+ type: 1,
208
+ text: String.fromCharCode(i),
209
+ decoration: { top: 2, bottom: 2 },
210
+ });
211
+ }
212
+ // 初始化键盘
213
+ gridRef.value?.init(keyboardItems);
214
+ });
215
+
216
+ watch(
217
+ () => inputText.value,
218
+ () => {
219
+ emits("updateInput", inputText.value);
220
+ },
221
+ );
222
+
223
+ function onBtnClick(name: "clear" | "back") {
224
+ switch (name) {
225
+ case "clear":
226
+ inputText.value = "";
227
+ break;
228
+ case "back":
229
+ if (inputText.value.length > 0) {
230
+ inputText.value = inputText.value.slice(0, inputText.value.length - 1);
231
+ }
232
+ break;
233
+ }
234
+ }
235
+
236
+ function onGridItemClick(evt) {
237
+ inputText.value += evt.item.text;
238
+ }
239
+
240
+ // 更新键盘向右焦点位置
241
+ function updateKeyboardFocusRight(name: string) {
242
+ gridFocusNameRight.value = name;
243
+ }
244
+
245
+ function onBackPressed() {
246
+ gridRef.value?.setItemFocused(14);
247
+ }
248
+
249
+ defineExpose({ updateKeyboardFocusRight, onBackPressed });
250
+ </script>
251
+
252
+ <style scoped lang="scss" src="../scss/search-keyboard.scss"></style>
@@ -0,0 +1,85 @@
1
+ <template>
2
+ <qt-view class="search-keyword">
3
+ <qt-grid-view
4
+ class="search-keyword-grid"
5
+ :listData="gridData"
6
+ :spanCount="3"
7
+ :padding="'40,80,0,0'"
8
+ @item-click="onGridItemClick"
9
+ >
10
+ <!-- 标题 -->
11
+ <template v-slot:header>
12
+ <qt-view
13
+ :type="KeywordType.TITLE"
14
+ class="search-keyword-grid-item"
15
+ :focusable="false"
16
+ >
17
+ <qt-text
18
+ class="search-keyword-grid-item-text"
19
+ style="height: 50px; color: white; font-size: 40px"
20
+ autoWidth
21
+ text="${text}"
22
+ gravity="center"
23
+ :focusable="false"
24
+ ></qt-text>
25
+ </qt-view>
26
+ </template>
27
+ <!-- 内容 -->
28
+ <qt-view
29
+ :type="KeywordType.TEXT"
30
+ class="search-keyword-grid-item"
31
+ :focusable="true"
32
+ eventClick
33
+ >
34
+ <qt-text
35
+ class="search-keyword-grid-item-text"
36
+ style="position: absolute"
37
+ text="${text}"
38
+ :showOnState="['normal', 'selected']"
39
+ :ellipsizeMode="2"
40
+ :focusable="false"
41
+ :duplicateParentState="true"
42
+ ></qt-text>
43
+ <qt-text
44
+ class="search-keyword-grid-item-text"
45
+ style="position: absolute"
46
+ text="${text}"
47
+ typeface="bold"
48
+ :showOnState="'focused'"
49
+ :ellipsizeMode="4"
50
+ :focusable="false"
51
+ :duplicateParentState="true"
52
+ ></qt-text>
53
+ </qt-view>
54
+ </qt-grid-view>
55
+ </qt-view>
56
+ </template>
57
+
58
+ <script setup lang="ts">
59
+ import { onMounted } from "vue";
60
+ import { qtRef, QTListViewItem } from "@quicktvui/quicktvui3";
61
+ import { buildKeywords } from "../adapter/index";
62
+ import { KeywordType } from "../adapter/interface";
63
+ import searchManager from "../api/index";
64
+ import launch from "../../../tools/launch";
65
+
66
+ const emits = defineEmits(["setLoading"]);
67
+ const gridData = qtRef<QTListViewItem[]>();
68
+
69
+ let loadTimer: any = -1;
70
+
71
+ onMounted(() => {
72
+ searchManager.getSuggestions("all").then((suggestions) => {
73
+ gridData.value = buildKeywords(suggestions, "all");
74
+ // 延迟关闭loading
75
+ clearTimeout(loadTimer);
76
+ loadTimer = setTimeout(() => emits("setLoading", false), 300);
77
+ });
78
+ });
79
+
80
+ function onGridItemClick(evt) {
81
+ launch.launchDetail(evt.item.jumpId);
82
+ }
83
+ </script>
84
+
85
+ <style scoped lang="scss" src="../scss/search-keyword-grid.scss"></style>