From bb88e754a5d4d7adfe004991b9661240202dd91f Mon Sep 17 00:00:00 2001 From: Li ZongYing Date: Fri, 29 Mar 2024 17:04:11 +0800 Subject: [PATCH] fix 368 --- HISTORY.md | 229 ++++++++++++++++++ README.md | 220 +---------------- app/src/main/AndroidManifest.xml | 8 + app/src/main/ic_launcher-playstore.png | Bin 25619 -> 0 bytes .../java/com/lizongying/mytv/CardAdapter.kt | 4 +- .../com/lizongying/mytv/ChannelFragment.kt | 2 +- ...logFragment.kt => ConfirmationFragment.kt} | 13 +- app/src/main/java/com/lizongying/mytv/Ext.kt | 21 ++ .../java/com/lizongying/mytv/InfoFragment.kt | 6 +- .../java/com/lizongying/mytv/MainActivity.kt | 58 ++++- .../java/com/lizongying/mytv/MainFragment.kt | 18 +- .../lizongying/mytv/NetworkChangeReceiver.kt | 16 ++ .../main/java/com/lizongying/mytv/Request.kt | 55 +++-- .../com/lizongying/mytv/SettingFragment.kt | 40 +-- app/src/main/java/com/lizongying/mytv/TV.kt | 2 - .../main/java/com/lizongying/mytv/TVList.kt | 68 ------ .../java/com/lizongying/mytv/UpdateManager.kt | 85 +++++-- .../java/com/lizongying/mytv/api/ApiClient.kt | 1 - .../main/java/com/lizongying/mytv/api/Info.kt | 19 +- .../com/lizongying/mytv/api/ReleaseService.kt | 7 +- .../main/java/com/lizongying/mytv/api/YSP.kt | 6 +- .../com/lizongying/mytv/models/TVViewModel.kt | 62 +---- .../com/lizongying/mytv/requests/MyRequest.kt | 12 +- .../res/layout/{dialog.xml => setting.xml} | 0 build.gradle | 4 +- gradle/wrapper/gradle-wrapper.properties | 2 +- 26 files changed, 502 insertions(+), 456 deletions(-) create mode 100644 HISTORY.md delete mode 100644 app/src/main/ic_launcher-playstore.png rename app/src/main/java/com/lizongying/mytv/{ConfirmationDialogFragment.kt => ConfirmationFragment.kt} (72%) create mode 100644 app/src/main/java/com/lizongying/mytv/NetworkChangeReceiver.kt rename app/src/main/res/layout/{dialog.xml => setting.xml} (100%) diff --git a/HISTORY.md b/HISTORY.md new file mode 100644 index 0000000..9c43951 --- /dev/null +++ b/HISTORY.md @@ -0,0 +1,229 @@ +## 更新日志 + +### v1.7.2(通用) + +* 支持节目列表网格样式和行样式切换,软件重启后生效 +* 节目列表样式变更 + +### v1.7.1(安卓5及以上专用) + +* 解决设置页更新闪退的问题 +* 凤凰卫视回归 +* 解决368问题 + +### v1.7.0(通用) + +* 网络请求优化 + +### v1.6.9(安卓5及以上专用) + +* 去掉港澳台和国际频道 +* 解决部分情况下3、6、8等频道无法播放的问题 +* 解决部分情况下启动后黑屏问题 + +### v1.6.8(通用) + +* 修复部分设备崩溃的问题 +* 修复部分设备凤凰卫视无法播放的问题 + +### v1.6.7(安卓5及以上专用) + +* 手机双击打开配置 +* 自动更新 + +### v1.6.6(通用) + +* 更新重庆卫视图标 +* 凤凰卫视增强画质 +* 凤凰卫视增加EPG + +### v1.6.5(安卓5及以上专用) + +* 增加CETV1图标 +* 稳定性提升 + +### v1.6.4(通用) + +* 增加CETV1 +* 增加凤凰卫视 +* 默认关闭开机启动 + +### v1.6.3(安卓5及以上专用) + +* 增加CETV1 +* 凤凰卫视增强画质 +* 默认关闭开机启动 +* 延迟菜单自动关闭时间 +* 解决一些可能导致首次打开时黑屏的问题 + +### v1.6.2(通用) + +* 修复按键无效的问题 +* 新的频道列表样式 + +### v1.6.1(安卓5及以上专用) + +* 增加凤凰卫视 + +### v1.6.0(通用) + +* 通用(春晚緊急修復) + +### v1.5.9(安卓5及以上专用) + +* 解决天猫魔盒闪退问题 + +### v1.5.8(通用) + +* 修复央视6画质差的问题 +* 增加兵团卫视 +* 播放失败重试 + +### v1.5.7(安卓5及以上专用) + +* 修复播放失败的问题 + +### v1.5.6(通用) + +* 解决部分设备系统时间不对导致播放失败的问题 + +### v1.5.5(安卓5及以上专用) + +* 修复播放失败的问题 +* 修复APP恢复后频道号、频道列表不自动消失的问题 + +### v1.5.4(通用) + +* 修复播放失败的问题 + +### 以下已不可用 + +### v1.5.3(安卓5及以上专用) + +* 修复部分情况下APP切换后无法继续播放的问题 +* 优化重试逻辑 + +### v1.5.2(通用) + +* 修复APP恢复后频道号、频道列表不自动消失的问题 + +### v1.5.1(安卓5及以上专用) + +* 性能优化 + +### v1.5.0(通用) + +* 修复部分情况下APP切换后无法继续播放的问题 + +### v1.4.9(安卓5及以上专用) + +* 同步v1.4.8 + +### v1.4.8(通用) + +* 频道号从1开始,CCTV5+为18 +* 提高CCTV6清晰度 +* 增加天津卫视、新疆卫视 + +### v1.4.7(安卓5及以上专用) + +* 修复部分用户CCTV13播放过程中卡住的问题 +* 调整CCTV的频道顺序 + +### v1.4.6(通用) + +* 10以下频道不再需要先按0 + +### v1.4.5(安卓5及以上专用) + +* 数字选台配置 + +### v1.4.4(通用) + +* 优化图标显示 +* 增加换台反转 + +### v1.4.3(安卓5及以上专用) + +* 支持频道反转配置 + +### v1.4.2(通用) + +* 支持安卓4 + +### v1.4.1 + +* 解决部分用户无法打开菜单的问题 + +### v1.4.0 + +* 解决极个别高版本机型黑屏问题 + +### v1.3.9 + +* 提高稳定性 +* 提高连接速度 + +### v1.3.4 + +* 部分错误会提示用户 +* 菜单3秒钟后自动关闭 + +### v1.3.3 + +* 部分错误会提示用户 +* 菜单3秒钟后自动关闭 + +### v1.3.2 + +* 增加重试,减少因网络问题导致的播放失败 +* 优化横幅banner + +### v1.3.1 + +* 增加CCTV 8K 超高清频道 +* 增加国际频道栏目 + +### v1.3.0 + +* 处理368可能失败的情形 + +### v1.2.9 + +* 支持3、6、8 + +### v1.2.8 + +* 兼容16:10等分辨率 + +### v1.2.7 + +* 此版本是为了测试安卓6.0以下版本,如之前可以正常运行,请勿安装。 + +### v1.2.6 + +* 支持安卓4.2 +* 解决部分频道无法播放的问题 +* 修复切换时有时没有恢复播放的问题 +* 左右键不再切换源 +* 增大频道信息标题尺寸,缩小与节目信息的间隔 + +### v1.2.5 + +* 美化频道信息显示 +* 优化节目单获取 + +### v1.2.4 + +* 改变换台滑动方向,上一个频道下滑,下一个频道上滑 +* 软件退出时,退出播放器 +* 播放相同的频道,不再重复加载 +* 暂时移除部分频道 + +### v1.2.3 + +* 固定视频比列为16:9 +* 隐藏全面屏底部的小横条 +* 移除移动专区 +* 修复一个闪退问题 +* 更换图标和应用名 \ No newline at end of file diff --git a/README.md b/README.md index 3185bf3..0acc4ae 100644 --- a/README.md +++ b/README.md @@ -4,214 +4,9 @@ ## 使用 -下载安装 [releases](https://github.com/lizongying/my-tv/releases/) - -更多地址 [my-tv](https://lyrics.run/my-tv.html) - -![image](./screenshots/img_3.png) -![image](./screenshots/img_2.png) -![image](./screenshots/img_1.png) - -## 更新日志 - -### v1.6.6(通用) - -* 更新重庆卫视图标 -* 凤凰卫视增强画质 -* 凤凰卫视增加EPG - -### v1.6.5(安卓5及以上专用) - -* 增加CETV1图标 -* 稳定性提升 - -### v1.6.4(通用) - -* 增加CETV1 -* 增加凤凰卫视 -* 默认关闭开机启动 - -### v1.6.3(安卓5及以上专用) - -* 增加CETV1 -* 凤凰卫视增强画质 -* 默认关闭开机启动 -* 延迟菜单自动关闭时间 -* 解决一些可能导致首次打开时黑屏的问题 - -### v1.6.2(通用) - -* 修复按键无效的问题 -* 新的频道列表样式 - -### v1.6.1(安卓5及以上专用) - -* 增加凤凰卫视 - -### v1.6.0(通用) - -* 通用(春晚緊急修復) - -### v1.5.9(安卓5及以上专用) - -* 解决天猫魔盒闪退问题 - -### v1.5.8(通用) - -* 修复央视6画质差的问题 -* 增加兵团卫视 -* 播放失败重试 - -### v1.5.7(安卓5及以上专用) - -* 修复播放失败的问题 - -### v1.5.6(通用) - -* 解决部分设备系统时间不对导致播放失败的问题 - -### v1.5.5(安卓5及以上专用) - -* 修复播放失败的问题 -* 修复APP恢复后频道号、频道列表不自动消失的问题 - -### v1.5.4(通用) - -* 修复播放失败的问题 - -### 以下已不可用 - -### v1.5.3(安卓5及以上专用) - -* 修复部分情况下APP切换后无法继续播放的问题 -* 优化重试逻辑 - -### v1.5.2(通用) - -* 修复APP恢复后频道号、频道列表不自动消失的问题 - -### v1.5.1(安卓5及以上专用) - -* 性能优化 - -### v1.5.0(通用) - -* 修复部分情况下APP切换后无法继续播放的问题 - -### v1.4.9(安卓5及以上专用) - -* 同步v1.4.8 - -### v1.4.8(通用) - -* 频道号从1开始,CCTV5+为18 -* 提高CCTV6清晰度 -* 增加天津卫视、新疆卫视 - -### v1.4.7(安卓5及以上专用) - -* 修复部分用户CCTV13播放过程中卡住的问题 -* 调整CCTV的频道顺序 - -### v1.4.6(通用) - -* 10以下频道不再需要先按0 - -### v1.4.5(安卓5及以上专用) - -* 数字选台配置 - -### v1.4.4(通用) - -* 优化图标显示 -* 增加换台反转 - -### v1.4.3(安卓5及以上专用) - -* 支持频道反转配置 - -### v1.4.2(通用) - -* 支持安卓4 - -### v1.4.1 - -* 解决部分用户无法打开菜单的问题 - -### v1.4.0 - -* 解决极个别高版本机型黑屏问题 - -### v1.3.9 - -* 提高稳定性 -* 提高连接速度 - -### v1.3.4 - -* 部分错误会提示用户 -* 菜单3秒钟后自动关闭 - -### v1.3.3 - -* 部分错误会提示用户 -* 菜单3秒钟后自动关闭 - -### v1.3.2 - -* 增加重试,减少因网络问题导致的播放失败 -* 优化横幅banner - -### v1.3.1 - -* 增加CCTV 8K 超高清频道 -* 增加国际频道栏目 - -### v1.3.0 - -* 处理368可能失败的情形 - -### v1.2.9 - -* 支持3、6、8 - -### v1.2.8 - -* 兼容16:10等分辨率 - -### v1.2.7 - -* 此版本是为了测试安卓6.0以下版本,如之前可以正常运行,请勿安装。 - -### v1.2.6 - -* 支持安卓4.2 -* 解决部分频道无法播放的问题 -* 修复切换时有时没有恢复播放的问题 -* 左右键不再切换源 -* 增大频道信息标题尺寸,缩小与节目信息的间隔 - -### v1.2.5 - -* 美化频道信息显示 -* 优化节目单获取 - -### v1.2.4 - -* 改变换台滑动方向,上一个频道下滑,下一个频道上滑 -* 软件退出时,退出播放器 -* 播放相同的频道,不再重复加载 -* 暂时移除部分频道 - -### v1.2.3 - -* 固定视频比列为16:9 -* 隐藏全面屏底部的小横条 -* 移除移动专区 -* 修复一个闪退问题 -* 更换图标和应用名 - -## 其他 +下载安装: +[github](https://github.com/lizongying/my-tv/releases/) +[gitee](https://gitee.com/lizongying/my-tv/releases/) 小米电视可以使用小米电视助手进行安装 @@ -221,11 +16,18 @@ adb install my-tv.apk ``` +![image](./screenshots/img_3.png) +![image](./screenshots/img_2.png) +![image](./screenshots/img_1.png) + +## 更新日志 + +[更新日志](./HISTORY.md) + ## TODO * 音量不同 * 大湾区卫视、广东4k超高清、广东珠江、三沙卫视 -* CETV教育频道 * CHC高清三个电影频道 * 地方频道 * 收藏夹 diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 3bb6bc5..03c65fd 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -7,6 +7,8 @@ android:name="android.software.leanback" android:required="true" /> + + + + + + + {(7T>~94{2&I6WV$ znBK2=|DKvn?x5kQq0#H>ZmY$v&y>~3-k(gN%`z<2*uZA6qy1=ZX8UY_z;(zqs#p8& zty{PL3}1U+zl(~B(hnvB9z1x!@)Dp1KTX091wV_V1(cMOq`m<6A3RuPBLcwRr{V&@ zAG}e+lT%XaQ~bX_`9F{SU%>odzWo28Ts#s8{`1D*@J)Qc#y4XMcx7&x+tq%fcK;0Pg;MMmSNwJaydt`?PRpU_sVdrrU)Y?d?RvN>>(GQHB2(W7k7g1LwJ( zRr*8&L0Q=EZ%rWmuYuWLK}A8;a0J6S2wZ!Khx<1SIbqrSUXeo}q$>#MD) zjoI$Y-!j?NER?x=m-fxLGaK#2%}|Ge7da|Lp5LKG7#;#RPbJ0E4MeA#(b|Y5MjCVac=!s_vx~Ku*31Q!|u-yH)ejQZ}h#z z!he3S(Tpe7Tr=nR+WQ@l8?Lo4v@e>C4I26KTu&2aM6zIL#$K+){hfS3dm;N`=IgL0 z|0i)LcI_>WrW2>EuN%n zwYzJv$~d1u7Ojf3GTrVqLKv~eT>Qp`e2ad9>S0wI;5CerjCsUIlzys_*EVZp5m=B5 z^uV@mOb+U8Ed6rXpvqekyFT1+?7z6eZbag+BdsQlnF+`Mr@Q&@4R(O8cvX7W!aHM` z>F;8)6$17bzmNR6B!Q2R)5DkcC0)~hjlFxc+aWrX^!eck)(@R^jk|&!CZaC+M=_Wn z=cRx{zQrtGi|iXK?<#6$BjcWsZ}JIWnlsv#67prdbRpWD>dB|mT)*zTzM#CO>b{P@ zzC=r!OjNwC9y)1gs^k-G5ii!xc9OX~P&8=@@ATdi2?^SzsfDg{PhB)8cvhybT8n?v z)7jpet9y)|k2`K{#lBd*DvfgdGEV&rc9d~8oNjUXh4;%}lCeL>wVUt^!L|KB04x1P zSic33YouMP(?2(Rm-6CAi*QL&A{yzACM+L8AzK};E|%Y(?HXBR3dpor1$3O&@Ci?9 zerrSd5r$@DOWzsvJ6p9Ya$CFOS?Q}g=$)g-t}TRNeFowA@vz%Bfmx_q*k!6aAl^`F|YFd*r)`Tkw^SM4)(`v__DKP$EXxv!VwYX+{$ zgRk^^?Oin%wUF>oc|}*E8ox!YY=zFYv+8!;Yg`^*(*f}%P7IY1O~bwz!;Ul$?XIK} zLm@g`^(tY#?|`b+@^$&|GX5*K8!j9)^6Zj@D{*=EMXvu`IW)MOjS$7c58ExCj!nB; z%NF9!W42H%{5XV>thnpdZg=_yD`|)9lenH!HpK^M7%ihbWV4^I`|{8q;Tx7VIzTUw z1;(;dj~=NGz$rCkw$Y}M_iTzX{Az!3_xt#8{F8?{31r81o(K?-LhU9f}sgDrM~9TYv2BbpCWW418G<%QMG~ zCp@1R*;~q}7Og~CE)bE5AQ^qecYEPRjc&>v7%F+94zF#psr=WwO|ubsZ&{DGpHYnL zWqL#GwAXKE7i{NPYPfC|mD|r9=t3u%6dATpUZ!xf=(C>%9S*vhc($O-w#Hjw8j18n z0rm(iX9K-O@UMhxnpcQj!f?d)LsiXABND??RPj^L(A-fZEE{RkK!jRvf8TB+{ITQW z)TZ{L+^$F~d0M(){(POrU_Xh+ZvA6;EKj`QzU1QI?{HotZVljuo_V>7rY1Hw2wtMQ z7H0T2+kR0pdoNDU_#74ZH~P$Eyj0sqNX6E4|I3&EPvIc z7ForLdR8inD+vbe-1I)YP(HMfC;GS*z+j!9^vrLs>9ugk2Pfp`*nQH|MpS>NgL3!r zl=|VOKt~j=jV(Z4wA{@7UZZn1v{HjDo>g{m>7i~pQv~r1=bStYGgx0_h(cGb7zBFL zSuL6zuOvDBkeIttyTz0&Pr=l_i#-oq5p&3R>(6Sj)b`q`svwR{4TBG`X0X=V_ZAh| zwT>2Tk_YWiG8@BpjmiZFu*_9v+ERThWmAxenrtzLT+O6yeQ^R}3cYHjbdz=ot@Frb(bNxE^_&EXUJiB4sB!5n!V(oq8B+4&^r~lj;{%*l$j9l zCgu%maT1CsWV22B!bHwyhibLw2u~Wn7>X!0P(N|(lBzOWY_rn|teW)Lfd5!!ohBRq zB-aEhNxc3@x(Hslawec;r?OUlcqlNJ`wWXQvc36kQqMFB!>eC`*JXI5J^RFfq0@b- zzixZ(mk;LnBT~tFZ!Q|j6u5gHFDk#(?Dgwnb2D!-O#GAYPO#N{0nxzwZl4)+?fdw_ z0Nw6)Kv&!Wd--jn8-#!%OPO&)52C|I;>LZdm%(=e2(Fk2>Db8<7{w-pVHNL!G7wmP zyNj8lKzO#0Y>f!T2FZA9Z>KM+FeCWnXEL;xBkfes=n;1bQrvGTjTk-5^Nd4&aEGsq6^rU>|(H=>4vh<*_#4$ zf;j$eij>ieLl*+`M)jToHz<7%Y9ZY_Aq8g+n>!pMH1Ya+&{G5@+ z-cl3}I`6O#(F5!FDJ>8aYge=1BIVvthO#tL1r5B*Z`VmBdR@TpfqG~J+BnPw%#mG9 z6z>y03w}S4_su1jlKbn<@x4Vf%>JpU%hW_d;$*4Atk{fxEwB5oB;&|L5N_m0axvtw zhS5%Aqx+)aRY=I;)cpBb7wl@&Aurqg-B8*Rp=iNt`nu_Dr>*hw;%~R#)AYpP7|be9 zSRou+%JKex#9o~%h;`x?@&etvQP#|V$E_ZYjB1+fZ9Faerg_{<%6~KDpi9tiC6ic_ z>|^O+{IXuLC%N|}y!j_t>e%Q3X|D_0iq*64Y=iSw1M)Mk zt4Z6kW-WV(rPD7As#@~YlX79`s=t}MfjoyK>4oqFMKaG)@{0&;@yoC4fnp^`^FS1Z zJ-c?UJ;C#AO0)e}-t3Wv2hV1{O+8$>GNZQ4=$mN?N^hEFSEy90l(~dY{!n*a|H(q6 zOQ#ieN4T9n61!I`x$z4!x0gplIEA~p{?_28{$3e8>U`u(%YhK@gU%r)aicNmCoL>n zJ4^i6$%&GgK(ZDouWqmO%Rn;V2R8dYuMoZEb%NQ4Uf2(E%#v0F>ZO=+hu=A=KCXtwbtCN*_wN#9Voz#cTWrgUZ*N-HJRIf?F)0shy_VOWSxEivs(j0L1Rk6# z`7`5%jEduSu$lwQ*Df9I^>d1ieOec{QL`xqs$BJV$=K0LU{YHx!R_X$)9!Z z=33>F5s@jXqJ1MWKE1$Dv=gCWYG+{EV*P|Z>rt=F(0k5pwV}E{S2r)(Q{x)?_@fq( zPZEYytqk($|0EsIhIrz@i74)GS1}qS0oX&U5U#yhz8}_F!WDjuMM&J_@W7{=mGcwd zmCs~SO&3psWzNM)$Y36Q|tHxGq2V^9S!j11K133kComO%P>-J_;Tu>GC;p;uj zEk{-DkI^D6FJu4@E2i+F2q|5YuXo6z6YgAJh~3FZbCq25^dNKVFXT)jKmO`@80>TM ziGmd^vNC2N^#s;{uF9rsuGFNb8NX|sDPq$g{TM^iih}+P{ekT!9C!#j$>VRoIW4pPpv#&mKxZ(zF0eFHuG!&ppP&qkf z>y3Rz-T|>ceGO|zi@fzb3tIG>|9lHupSi5U##b|NvEqkaK!4Y7=e@vmw!&ul&9cR< zVx$d{Y)4rx&^UT>t1`E>H0Wi>G&;JjNFXl&1*IseWzb{0m6h=;g!6QCzOcJ}hMLCM zg7Hb;Fgk?nQGl=G55eF1FJ8S>;z^Wvj&{EjJ6OWE=o#OQDqVRwZ2Oj-^qiRdJDSDHOVD`5j| z?T~O+(UmVrKl(Go-bd#`|xleg*L5lf4Ufqry*Rid27c z7DF`x!BJ7NMyr4P{Fe;y(X){9cGuEzB;nK$tu`ys`>K1&`Nd4fn?30`9J$T4sH|Cs%Ey#l1gq1 z->4d1GIfS^4#AqO{g*W;H`JVP|3vEYxhqS&*(Y(3VR4G8065npVXLJzy zd{(*~GDMo@a+tjl7LVZEw846`|Hzd5_ay6)I$q{fGeug8aI`|0WztVOb)rCI})->tPmY-;jgn)wdP zb5{h=!ED@uDK+8Vwh36Kq=A9}73)ikA8L`gX|6S2s-<+>*K+VNY<+5d$DH+)S+>KY zyFu=+gpjxiQ#bsiGmfEUmaM}Mz3;ik?k{r!n(I(>H=@vHK(kI#S%sqh z3HIGrt#(|Sbt2n3CiSC&RwxA*K%0x-)yWCaqWwoDfY5d5d!YaOYws~Ch4AG41>8$Y zfOox%qb7o3eqn<~d>)$fx^8te!U?R>Vfe+r`u z4B6EB{aY{KO^@624N)KTMC%uDN-jK0Tz(Co#pFcc%dKz$7wZ1R0De8iEa^SQ{EDI9 zufM0@-A94GlbZdO%{+OEY=?Qy5>_-q@Rv@TW;e)jzlGwST3tM9)p!D}xo4BDQ7Xx( ze-$_Nv?6zqB8x1!(_Smj&S5Z5I)L}vA&B*`McERO4+x@Kp(AcII~_85b@2HtY_k@! z_8fjwdmq)z2vOo8-ITsHOiMwo56A!~ z&UIkC$S2n;l8E{lXU*fOdFMa^^1BreTSkh$*ARzp!a5;T z)-f$8ZZe|KA$h!Ii!|PVsZp3U?^1g(Ct$0McV98?t&lG7&NL7Edi90p731NWxd~@k zndRhT9Ff$Kl2sy6MH!yu>_$11{)fH^e|pPHQs6|VEIwNk+0O@G6WOJsFbS(=8|h$f zU9I4{ll;cSZ|ffR+}d$&RiYMzTMjRgeTPvxWk6_T*x$6?mh*Pe@gr8@I!p*uhcr;}Ng z({09!i-+Af!$)6l19PfxKR-Qr>tSJ`W~cs99mfkKWU({pHaE}GaoYo@^AEw0R1|F} z@Zydr_01F6eSVSi%g{hxhw*!om? zJ8~dVT5nb(`x!FfQkC!WyV?z4WD!~iYi-RFv6!R+c4Jro?j9?-Plm=X@Bsa%gm|q+ z>bI&?)yC-07o_zNV=M&@p%0NR!j2z(ruc|L_qYJtsD}c@+BsH@yFQhq`VSBgOqsFC zIBP}Q7cu}!6(+!vv6wEfs7I4&kSD;)Wt;xocn6JAF2V~HR8G^&efEa)uS_ilcPzG% zh?TAFPN1P5CRd~#te_zS$bhd2L!Z25=Z4?gz>#B=+#wsUJ=2lerHPdfh$TxgIatqlW@C&SuV_F)v$_gDG5u&Nb7+KAWMj;Bo$G_v5WmWlH zq4D*FUjR&t>CMm^VM;F?h?f^5IMMo3Y(~ zRgb+c^2{`vQ7*!eEH6cg|2kmn&pqFLc3BAhuDMG2W?w8|u9inc3dV~qt zD2u*lQ%yqHipLE=;#-Sefs2! zNsEHW65|Eg+5Mt-)jGz?Xp#Px09Y~X;x4_1R=SZeh)wq;;4@(@FBVaFx*ZQkG3%UElC zZin*L*?Yspp`!l@h)7ryNC=D!HKviII7Ky-ZqI|EA2h{e z|FG|dzYaDrC4vvg99t6uJ!-*040pZyMb3XXche9Z5N5fm_jYlCqdwEDe zJ~8DzX9D*YB3-sWd%uDk!X=Va0LM}wAJ9=K=_>g$?{ok%Cm(v^V2kG=TCI9XxO7dAW51C7(2ZOY$|Ev_%fda>Vqg^O;g}jX9jo^A2SAlq(Q|53vSt#nx4D5J!;zAppWRN% z_Qv93WAaIh>tfZs&)$L%TKc8#zXS?(2)qec(LcJ*QAnS^M7!8@vi5WBin*FSEe|>H z-w|<5H3afE74R%w5F$Tbg3Z)q{#U6x@y&bAekDG#Hc>;zqp3V6DCsK=WE1xJ|LX-l z;NajNB?L7M#HN*ZcRp_`qm}x-C;XGZY%hj~u0ebK(Tf@F)>-CQOb2AO06!G=?-I!{ zvZd3)%q_=t>h<$;1DA0(8YwOQ%+!=&O#WTq9RQ+UTI3(})1y_XR43!x=rfr81jjuA z{Og>-M?O5wl>)+UQ7BZa0}m6nMZw26e3(|9)M2SsUbi9DTbUf7<%DoN3pf&oRVpF1 zmqS8aCakA2Y6r-h(t|}EC-mNYr2~wm=*0*>YLfxYf7pQcn-F-e#6X}AI?#4jb`yfQ zaplMD0loA_ae!3ge}bG1GTLRn7Pn11bmSy7l6K6v|5oz)&}7(T9)~wL zUOn&zXR!v6eRFz<_}PvN>`gpw;ctKLd6~=DNl~d9Ug+3Qhy62a3vbTT_KyZvaq=&l zI}nwfK^e7|!DQjjgDA2a1tX!D_|}tG|7aNUrN9tu^&}aP7vk3VdSlC^N59&m#ilV= zAPWTKf0*g8HJ`NAlpkE&26O1!ksG=N@ki} ziSLiL^UrCjM;p6ZqF4|i1jIsDh|5A!kJ{p4Ce|S z{A`+Ki3Q|zeq#MtdBC-{5xwLpsj+)vwucC}UI~osR*c977qNfm*YXSOz^Z#z3J_~G zJ%q5ga`1Qo`N3OD-xJNAA0qer*oq+!8%Y}CAoC4+THg)-TLC#EUdoUIbx+T!?pcqr z#OKcS$jr{li0ge!ea_GQ639RJE%QRuGa)Pr7QO$Rlsw#2HZS%GoM|cf0l*{$Y2>$W z8VyPee9i$Jm;?r{6uics+^cK1D^5@TSGImv;(%ml;q>ipL%REQIM{NrGBr@q1O zK)`~1iGqp_90cZ<~@nN+5?h3ehdCt;V!(;LP zR`2+FIeVsO#n1wCf7AagQ@G4z%Le^XMs$*3XX~dNG^S>l_YMr%SQ|_)R{xY1J>NZn zX6s{x`m7TP$eI;`6MXRHqR8^4nr1&l;j*l~Dt{Q4|EuVBY|ANAw4ZHsk4I@YA8K`tl@K+e(psG zG}PM`=-TZNm4LCCMpS%y40BE5V?vxgqSe7xS~QYKRiCM`vDTS--qoEFl?UdsFT^bk z1*_gyC*?g(2JX*)bHocBY|x%5bBF-ZoiX(@lRiSXt3_6s%jvzv>fzDG!2vdt@S8xH zRuxBt3L(HQe!?pLXul9e_usRyzOSFzgV6Z~A)e$B(HVfO-{C?z993k^_gnLWhB6t@ z77-ThdHJERpSW}2n3aH$=oXJA)NO%DKcoDG%$LijoSgG(*I>=2rVWb|qW^mZHuW~p z(>*WB{qlmLr|a6c93Y{oW=tod&^;x+uWrE!>H@Zar5e{fu?iEXO44Vex&Mvn#{fV! z5Arh!WjDS#vLM==&W~Vx`!;-fD=L<*`&0Fd3xXVB%53flO5DZ(P<}JGl1@GCvWbq1 zkG3K?neT9dj7zxnyvUWq+sERliiVSLUa%?&3l`@UgA|}E=tS*!0b1g^KV;D0g1Y2o z~?3=#E~Nk zB>QUaY`C^wNg;hQ$QT!N@H%CD-urW@<-kui-q>SNbT5@}0#b5tl4aViSoz&|cF+AP z&FhO3q;(73hDhQ*6S5%muPc&*87@xv7JQ!$&;o5by7)kMot=k8&13|dHrBh(+KE$| zkh%5vY3HC`W|RVP=ssg(hHTK4vMrD=_U<4VEhRt{Dy;t==rT5+`Q=^u3Ff*SafU>~ z?7I;}P`(!2Ib_3c+&A9kC^!a?P{ru|JB9lb={eBs#sOHfkO;zxaYylc~3)|kT zro3+U33s|xSE6V35HB|Rg)d`c-65&h>xvX|{F|uDKbM#vLFiW;C`m^u3NMR1uwsWB z-NFY_8Khl=ipLs+&I?-xy8{{77bn+Fl;Q(CF{!Z%L;$xrJ~@vH0!xsG-J4k3Xih+` znlpkui-L{~kTUrGwp?SBNGufQdLKx7lgW{^y=};!Q26!ly=RawX2x{E98jSziL#=Y zGINK-@iKGFYZF~tuN3K%6qj;!pdKvb^_k_+npJKmzDkKKZnI<#%8VFZjPQPStn@mIU+7 zFqHLV-#7cdl4@n4;EP8%CR6|GcTFxaKTMM(YL}v5L4xDnZ_cJk8eHbvD|EO>0B*q^ z!MwHJDTjxNrxdq*Iy${)(KiQ$G!h+K%r>?;^#RLVx?#7%9*S(Uf1@lia45)?_J5J6 zi4bx0M{6@OzdI(y;>ms<+n0#>F{%kxqT7E1J#}eKnbz5O@kQQ8X!oE@i zbADF;_G)YoD6-+Z*Yf4{NhaiwHvnRp7xn72ay-dHk4pj*iTEx`fG+7e*f7+@ zzNs?)ee+`J&wTcsSON;1`d4^zk$B4AOCs?E2cp)3 za7X*@u!5@#<)hzCN?WA)jeomFk@go&(G8|J0~S#fH+W}Fi9sR-_&_HLoSF70bI>k@ zoFE?bfDo98pLLn!E=fvzoO96==Y7&LYR{F=>_BdO6T$l_W4*(Vcc!%CaxB^vjr*-v z3Upz5Z1eC4A~T@!vGtk*ZvH}B$14p6gF9qc%!66riO(FT_r_0MSj?JiF#YwF=K;Q9LhUhRQK~tr6cyx&R9sY*Ycw9&IT@x5Fq3H|fs3al2bto>3I}4ZXL{gTX$Zlo)M^JTDv~p$S!>3;br!qkOC0_bsiYAB zc7vPAH|K4#e2a8Zcz3ZNB9-Cv>$AEpS?mxCa3lEUeAR%$w2opFD!g-`Pd!bk8I!7w za4~~*IPwr>TbVl&N| z;5Sw_$pc1nrdaW%vSyq8^Ak{PqXA0k{?}H*DFL{Qoon#-6EpboY-Bb#mn{x5AwvHV z8ds-ho6u;T)MpeQ4Ijbb_SQ2q%k}8vaJ58LEXxzVcb)4UG7J^*@uyVryfQT&lVhP!T1dB}V&**eZ8Q|>)3awWR%N<4?7Y)XIu>NP>vY!qKEJfuEYnHvs)-Dek5ObUeuxg<5^XBnrSG+1=Y2B0&viBOl!S4NHJ0k|hLa zv-Ue@&6PPauIySFkWzb(qPJlCAiD){%DE7g2hO~#jYq%#lMETYfQ$zeeYE4_VU)S} zM;?%+KjdRTX&QbQH!ee!r6P%^+B>LIg^BDoH(|w>ie8F|4AM((LW4~BSm|z&r`XW> zrn;Nn42r|=sqjKk(IQ>E9EBts0??P1J~wQhju+KqG7{t+Ij27>n2v$^tOo3N7zNwR zQZ26mo*dqNdc?IbXF$i?EQ|k@qK^N@!r#raXy@kxSByh2cB!#!=`lCoQNcO%aWff+ z2(ivN^6k_f@^;ABHu)FWpX1uVio-Swt|8Lf$A27lvK)5AL@B};zhTuc;nQGjH&A4c z)C)O;;V}!7gutlQ90nD-18V12WK?<-pstp@Jspn)%=2F8+EojKpcSzADB14Zv&@c+e@3FF)I&MvPF27M2)ob zNZm>j_Z7$BdghD<8Ij=Nv7B_LC1jqCdq-sjqnYt!Q08Bf0t_>9@0{T zbw$~kTn{b@p#n((%fh^S6j=*7>m6A_-wG*lP)L@+o4ZC8dgXtSdtfQK323%)@o4oUJ?xybv zMHgP5Z`^KW5}5z66HW$z`7>;Gy>ws<`+= z2GoJodQaVHWuv6y+-yJWRUKK?U`K-T&OwE*wo3q3bW!TZ!K%C9OEDgMWi?bmId2& z--Pz1KRLrJs(wi%Yt4teH&eU5Ghw;XEY;?ahb(-L)N58vW@coHMT)(ipG-%JeSeJJ zhWi1_?mBmbgqePsbUT=JI&1cGxO0OM3$pxFy;qa0gj8f}xl&IW1F$a&Y`^AXNi-jokyUXc8p_eA)H)u1iutD_lL;OF9S=3At9L7W<7nCHS1QEs>hO=v?6&d;X06L*!<3b{YB`9&_G z)_i9XMJ$VTkH}E^VM<5@5QS<`d*xA9ICUe=HVSJuhedl{cxbW><##Tu6uB1p9h6br z6`nTi(6tj8TmPJIx`wmoSI6;7!&iyiRdT=VST~()bzDO0HB(#SSBJ&}LDfaHH$LaF zy1oRQveKRZRmz{*A|Qzvze_#H0!+)u`#VLek)MBPreEcSg(Zflt}EIx*2kK#Pl9NN z6#Ug=B2`~k&vM$dIy4O*&jqLpzlk(;Z*TqrnS9=-Nz(u2gpkjwKbLb4f#t3P;p)iC zqYs5AhgBmSNSiYS^00U@nd>bT1d8Gs3VMR-zEBVeC5wH+z9jWL0ba1AG39U?Z@r1K zw?Hc z;@Ju5kfRKFvYrH;Kxr8z0Z?w2OVSSqYl*Y{Xof5QCn*LAA8MS~>+plC&kvzd<;F;d z5U(Tu4gvs0ilxS?=>ttcXEo9{qYLMdvmAv0^%^SJL^L@UJ^4@l#Tq#y%$S>`!l3u{ zrC(>ps|z_@*_Z51gH2})qW3cs?t-Hy!E%u%{WVSx++o-64?=&ghS-dhwCb_Sx{au4 zj(}p~kw}M8c4jQcfyWR-GUe+DY8*AK`ZK)z0v6(wn~W(sRvwR@C<^N?zKfXcc9(Ek zO=&A1+zjOikekcsyM(dzYcOzg9pw^&&7UhAkZV>{h9As;*#Iv#k`-nOOfBQBr~8A1 zZJfTv?g`E_d8tr2f|B+HMQ^Yy)m5A=ro0<#aCRB48J5pXgZ!j? zRv+0h=I!==IWgyWUdr_J-UN;#6TcK2pQVzl>3z{Ku6oq1Z_&Xl5U|4_Y160PX!emz^G&6qY0S6gKT1HXG42>dRxa81L+8);jz0W(P3qzO&y-BBRMGOF z&<*~RqbpkL7qicNxp}DhjxQPLa_LzjF)WL_E4Uzf@=>#Ct#eAeFmU5$!d$ZWAv18d zD1&uc`G~(2Lt|ZXvL1Z7-|1_S7e|j5N}li|(%z3lv~h*=C#k$k=DYiMTZ?Nf@kzr; zhZnYXtR(cNf5)1JOA|jDa+hs()gp9Cdl8QI?{{Tvgt+vQh6lA+NaZEQ{3-O_e87r7 z?%E%`P(5Jzl%)aO5nO+^=1|DyJ?NiYO5Gnh6S$;R1da(c%N2QzRb0pVnWeEcwK6bx-(B z#{Nk{P(v*#z;<>0K011hV=;HOgmkfuq#WH*Nf~7VO&g|W~1TIs?PrXgnv+D6ida@Ky{c6bB&17(%e8KiFRGj4{kOQJUh|Ya$YaX0>+q8QqMFHA7M;+)IW8uoS z5dv}Q)@H2&2|U&uNq2jVYzDEb#6V(z+cDsAM~gp?(~|=FB%4qnWN3L02@u>>?EFh? z%y-XOjGf_$V-$mseORW5cf#vCv@-e?swe2$2386$iQf+gv>gbpqAVo$UF%D6$xtX*io?Rk(-1$GX4opg_}f5)-nuosR%mAlqNfIjK zSjF~X+=@op%0%oB(P>@p0xt?+xw+(?K6($?@D6VzC79l5u z=96yB_CL=TjeB$|^YPm_UVhoJ9@2;M_DR3U?g*?ERZqlE#H3!^^t$O=CdKtEDB_Tn zvapr0U+3thfTIckTi$3!$FZ4xd*$(#Tx!}jX^S`W(=H2>hLb+5f+w`T5A9$SD@d8p@L!uY`zTYN z%V^9wlZGgYMi+Dy1H(`mAocuUrOke85;p$cUJhzWzqaAYna#KCC+ql zwj0w33SXwGO|&KV?*2OcsR{r*gc>j&qYh~h( zppnbG5pz~L8DNg#W4ETB;Yx@z4=i}u7Wp{blIOCrvs#w8+>zz(NAAHml1NYEv`vFb zY*|jp2lZVZW__r>PTaT0F1K#$Y;n!C`+wj6taj2q(XJ8RPw{%o(5wa2MTA=w(x10k zi%DC*8EnthX^M+(XWS8XV|bm!-rFQYgwM#Zj#0$7K+SR+553+{(lq<+XGxp#%}*fpM>UNb8+n`% zW7M%vBOZ6(WUJm~SB+|Y6|LCil_<91KAx;FmH~_B-LO1OA$ViXL*C4{cRPytJe;n-QknKXU;@h5} zT*x8q39IEb*#;vh94hv~5B*(q7cX=ux@JAcz$N6O717%$U{T%r2D-DfcmgRZ4gJdi z$jxj1cxLz4M+nx>)cO*xHKcXm-hJKgxhs5CLUnt-4%*Hj{mfr6CbawFR=|gU4p?#z z;F^n0rL&nT9*yfo?Tpt)Qs7mGyu*BCrtdwYA3SKEe4}}Mx%mL(je_MG>}>JX^&1@o zlMb9T6R9TIhRF&itpq>p-Sx7ha>52E5~l0Bn9IxMI6W(}3a0sF5dNQ*L2E9E50p<-IdEOH`h}r< zKmfUc8D-c zCf>4&yeIfpS5!0)=M%0#BG*%kS<03pkYj->X~_{JhvNAI{>}j-;D5AZ0m8B)a`GIb zAG!Ro4I6rbcpoW{NGb(~YRxFK_}Ao$l%TqI3Dk{K27q0t&kR+KwH2ixH?t8hM{9{+yLIk)gSWeTauG{^*AkXyD)#F zEi@U!tg#@}->pNh{i((*DieO+UbXvMa`7ViHC9^jNKPkIaPa5^QiPZaPR_?~ zHQd&iS%?p6QHE(On?!7x;#md4sL z`-zO^9jE?4e2$~D2*i;8@GeMntMMYnZ6ODX6GPgyps)Wr-wnE|ed(ac33<}f6zSdKC`s)B{ zailNwx0nDWaeF2~sEK0$QO3M_i5*Sp#+YvnG69WK`g}iA+PfiVWG@&^J;qE!lu5Yz z-_j|Jzr}C&@KFGJOt=*_{;L{_te#yMb&3IcSN%sqyom<*s9~?F0C%~WZH7G7-x&C0CCClmW@|Z{f9TFm&%=G~EY$oGGy#_zhAUTA zhonofErW81jhtUDt}7q*YgkU|maEE2#wgi0C`s}AZ5~qtb#zs-ppz{hvLz^LL1l$d z@#~9Hky4$XnAhw8&WMXNfnuje0d*j1RBWLyvhVZ?uxN+WSkMN{-YwK<7^D?S!F4>n z?~Z**O9Vt4d>{bS-@Nwp-L*by*HA=He#7oxiLGphkO4&cAMgP_H}2WZXHcQ+nRXkF zgNyy_K6QE;i59<3GYp2sGm~%QwG`?*L@M@q)?RVfpn`@2N!BEC8$t>$tE9$ zvCP)yq$CAZ$Mv{o#XZ+Chhp`q{}dm!GSHDdXAqOHzr6EU}5z%20yf~Gzu@YCJh!|OZ`eZ zxokq^VT@gCwY1AWlR=I>JCg67lWr%q>E^~UW7y1z*=KTenyH z?HkV8#5M*lF2(b<`e7Gu<16@VACy@s0+UAxhdfl!!)ok znkR$;6;8dm7e6PhS~HDv^BR2&>Hs+p>qR$1i{{y#Ra6znyG2QpGT;DwS`0~Ca{vCT zraQMqFxyKKGxCY(hp%ve*Hdm#j2y+JWS!D^t=0kYa#{RdwB zeC97zCrS`}GtVvm)2r8@8XqWr$`4Ysl$dI=5|=T0SIiH^ZXvC+tX=nVP#|DILg#WQ z$_!s!Cb8?e;c)*;i@(BNRR2i-sO_Kv0TH_iM=fIb61V+ez5@hE8VWp6vP#`p=WBVp z?|8S{RJs4X&Z!9EKJ8SMqROv-S!<=Y=t^ zDM=`Y%P%n+c8x1fF|~ViA=l@OhmknD$nVuWyIHB<_au%$;omnf@3_@-sRGn67V5!1 zV&ZWX!-wr_#@C#Nz!uZRr7NRoHY+;B9u|{19<=hCt92E!_m4Kr_q(6n&qND+%)h-7 zVdY|EfA;^hcOL$1z3=}IwRi0q)hd14qpE6^79G?mrS?{(wotKZ*4{-!C{?TWj@mVY z7}bg(_9#v4AjbFj{TaW<=U+JIKG(VL>w3PP_x{2;stH@Y;J9jQ8WTt{Fwwh<_3}?9%pR&$+ITNiAbX++qsS$F zNXB4j(`uJ^yD?*Qt~DjY?|ZFmp=y&bzgM+xn~*En%PF9d+lQm;f5a`feBf>s4uNRDiY}M%_Mjf;9`cj zK~D&|t%OD`C*f8~KX2jigJu5FHVxR|W#Liz>47k|q9l zCY@POO92XSnI4D%3ld~6k8>$EiiQZkQEWO63?LoThPp@A!{&afjQ4|5&Q+Q(=l-x8 z0L%eAa{6h=0a!@%(S{8a2(9I=3{{5cqYP?DJ6*0>$rmT_Cgiw|3fK)_4nQ`mNIQER zw$wL&+l#FN*@zK@k&COJ>eiz|k*&x6AVxfzfC@Ty>GfTfr)6G^ZTh4UEac(62Od^h z4VZOFP&dgBoo;#myqhQIQnRb(=-(63k>L);XPn4OGQQW!^OIu zM8~p$q8NCbjbR$}=RxO#!i|-J$E{U5(WDPUmtR^1ZW`4&kDS)B%!cvHrN!p)tRn%; zWq(~lO2^Faqta+5e~g{*YZVQb-$8jyUE$q1T=}dQIHF)T=NR&wA}fuOvA502KVf(V ztP*JWV!8Gz*=k=w0deae+2>eRk!?1n&J&$Ch3~ZQKb;|2nAH6jfpLxQ9(q*8LHrD) z@igJDFNyhkGwe#R3Pqm;;DgBk1}Ee+tp#8V0Y1lZ`}Hd0rzdrh0co$^Ezcv3zJngR zOJR`0Wl6{Gc@`aR!ZePu7mV)LK;*Z@UHeRai-lm!l{_b2B-_criA@C&$Xt_ zOdSZgmXxIOk!x>VVszawyNl+*n4wV6SdAGI=-#^+BHUI#rV&5Ub5=G`&g!*06DqJ~ zsQ0^?X7F6;7&mHB2kCta%V{)QPV^O; zuoPwY{hkrYaI?V`NEso)>{P(d9_S`g2PCsT17eu|wxl?otL$py^MRnMo%0QFeVPXI z4;A2d-K$0NJW^9_A=q(8jES>YCp%^DqI}zA{+Y1?p@gX1YYQJnRvk#5YHGA_DhK4` zQX$>oxb-giuDTBUMcbFMZ|nhu$yWma##4n(rsfe`wdkufDH-~$WR9@ zv^(A8RkV>IA^LkLYbVE@R;v3HzJ%Ac-V}EpG9a|E{HF<8_|A22a*61Te*0Nm zt^z6SPlui5rs}{i>Qu*H{vk4k-QOi0=8gzzxA}efE{Ssl2pW^91N&EMq|fpa49JUK zu-n9|>d&kSGzb0-&nK4_z%^t)V*_>LW0aYWEB9jH+z7YAC}$BnP>E@A3ZKkPG0jLr zb$*-d#qW;^(Yhv!>Kw;^+1&esbX8AA!=UNZE!)O=*W2sf9&*Eskj9!|ie5GQZKfww z!wwm`_l5v4W2vSX*v`Ffoa@1T)>1g`N2xTe7DL5V&qqxfd{Nu;wR6Ey*jGB2tRp_8 zolB8%|C}IJFbT+|J?5xnFEJqA!U}-R`3N&U8LprW!c6`Gt%{8w!b|mjDme)J1~G$9 zwjWw4g*el(QL%&+Oc=^2nb)^h0%!tbl=VH;3^$3o&C>jh1Z(q)Dy!SU9uM(kPMvkp zAj)pR&;JJ5^Uv@;B#5de{PjTN@7M8T4NkKzvKb+4ood^}Cl#EBqA?2h)tUsJ?!MKk_C_PO@& zW!4Qkh>G^TpPG8mCVadSZJ1YTo7Nr1IKDSSvY?%uy4lkPlJIxou`8sfD6nQ_pocvP6;PhLsCKX5)rPJ$AnCZ>_P1R%i+5l)ADFZMY`bT1n+r63 zKUxsFe-=3)v1d;bTBNuAP2OWn&urKG+Bbwifkzrp?uLFSgrscF6N?rcUD99n^2fOa zAr6mo2E(YIW-vWu)cgaMJoA^3lNVEFk~|6;0Zx+}$|n35#gvTO38_iDqreE%<-Yey z3BA=pXSI`WyVM8I-qh7bzpnGvMQo)uNhnPa@|3<&FHVGPW7lR+L{!fNzug|~?k}_* zWisiMt*ZJ&={18AhwD&-?VwESVjNMWC<66ZVRx_>NuB38wxs<3PIkEV_OT(LU*A!; zWVZ=>KGbM>sT5FtXmao|*l#_N-q7%ZuFBrvB;4W!+E@CoD?=1NXH>xtA1hd8QyA(H z9nF7xdy4#!La~2>xY!On8xGJ0*y%Nx$8v0rZA#fqcn}a9oVLqI;gpZ1(bp@WD3!JF zaIGav7=x*~q0ra+eGBMu$#M8t%Ufo{^msPU5-mex-H)-M4iB~I>^t0rWFh#ee!#cX z3votVf(?1=2dwyIlq8?Sfu{Te+8fZ>*6lp(SqpCs^HHok+qTU%F#;Y0GF?Uf1n$=RXUzunwve!5CY*C|2Py%L28piGZK`i zE(!tV^Dd8xCh8S47pv`Hv5!@6)!eb;dN#|PZbMh@h{@Vgwf?B^e!+U>L_&M)){U`; zBlWYZ3XbjEj}~Iqd?Keg<7*}6ik4zll!4sMaJ_J@ezK?3Qv@5y9A;`uuF))MYl89kp5nx6H zGgt7b5fgih(=mg=oV+Sy<>6wP+&esTj}jn0t4T|pp2)^ZmW}ma(cs{VpBKzHz__#* zEl{{jWc(RqB_Was1?IzM;p;K`LepL?8kiSHIoH-UrJh!s977XK>uZ!P>Rp}vJv`_| zqb~db`pJpchf=ns#fHIR6fL3&s_sDmzJO@OQvP|Gp%1jeFVG3ll3!`cx4w8Z(20BF z7+)9V02oV^dlMcbHTRA`ATKWdQp~*Y+D`(ceBKWB1$#kK```QpsfZ+4+;m3 z&RYdcY1*(vixCt*5~rv!e+{IS0@pl!alVce@MuuDW3Fzc6n5xtOijGQzY2)>TW`N$S+&;gi7_Y~)$SA}=3JT?Dlg?pb$8 zwNYu6N6LbvTA{mPy{e;I9^;}ur#sg%qK3nu&UHbty-@FuRLXaUc?9Ce=${zRm*z{U zp3JKe={%bIuaa^rD%GDdrM3GGHsE54ZVwPkj5XP;>o4f2X_z3&-c?9rvRbXOKaDVN z&zM!1#2>-SSzKRUz_d7sMw2za66d->QK3 zw54!Z{FK^JJL;P{ij|zvqW7JmFLDcS_B!HA{9uZ_r1{71IU)~m4M5GCsk^x+H=J1- zNEurER*q`u=mR8-e;>BdjylL*GTqRzc@jNkZ8&`*woHA8jxf#{V@hLk=S=w?)ykG_#LZm=}V zhzMh$A2)nFYbSP&0r-;K6!_NHM)MU#ke*GQ(q}mRrxwi$vbe+LudR$#m?)O#!rAwa z%Y|jFmh&V6&NS#{xrVSw4_>(c6X~rOW!$Y_dkrk$_2JQ80w`?fZG! z69&+k<1c|UVVwpC%UbG1GP_s)*Q2tYAosf*owXB1eUhk9fdJ<>Tz*cO4S141>sDDb z9U49m)t%^Frxh-qU<&l^y18#&>;`cl?1H)@nAac=(!f;jGs_tx7< zneLzwk?T7{x5DJ|zvFe+HOn@(wWk+a?e}=ZRX%k}+nsfdz}A4YGbXp{4O^{Pa&@l^ z$Al!zkoBzslgszlw??LAk;)cghxP`h(#HDfG`K-ro23kpj;hm2-otqCvrygn+FZqj zS^*dD*%*jL!g4t|y|FJp>Z$F*ygF%#XZsie{|@3Emveu|aK!U4^v$f*Y_VNC+Gzml znxP&zd*xcJ{k|tc4PNZU27uSfe|mg9gAOXMoBLm6H*2pB)EWfPAX^2}@qjj*9#&Oj zy2eOQNibvUxx~JTS3|lwdaZCOvUcz5f5%6#SUYc58!Q6Trs1pbwT#1&%i_tZ z!kXXk*(4k zr19rRU2RDB9+@_v#$+MYwj!1bU*mdH^Pzt#$;o@^_ZRy5j|7~ApWukHY9zW}`tgIq zZU|!5&!4roOlylCxEfvG;k{0lb=f0W_mri0rZ!T%G225EfAhH~B@(WTjrJ;#$#Ncg#42bTChtRGRMVBB^_KhOfpZ-^d9k+u3y^KNR*|v&;Y^ zrTE?Z4`V`5eGdT>l#Cqry#{586L`OXLy1_)-Hq`Avy6m+4T1cogwRfJBiBVC1J{6_ z3H!>p+Vorf?~k5KgMe&TXl39F$XDTbOW6R;H-ZzZDnNKqk$u$U2NALN6aUy@WN)k1 zf)7ebc*Pxtg;{{mcar{0r0H)Y8z*1M--E4#l%2jwX+@4ynIGtR-F0hxzDqUfibekJ zs85UrtO%J(BAn|luB7m_ebqE+23)2#=%TIcQ=RP`K}>jx28XgTd|1IY&ON{-3uL_U zsYdcjE&tVi+yzjfl^1Zj_+9!90&wI1r1SrC?*H$~f&bmO@c;FN6C0P5+%9h~yBi!l RfXhBWPc>dVu2i=U`#-B3L! @@ -23,12 +25,11 @@ class ConfirmationDialogFragment(private val listener: ConfirmationDialogListene ) { _, _ -> listener.onCancel() } - // 创建并返回 AlertDialog 对象 builder.create() } ?: throw IllegalStateException("Activity cannot be null") } - interface ConfirmationDialogListener { + interface ConfirmationListener { fun onConfirm() fun onCancel() } diff --git a/app/src/main/java/com/lizongying/mytv/Ext.kt b/app/src/main/java/com/lizongying/mytv/Ext.kt index 3a2d0bb..e848b10 100644 --- a/app/src/main/java/com/lizongying/mytv/Ext.kt +++ b/app/src/main/java/com/lizongying/mytv/Ext.kt @@ -7,6 +7,8 @@ import android.content.pm.PackageInfo import android.content.pm.PackageManager import android.content.pm.Signature import android.content.pm.SigningInfo +import android.net.ConnectivityManager +import android.net.NetworkCapabilities import android.os.Build import android.util.Log import java.security.MessageDigest @@ -71,6 +73,25 @@ val Context.appSignature: String return hashSignature(sign) } + +val Context.isNetworkConnected: Boolean + get() { + val connectivityManager = + this.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { + val network = connectivityManager.activeNetwork + val networkCapabilities = connectivityManager.getNetworkCapabilities(network) + return networkCapabilities != null && + (networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) || + networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) || + networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET)) + } else { + val networkInfo = connectivityManager.activeNetworkInfo + return networkInfo != null && networkInfo.isConnectedOrConnecting + } + } + private fun hashSignature(signature: Signature): String { return try { val md = MessageDigest.getInstance("MD5") diff --git a/app/src/main/java/com/lizongying/mytv/InfoFragment.kt b/app/src/main/java/com/lizongying/mytv/InfoFragment.kt index 9dcf92d..a41a7ce 100644 --- a/app/src/main/java/com/lizongying/mytv/InfoFragment.kt +++ b/app/src/main/java/com/lizongying/mytv/InfoFragment.kt @@ -29,13 +29,13 @@ class InfoFragment : Fragment() { } fun show(tvViewModel: TVViewModel) { - binding.textView.text = tvViewModel.title.value + binding.textView.text = tvViewModel.getTV().title Glide.with(this) - .load(tvViewModel.logo.value) + .load(tvViewModel.getTV().logo) .into(binding.infoLogo) - Log.i(TAG, "${tvViewModel.title.value} ${tvViewModel.epg.value}") + Log.i(TAG, "${tvViewModel.getTV().title} ${tvViewModel.epg.value}") val epg = tvViewModel.epg.value?.filter { it.beginTime < Utils.getDateTimestamp() } if (!epg.isNullOrEmpty()) { binding.infoDesc.text = epg.last().title diff --git a/app/src/main/java/com/lizongying/mytv/MainActivity.kt b/app/src/main/java/com/lizongying/mytv/MainActivity.kt index a0f38e6..816871a 100644 --- a/app/src/main/java/com/lizongying/mytv/MainActivity.kt +++ b/app/src/main/java/com/lizongying/mytv/MainActivity.kt @@ -1,5 +1,9 @@ package com.lizongying.mytv +import android.content.Context +import android.net.ConnectivityManager +import android.net.Network +import android.os.Build import android.os.Bundle import android.os.Handler import android.os.Looper @@ -7,11 +11,17 @@ import android.util.Log import android.view.GestureDetector import android.view.KeyEvent import android.view.MotionEvent +import android.view.View import android.view.View.SYSTEM_UI_FLAG_HIDE_NAVIGATION import android.view.WindowManager import android.widget.Toast import androidx.fragment.app.FragmentActivity +import androidx.lifecycle.lifecycleScope import com.lizongying.mytv.models.TVViewModel +import kotlinx.coroutines.CoroutineStart +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.async +import kotlinx.coroutines.launch class MainActivity : FragmentActivity(), Request.RequestListener { @@ -31,9 +41,20 @@ class MainActivity : FragmentActivity(), Request.RequestListener { private val delayHideMain: Long = 10000 private val delayHideSetting: Long = 10000 + init { + lifecycleScope.launch(Dispatchers.IO) { + val utilsJob = async(start = CoroutineStart.LAZY) { Utils.init() } + + utilsJob.start() + +// utilsJob.await() + } + } + override fun onCreate(savedInstanceState: Bundle?) { Log.i(TAG, "onCreate") super.onCreate(savedInstanceState) + setContentView(R.layout.activity_main) Request.onCreate() @@ -53,6 +74,26 @@ class MainActivity : FragmentActivity(), Request.RequestListener { .commit() } gestureDetector = GestureDetector(this, GestureListener()) + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { + val connectivityManager = + getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager + connectivityManager.registerDefaultNetworkCallback(object : + ConnectivityManager.NetworkCallback() { + override fun onAvailable(network: Network) { + super.onAvailable(network) + Log.i(TAG, "net ${Build.VERSION.SDK_INT}") + if (this@MainActivity.isNetworkConnected) { + Log.i(TAG, "net isNetworkConnected") + ready++ + } + } + }) + } else { + Log.i(TAG, "net ${Build.VERSION.SDK_INT}") + ready++ + } + } fun showInfoFragment(tvViewModel: TVViewModel) { @@ -120,11 +161,15 @@ class MainActivity : FragmentActivity(), Request.RequestListener { handler.postDelayed(hideMain, delayHideMain) } - fun settingActive() { + fun settingDelayHide() { handler.removeCallbacks(hideSetting) handler.postDelayed(hideSetting, delayHideSetting) } + fun settingNeverHide() { + handler.removeCallbacks(hideSetting) + } + private val hideMain = Runnable { if (!mainFragment.isHidden) { supportFragmentManager.beginTransaction().hide(mainFragment).commit() @@ -146,7 +191,7 @@ class MainActivity : FragmentActivity(), Request.RequestListener { fun fragmentReady() { ready++ Log.i(TAG, "ready $ready") - if (ready == 5) { + if (ready == 6) { mainFragment.fragmentReady() } } @@ -166,6 +211,11 @@ class MainActivity : FragmentActivity(), Request.RequestListener { return true } + override fun onDoubleTap(e: MotionEvent): Boolean { + showSetting() + return true + } + override fun onFling( e1: MotionEvent?, e2: MotionEvent, @@ -207,7 +257,7 @@ class MainActivity : FragmentActivity(), Request.RequestListener { Log.i(TAG, "settingFragment ${settingFragment.isVisible}") if (!settingFragment.isVisible) { settingFragment.show(supportFragmentManager, "setting") - settingActive() + settingDelayHide() } else { handler.removeCallbacks(hideSetting) settingFragment.dismiss() @@ -449,7 +499,7 @@ class MainActivity : FragmentActivity(), Request.RequestListener { Request.onDestroy() } - override fun onRequestFinished() { + override fun onRequestFinished(message: String?) { fragmentReady() } diff --git a/app/src/main/java/com/lizongying/mytv/MainFragment.kt b/app/src/main/java/com/lizongying/mytv/MainFragment.kt index 4f2c50e..a3c96bd 100644 --- a/app/src/main/java/com/lizongying/mytv/MainFragment.kt +++ b/app/src/main/java/com/lizongying/mytv/MainFragment.kt @@ -122,7 +122,7 @@ class MainFragment : Fragment(), CardAdapter.ItemListener { tvListViewModel.tvListViewModel.value?.forEach { tvViewModel -> tvViewModel.errInfo.observe(viewLifecycleOwner) { _ -> if (tvViewModel.errInfo.value != null - && tvViewModel.id.value == itemPosition + && tvViewModel.getTV().id == itemPosition ) { Toast.makeText(context, tvViewModel.errInfo.value, Toast.LENGTH_SHORT) .show() @@ -132,18 +132,18 @@ class MainFragment : Fragment(), CardAdapter.ItemListener { // not first time && channel not change if (tvViewModel.ready.value != null - && tvViewModel.id.value == itemPosition + && tvViewModel.getTV().id == itemPosition && check(tvViewModel) ) { - Log.i(TAG, "ready ${tvViewModel.title.value}") + Log.i(TAG, "ready ${tvViewModel.getTV().title}") (activity as? MainActivity)?.play(tvViewModel) } } tvViewModel.change.observe(viewLifecycleOwner) { _ -> if (tvViewModel.change.value != null) { - val title = tvViewModel.title.value + val title = tvViewModel.getTV().title Log.i(TAG, "switch $title") - if (tvViewModel.pid.value != "") { + if (tvViewModel.getTV().pid != "") { Log.i(TAG, "request $title") lifecycleScope.launch(Dispatchers.IO) { tvViewModel.let { Request.fetchData(it) } @@ -191,7 +191,7 @@ class MainFragment : Fragment(), CardAdapter.ItemListener { } override fun onItemHasFocus(tvViewModel: TVViewModel) { - tvListViewModel.setItemPositionCurrent(tvViewModel.id.value!!) + tvListViewModel.setItemPositionCurrent(tvViewModel.getTV().id) val row = tvViewModel.getRowPosition() @@ -211,8 +211,8 @@ class MainFragment : Fragment(), CardAdapter.ItemListener { return } - if (itemPosition != tvViewModel.id.value!!) { - itemPosition = tvViewModel.id.value!! + if (itemPosition != tvViewModel.getTV().id) { + itemPosition = tvViewModel.getTV().id tvListViewModel.setItemPosition(itemPosition) tvListViewModel.getTVViewModel(itemPosition)?.changed() } @@ -245,7 +245,7 @@ class MainFragment : Fragment(), CardAdapter.ItemListener { } fun check(tvViewModel: TVViewModel): Boolean { - val title = tvViewModel.title.value + val title = tvViewModel.getTV().title val videoUrl = tvViewModel.videoIndex.value?.let { tvViewModel.videoUrl.value?.get(it) } if (videoUrl == null || videoUrl == "") { Log.e(TAG, "$title videoUrl is empty") diff --git a/app/src/main/java/com/lizongying/mytv/NetworkChangeReceiver.kt b/app/src/main/java/com/lizongying/mytv/NetworkChangeReceiver.kt new file mode 100644 index 0000000..98ed850 --- /dev/null +++ b/app/src/main/java/com/lizongying/mytv/NetworkChangeReceiver.kt @@ -0,0 +1,16 @@ +package com.lizongying.mytv + +import android.content.BroadcastReceiver +import android.content.Context +import android.content.Intent +import android.net.ConnectivityManager + +class NetworkChangeReceiver : BroadcastReceiver() { + override fun onReceive(context: Context?, intent: Intent?) { + if (intent?.action.equals(ConnectivityManager.CONNECTIVITY_ACTION)) { + val isConnected = context?.isNetworkConnected + if (isConnected == true) { + } + } + } +} \ No newline at end of file diff --git a/app/src/main/java/com/lizongying/mytv/Request.kt b/app/src/main/java/com/lizongying/mytv/Request.kt index a27cdb5..5b8627a 100644 --- a/app/src/main/java/com/lizongying/mytv/Request.kt +++ b/app/src/main/java/com/lizongying/mytv/Request.kt @@ -57,7 +57,9 @@ object Request { private var listener: RequestListener? = null private var initRetryTimes = 0 - private var initRetryMaxTimes = 1 + private var initRetryMaxTimes = 0 + + private var openid = "vu0-8lgGV2LW9QjDeucB9H12d_6HQWmTFgqCWg5o-VBQN4" fun onCreate() { Log.i(TAG, "onCreate") @@ -86,7 +88,7 @@ object Request { private fun fetchAuth(tvModel: TVViewModel, cookie: String) { cancelCall() - val title = tvModel.title.value + val title = tvModel.getTV().title val data = YSP.getAuthData(tvModel) val request = AuthRequest(data) @@ -166,7 +168,7 @@ object Request { handler.removeCallbacks(btraceRunnable) } - val title = tvModel.title.value + val title = tvModel.getTV().title tvModel.seq = 0 val data = YSP.switch(tvModel) @@ -254,7 +256,7 @@ object Request { } } } else { - Log.e(TAG, "$title status error") + Log.e(TAG, "$title status error $data") if (tvModel.retryTimes < tvModel.retryMaxTimes) { tvModel.retryTimes++ if (tvModel.getTV().needToken) { @@ -309,13 +311,13 @@ object Request { tvModel.needGetToken = false tvModel.tokenYSPRetryTimes = 0 val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109;yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109;yspopenid=$openid; vusession=$token" fetchAuth(tvModel, cookie) } else if (response.code() == 304) { tvModel.needGetToken = false tvModel.tokenYSPRetryTimes = 0 val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=$openid; vusession=$token" fetchVideo(tvModel, cookie) } else { Log.e(TAG, "info status error") @@ -348,7 +350,7 @@ object Request { }) } else { val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109;yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109;yspopenid=$openid; vusession=$token" fetchAuth(tvModel, cookie) } } @@ -366,13 +368,13 @@ object Request { tvModel.needGetToken = false tvModel.tokenYSPRetryTimes = 0 val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=$openid; vusession=$token" fetchVideo(tvModel, cookie) } else if (response.code() == 304) { tvModel.needGetToken = false tvModel.tokenYSPRetryTimes = 0 val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=$openid; vusession=$token" fetchVideo(tvModel, cookie) } else { Log.e(TAG, "info status error") @@ -405,7 +407,7 @@ object Request { }) } else { val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=vu0-8lgGV2LW9QjDeuBFsX8yMnzs37Q3_HZF6XyVDpGR_I; vusession=$token" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205; yspappid=519748109; yspopenid=$openid; vusession=$token" fetchVideo(tvModel, cookie) } } @@ -413,7 +415,7 @@ object Request { private fun fetchFAuth(tvModel: TVViewModel) { cancelCall() - val title = tvModel.title.value + val title = tvModel.getTV().title var qa = "HD" if (tokenFH != "") { @@ -455,6 +457,9 @@ object Request { } if (tvModel.getTV().needToken) { + if (token == "") { + tvModel.needGetToken = true + } if (needAuth) { fetchAuth(tvModel) } else { @@ -462,7 +467,7 @@ object Request { } } else { val cookie = - "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; deviceModel=120; appid=1400421205" + "versionName=99.99.99; versionCode=999999; vplatform=109; platformVersion=Chrome; updateProtocol=1; deviceModel=120" if (needAuth) { fetchAuth(tvModel, cookie) } else { @@ -482,10 +487,14 @@ object Request { .enqueue(object : Callback { override fun onResponse(call: Call, response: Response) { if (response.isSuccessful) { + val o = response.body()?.o val t = response.body()?.t val f = response.body()?.f val e = response.body()?.e val c = response.body()?.c + if (!o.isNullOrEmpty()) { + openid = o + } if (!t.isNullOrEmpty()) { token = t Log.i(TAG, "token success $token") @@ -516,7 +525,7 @@ object Request { Utils.setBetween(c * 1000L) Log.i(TAG, "current time $c") } - listener?.onRequestFinished() + listener?.onRequestFinished(null) } else { Log.e(TAG, "token status error") handler.postDelayed( @@ -527,7 +536,7 @@ object Request { initRetryTimes++ fetchInfoV2() } else { - listener?.onRequestFinished() + listener?.onRequestFinished("状态错误") } } } @@ -543,7 +552,7 @@ object Request { initRetryTimes++ fetchInfoV2() } else { - listener?.onRequestFinished() + listener?.onRequestFinished("网络错误") } } }) @@ -605,11 +614,11 @@ object Request { } fun fetchBtrace(tvModel: TVViewModel) { - val title = tvModel.title.value + val title = tvModel.getTV().title val guid = YSP.getGuid() - val pid = tvModel.pid.value!! - val sid = tvModel.sid.value!! + val pid = tvModel.getTV().pid + val sid = tvModel.getTV().sid yspBtraceService.kvcollect( c_timestamp = YSP.generateGuid(), guid = guid, @@ -675,8 +684,8 @@ object Request { } fun fetchYEPG(tvViewModel: TVViewModel) { - val title = tvViewModel.title.value - yspProtoService.getProgram(tvViewModel.programId.value!!, getDateFormat("yyyyMMdd")) + val title = tvViewModel.getTV().title + yspProtoService.getProgram(tvViewModel.getTV().programId, getDateFormat("yyyyMMdd")) .enqueue(object : Callback { override fun onResponse( call: Call, @@ -700,8 +709,8 @@ object Request { } fun fetchFEPG(tvViewModel: TVViewModel) { - val title = tvViewModel.title.value - fAuthService.getEPG(tvViewModel.pid.value!!, getDateFormat("yyyyMMdd")) + val title = tvViewModel.getTV().title + fAuthService.getEPG(tvViewModel.getTV().pid, getDateFormat("yyyyMMdd")) .enqueue(object : Callback> { override fun onResponse( call: Call>, @@ -738,7 +747,7 @@ object Request { } interface RequestListener { - fun onRequestFinished() + fun onRequestFinished(message: String?) } fun setRequestListener(listener: RequestListener) { diff --git a/app/src/main/java/com/lizongying/mytv/SettingFragment.kt b/app/src/main/java/com/lizongying/mytv/SettingFragment.kt index ef2e271..bcb36a1 100644 --- a/app/src/main/java/com/lizongying/mytv/SettingFragment.kt +++ b/app/src/main/java/com/lizongying/mytv/SettingFragment.kt @@ -5,12 +5,12 @@ import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import androidx.fragment.app.DialogFragment -import com.lizongying.mytv.databinding.DialogBinding +import com.lizongying.mytv.databinding.SettingBinding class SettingFragment : DialogFragment() { - private var _binding: DialogBinding? = null + private var _binding: SettingBinding? = null private val binding get() = _binding!! private lateinit var updateManager: UpdateManager @@ -26,15 +26,15 @@ class SettingFragment : DialogFragment() { savedInstanceState: Bundle? ): View { val context = requireContext() // It‘s safe to get context here. - _binding = DialogBinding.inflate(inflater, container, false) - binding.version.text = - "当前版本: ${context.appVersionName}\n获取最新: https://github.com/lizongying/my-tv/releases/" + _binding = SettingBinding.inflate(inflater, container, false) + binding.versionName.text = "当前版本: v${context.appVersionName}" + binding.version.text = "https://github.com/lizongying/my-tv" binding.switchChannelReversal.run { isChecked = SP.channelReversal setOnCheckedChangeListener { _, isChecked -> SP.channelReversal = isChecked - (activity as MainActivity).settingActive() + (activity as MainActivity).settingDelayHide() } } @@ -42,7 +42,7 @@ class SettingFragment : DialogFragment() { isChecked = SP.channelNum setOnCheckedChangeListener { _, isChecked -> SP.channelNum = isChecked - (activity as MainActivity).settingActive() + (activity as MainActivity).settingDelayHide() } } @@ -50,7 +50,7 @@ class SettingFragment : DialogFragment() { isChecked = SP.bootStartup setOnCheckedChangeListener { _, isChecked -> SP.bootStartup = isChecked - (activity as MainActivity).settingActive() + (activity as MainActivity).settingDelayHide() } } @@ -58,23 +58,34 @@ class SettingFragment : DialogFragment() { isChecked = SP.grid setOnCheckedChangeListener { _, isChecked -> SP.grid = isChecked - (activity as MainActivity).settingActive() + (activity as MainActivity).settingDelayHide() } } updateManager = UpdateManager(context, this, context.appVersionCode) - binding.checkVersion.setOnClickListener(OnClickListenerCheckVersion(updateManager)) + binding.checkVersion.setOnClickListener( + OnClickListenerCheckVersion( + activity as MainActivity, + updateManager + ) + ) return binding.root } fun setVersionName(versionName: String) { - binding.versionName.text = versionName + if (_binding != null) { + binding.versionName.text = versionName + } } - internal class OnClickListenerCheckVersion(private val updateManager: UpdateManager) : + internal class OnClickListenerCheckVersion( + private val mainActivity: MainActivity, + private val updateManager: UpdateManager + ) : View.OnClickListener { override fun onClick(view: View?) { + mainActivity.settingDelayHide() updateManager.checkAndUpdate() } } @@ -84,11 +95,6 @@ class SettingFragment : DialogFragment() { _binding = null } - override fun onDestroy() { - super.onDestroy() -// updateManager.destroy() - } - companion object { const val TAG = "SettingFragment" } diff --git a/app/src/main/java/com/lizongying/mytv/TV.kt b/app/src/main/java/com/lizongying/mytv/TV.kt index ee8b1ff..019d370 100644 --- a/app/src/main/java/com/lizongying/mytv/TV.kt +++ b/app/src/main/java/com/lizongying/mytv/TV.kt @@ -7,7 +7,6 @@ data class TV( var title: String, var alias: String = "", var videoUrl: List, - var videoIndex: Int = 0, var channel: String = "", var logo: Any = "", var pid: String = "", @@ -23,7 +22,6 @@ data class TV( "id=" + id + ", title='" + title + '\'' + ", videoUrl='" + videoUrl + '\'' + - ", videoIndex='" + videoIndex + '\'' + ", logo='" + logo + '\'' + ", pid='" + pid + '\'' + ", sid='" + sid + '\'' + diff --git a/app/src/main/java/com/lizongying/mytv/TVList.kt b/app/src/main/java/com/lizongying/mytv/TVList.kt index 7c3cbea..bb64375 100644 --- a/app/src/main/java/com/lizongying/mytv/TVList.kt +++ b/app/src/main/java/com/lizongying/mytv/TVList.kt @@ -13,7 +13,6 @@ object TVList { "CCTV1 综合", "CCTV1", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226231/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/d57905b93540bd15f0c48230dbbbff7ee0d645ff539e38866e2d15c8b9f7dfcd.png?imageMogr2/format/webp", "600001859", @@ -27,7 +26,6 @@ object TVList { "CCTV2 财经", "CCTV2", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226195/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/20115388de0207131af17eac86c33049b95d69eaff064e55653a1b941810a006.png?imageMogr2/format/webp", "600001800", @@ -41,7 +39,6 @@ object TVList { "CCTV3 综艺", "CCTV3", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226397/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/7b7a65c712450da3deb6ca66fbacf4f9aee00d3f20bd80eafb5ada01ec63eb3a.png?imageMogr2/format/webp", "600001801", @@ -58,7 +55,6 @@ object TVList { "http://39.134.24.161/dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226191/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226191/index.m3u8" ), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/f357e58fdbcc076a3d65e1f958c942b2e14f14342c60736ceed98b092d35356a.png?imageMogr2/format/webp", "600001814", @@ -72,7 +68,6 @@ object TVList { "CCTV5 体育", "CCTV5", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226395/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/0a6a7138952675983a3d854df7688557b286d59aa06166edae51506f9204d655.png?imageMogr2/format/webp", "600001818", @@ -86,7 +81,6 @@ object TVList { "CCTV6 电影", "CCTV6", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226393/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/741515efda91f03f455df8a7da4ee11fa9329139c276435cf0a9e2af398d5bf2.png?imageMogr2/format/webp", "600108442", @@ -100,7 +94,6 @@ object TVList { "CCTV7 国防军事", "CCTV7", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226192/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/b29af94e295ebdf646cefb68122c429b9cd921f498ca20d2d8070252536f9ff9.png?imageMogr2/format/webp", "600004092", @@ -114,7 +107,6 @@ object TVList { "CCTV8 电视剧", "CCTV8", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226391/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/ad51de94426a0ba039e6dd6a8534ea98ecc813a6176bde87b4f18cc34d6d7590.png?imageMogr2/format/webp", "600001803", @@ -128,7 +120,6 @@ object TVList { "CCTV9 纪录", "CCTV9", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226197/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/2ed1b4deeca179d5db806bb941790f82eb92a1b7299c1c38fe027f95a5caee5e.png?imageMogr2/format/webp", "600004078", @@ -142,7 +133,6 @@ object TVList { "CCTV10 科教", "CCTV10", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226189/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/aa6157ec65188cd41826e5a2f088c3d6d153205f5f6428258d12c59999e221aa.png?imageMogr2/format/webp", "600001805", @@ -156,7 +146,6 @@ object TVList { "CCTV11 戏曲", "CCTV11", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226240/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/ed12ed7c7a1034dae4350011fe039284c5d5a836506b28c9e32e3c75299625c0.png?imageMogr2/format/webp", "600001806", @@ -170,7 +159,6 @@ object TVList { "CCTV12 社会与法", "CCTV12", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226190/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/484083cffaa40df7e659565e8cb4d1cc740158a185512114167aa21fa0c59240.png?imageMogr2/format/webp", "600001807", @@ -184,10 +172,8 @@ object TVList { "CCTV13 新闻", "CCTV13", listOf( - "https://live-play.cctvnews.cctv.com/cctv/merge_cctv13.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226233/index.m3u8" ), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/266da7b43c03e2312186b4a999e0f060e8f15b10d2cc2c9aa32171819254cf1a.png?imageMogr2/format/webp", "600001811", @@ -201,7 +187,6 @@ object TVList { "CCTV14 少儿", "CCTV14", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226193/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/af6b603896938dc346fbb16abfc63c12cba54b0ec9d18770a15d347d115f12d5.png?imageMogr2/format/webp", "600001809", @@ -215,7 +200,6 @@ object TVList { "CCTV15 音乐", "CCTV15", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225785/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/2ceee92188ef684efe0d8b90839c4f3ad450d179dc64d59beff417059453af47.png?imageMogr2/format/webp", "600001815", @@ -232,7 +216,6 @@ object TVList { "http://39.134.24.162/dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226921/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226921/index.m3u8" ), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/53793fa7bacd3a93ff6dc5d2758418985e1f952a316c335d663b572d8bdcd74d.png?imageMogr2/format/webp", "600098637", @@ -246,7 +229,6 @@ object TVList { "CCTV17 农业农村", "CCTV17", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226198/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/ddef563072f8bad2bea5b9e52674cb7b4ed50efb20c26e61994dfbdf05c1e3c0.png?imageMogr2/format/webp", "600001810", @@ -260,7 +242,6 @@ object TVList { "CCTV5+ 体育赛事", "CCTV5+", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226221/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/649ad76a90bfef55b05db9fe52e006487280f619089099d5dc971e387fc6eff0.png?imageMogr2/format/webp", "600001817", @@ -274,7 +255,6 @@ object TVList { "CCTV4K 超高清", "CCTV4K", listOf(), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/3e9d06fd7244d950df5838750f1c6ac3456e172b51caca2c16d2282125b111e8.png?imageMogr2/format/webp", "600002264", @@ -288,7 +268,6 @@ object TVList { "CCTV8K 超高清", "CCTV8K", listOf(), - 0, "央视", R.drawable.cctv8k, "600156816", @@ -302,7 +281,6 @@ object TVList { "风云剧场", "CCTV风云剧场频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226950/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/4d549e53e6d0f632d5a633d1945280797b153e588f919221a07faa869812cc89.png?imageMogr2/format/webp", "600099658", @@ -316,7 +294,6 @@ object TVList { "第一剧场", "CCTV第一剧场频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226959/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/a556bd7d93ce65e18f243a8892b5604f4faa994a4897315914216a710a706208.png?imageMogr2/format/webp", "600099655", @@ -330,7 +307,6 @@ object TVList { "怀旧剧场", "CCTV怀旧剧场频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226972/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/5661bd04fecdb6e899f801147a22ab5d3a475bf2b62e30aec2c0023190ebc9b1.png?imageMogr2/format/webp", "600099620", @@ -344,7 +320,6 @@ object TVList { "世界地理", "CCTV世界地理频道", listOf(), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/bb3c6c9e145d698137f5bb64a582021a01b51344b929003630eb769ea65832a9.png?imageMogr2/format/webp", "600099637", @@ -358,7 +333,6 @@ object TVList { "风云音乐", "CCTV风云音乐频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226953/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/bbf1d024c5228b8dd128b0e3cb1717d173fab4ee84c3a4c8a57b1a215362ca3b.png?imageMogr2/format/webp", "600099660", @@ -372,7 +346,6 @@ object TVList { "兵器科技", "CCTV兵器科技频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226975/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/4c6b6a6d3839889f34d33db3c2f80233b26b74d3489b393487635f8704e70796.png?imageMogr2/format/webp", "600099649", @@ -386,7 +359,6 @@ object TVList { "风云足球", "CCTV风云足球频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226984/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/cd1e2bb52b06a991de168733e5ff0f1d85adc8042d40c8f393f723543e5dd08a.png?imageMogr2/format/webp", "600099636", @@ -400,7 +372,6 @@ object TVList { "高尔夫网球", "CCTV高尔夫·网球频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226978/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/cdd1b31ede7a5ad049ed53d9a072422f829e72dd062ed2c19e077fdd01699071.png?imageMogr2/format/webp", "600099659", @@ -414,7 +385,6 @@ object TVList { "女性时尚", "CCTV女性时尚频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226969/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/fa28955ce8b2539d728bf4c6a13a46ff57ad76eae46627f7bcfb1ed8a613d3fc.png?imageMogr2/format/webp", "600099650", @@ -428,7 +398,6 @@ object TVList { "央视文化精品", "CCTV央视文化精品频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226981/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/14ac5ce40482cacd3d4b37435222bfe86af2b452a2f04ecbfc1d13d76edd7c57.png?imageMogr2/format/webp", "600099653", @@ -442,7 +411,6 @@ object TVList { "央视台球", "CCTV央视台球频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226956/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/10e14a92478011aa6c3c8562e62127f3b1908e29fcd78e4b2b24b9e6d3ec2fbc.png?imageMogr2/format/webp", "600099652", @@ -456,7 +424,6 @@ object TVList { "电视指南", "CCTV电视指南频道", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888893/224/3221226987/index.m3u8"), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/244d72c0eb1615ed7d51c2f5db5a67f306aa3f58c05bc2d34de3aa7e956dc8c9.png?imageMogr2/format/webp", "600099656", @@ -470,7 +437,6 @@ object TVList { "卫生健康", "CCTV卫生健康频道", listOf(), - 0, "央视", "https://resources.yangshipin.cn/assets/oms/image/202306/54a6863656fdfd8f803be193ddf22441c5000a108833889816fd2d8911715ce8.png?imageMogr2/format/webp", "600099651", @@ -486,7 +452,6 @@ object TVList { "东方卫视", "东方卫视", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226217/index.m3u8"), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/9bd372ca292a82ce3aa08772b07efc4af1f85c21d1f268ea33440c49e9a0a488.png?imageMogr2/format/webp", "600002483", @@ -503,7 +468,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226307/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226211/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/4120e89d3079d08aa17d382f69a2308ec70839b278367763c34a34666c75cb88.png?imageMogr2/format/webp", "600002475", @@ -520,7 +484,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226477/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226194/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/7a6be5a2bb1dc53a945c016ff1f525dc4a84c51db371c15c89aa55404b0ba784.png?imageMogr2/format/webp", "600002508", @@ -537,7 +500,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226546/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226201/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/ac4ed6058a87c101ae7147ebc38905d0cae047fb73fd277ee5049b84f52bda36.png?imageMogr2/format/webp", "600002505", @@ -554,7 +516,6 @@ object TVList { "http://39.134.24.166/dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226200/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226200/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/380ad685c0c1d5b2c902246b8d2df6d3f9b45e2837abcfe493075bbded597a31.png?imageMogr2/format/webp", "600002521", @@ -571,7 +532,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226344/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225764/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/3c760d0d00463855890e8a1864ea4a6b6dd66b90c29b4ac714a4b17c16519871.png?imageMogr2/format/webp", "600002503", @@ -585,7 +545,6 @@ object TVList { "山东卫视", "山东卫视", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226209/index.m3u8"), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/22d403f07a7cf5410b3ad3ddb65a11aa229a32475fac213f5344c9f0ec330ca1.png?imageMogr2/format/webp", "600002513", @@ -599,7 +558,6 @@ object TVList { "广东卫视", "广东卫视", listOf("http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226216/index.m3u8"), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/28886880a4dc0f06fb7e0a528a1def0591d61a65870e29176ede0cc92033bbfd.png?imageMogr2/format/webp", "600002485", @@ -616,7 +574,6 @@ object TVList { "http://live.gxrb.com.cn/tv/gxtvlive03/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225770/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/54b7e97cb816bb223fe05f3fc44da2c7820eb66e8550c19d23100f2c414ecc38.png?imageMogr2/format/webp", "600002509", @@ -633,7 +590,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226409/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226202/index.m3u8" ), - 0, "地方", R.drawable.chongqing, "600002531", @@ -650,7 +606,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226480/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225767/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/74925962148a6d31c85808b6cd4e444c2a54bab393d2c5fc85e960b50e22fa86.png?imageMogr2/format/webp", "600002525", @@ -667,7 +622,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226406/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225750/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/d545becdc81c60197b08c7f47380705e4665ed3fe55efc8b855e486f6e655378.png?imageMogr2/format/webp", "600002493", @@ -684,7 +638,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226474/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225793/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/4eb45f4781d33d872af027dc01c941559aab55667dd99cc5c22bef7037807b13.png?imageMogr2/format/webp", "600002490", @@ -701,7 +654,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221225728/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226222/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/f4f23633c578beea49a3841d88d3490100f029ee349059fa532869db889872c5.png?imageMogr2/format/webp", "600002309", @@ -718,7 +670,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226327/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226215/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/d8273ae9be698ce2db21f5b886ecac95a73429593f93713c60ed8c12c38bf0d3.png?imageMogr2/format/webp", "600002498", @@ -735,7 +686,6 @@ object TVList { "http://hw-m-l.cztv.com/channels/lantian/channel01/1080p.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226199/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/a66c836bd98ba3e41a2e9a570d4b9c50dedc6839e9de333e2e78212ad505f37e.png?imageMogr2/format/webp", "600002520", @@ -752,7 +702,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226391/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226203/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/f35fa04b51b1ee4984b03578b65403570868ebca03c6c01e11b097f999a58d9b.png?imageMogr2/format/webp", "600002532", @@ -769,7 +718,6 @@ object TVList { "http://39.134.24.166/dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226205/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226205/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/d59fec04c902e3581c617136d02d4b9b8c4cbe64272781ddd3525e80c823edb7.png?imageMogr2/format/webp", "600002481", @@ -786,7 +734,6 @@ object TVList { "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225768/index.m3u8", "http://39.134.24.166/dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225768/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/3276a414ae0eaa0f116f2045cd913367967d0c7c1e978e8621ac3879436c6ed7.png?imageMogr2/format/webp", "600002516", @@ -803,7 +750,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226341/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225766/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/3208fe6564a293c21b711333fb3edb05bb5b406cff840573c9a8d839680a1579.png?imageMogr2/format/webp", "600002484", @@ -820,7 +766,6 @@ object TVList { "http://ottrrs.hl.chinamobile.com/PLTV/88888888/224/3221226465/index.m3u8", "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221225769/index.m3u8" ), - 0, "地方", "https://resources.yangshipin.cn/assets/oms/image/202306/6e060391fde0469801fc3d84dbf204b4f8d650d251f17d7595a6964c0bb99e81.png?imageMogr2/format/webp", "600002506", @@ -834,7 +779,6 @@ object TVList { "天津卫视", "天津卫视", listOf(), - 0, "地方", R.drawable.tianjin, "600152137", @@ -848,7 +792,6 @@ object TVList { "新疆卫视", "新疆卫视", listOf(), - 0, "地方", R.drawable.xinjiang, "600152138", @@ -862,7 +805,6 @@ object TVList { "兵团卫视", "兵团卫视", listOf(), - 0, "地方", R.drawable.bingtuan, "600170344", @@ -876,7 +818,6 @@ object TVList { "CETV1", "CETV1", listOf(), - 0, "地方", R.drawable.cetv1, "600171827", @@ -892,7 +833,6 @@ object TVList { "凤凰卫视资讯台", "", listOf(), - 0, "港澳台", "http://c1.fengshows-cdn.com/a/2021_22/79dcc3a9da358a3.png", "7c96b084-60e1-40a9-89c5-682b994fb680", @@ -906,7 +846,6 @@ object TVList { "凤凰卫视中文台", "", listOf(), - 0, "港澳台", "http://c1.fengshows-cdn.com/a/2021_22/ede3d9e09be28e5.png", "f7f48462-9b13-485b-8101-7b54716411ec", @@ -920,7 +859,6 @@ object TVList { "凤凰卫视香港台", "", listOf(), - 0, "港澳台", "http://c1.fengshows-cdn.com/a/2021_23/325d941090bee17.png", "15e02d92-1698-416c-af2f-3e9a872b4d78", @@ -936,7 +874,6 @@ object TVList { "CGTN", "CGTN", listOf("http://live.cgtn.com/1000/prog_index.m3u8"), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202306/a72dff758ca1c17cd0ecc8cedc11b893d208f409d5e6302faa0e9d298848abc3.png?imageMogr2/format/webp", "600014550", @@ -950,7 +887,6 @@ object TVList { "CGTN 法语频道", "CGTN法语频道", listOf("https://livefr.cgtn.com/1000f/prog_index.m3u8"), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202306/a8d0046a47433d952bf6ed17062deb8bd2184ba9aec0f7781df6bf9487a3ffcf.png?imageMogr2/format/webp", "600084704", @@ -964,7 +900,6 @@ object TVList { "CGTN 俄语频道", "CGTN俄语频道", listOf("http://liveru.cgtn.com/1000r/prog_index.m3u8"), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202306/bf0a820893cbaf20dd0333e27042e1ef9c8806e5b602b6a8c95af399db0bc77a.png?imageMogr2/format/webp", "600084758", @@ -978,7 +913,6 @@ object TVList { "CGTN 阿拉伯语频道", "CGTN阿拉伯语频道", listOf("http://livear.cgtn.com/1000a/prog_index.m3u8"), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202306/2e44e2aa3e7a1cedf07fd0ae59fe69e86a60a2632660a006e3e9e7397b2d107e.png?imageMogr2/format/webp", "600084782", @@ -995,7 +929,6 @@ object TVList { "http://livees.cgtn.com/500e/prog_index.m3u8", "http://livees.cgtn.com/1000e/prog_index.m3u8" ), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202309/7c337e3dbe64402ec7e4678a619a4a6d95144e42f35161181ff78e143b7cf67a.png?imageMogr2/format/webp", "600084744", @@ -1009,7 +942,6 @@ object TVList { "CGTN 纪录频道", "CGTN外语纪录频道", listOf("https://livedoc.cgtn.com/500d/prog_index.m3u8"), - 0, "国际", "https://resources.yangshipin.cn/assets/oms/image/202309/74d3ac436a7e374879578de1d87a941fbf566d39d5632b027c5097891ed32bd5.png?imageMogr2/format/webp", "600084781", diff --git a/app/src/main/java/com/lizongying/mytv/UpdateManager.kt b/app/src/main/java/com/lizongying/mytv/UpdateManager.kt index 5616bdd..d283338 100644 --- a/app/src/main/java/com/lizongying/mytv/UpdateManager.kt +++ b/app/src/main/java/com/lizongying/mytv/UpdateManager.kt @@ -1,5 +1,6 @@ package com.lizongying.mytv +import android.app.Activity import android.app.DownloadManager import android.app.DownloadManager.Request import android.content.BroadcastReceiver @@ -13,8 +14,10 @@ import android.os.Environment import android.os.Handler import android.os.Looper import android.util.Log -import android.widget.Toast -import com.lizongying.mytv.api.Release +import androidx.core.app.ActivityCompat +import androidx.core.content.PermissionChecker +import androidx.core.content.PermissionChecker.checkSelfPermission +import com.lizongying.mytv.api.ReleaseV2 import com.lizongying.mytv.requests.MyRequest import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers @@ -27,52 +30,81 @@ class UpdateManager( private var settingFragment: SettingFragment, private var versionCode: Long ) : - ConfirmationDialogFragment.ConfirmationDialogListener { + ConfirmationFragment.ConfirmationListener { private var myRequest = MyRequest() - private var release: Release? = null + private var release: ReleaseV2? = null private var downloadReceiver: DownloadReceiver? = null fun checkAndUpdate() { + if (!haveStoragePermission()) { + return + } CoroutineScope(Dispatchers.Main).launch { + var text = "版本获取失败" try { release = myRequest.getRelease() - updateUI(release) - Log.i(TAG, "versionCode $versionCode ${release?.data?.versionCode}") - if (release != null) { - if (release?.data?.versionCode!! >= versionCode) { - val dialog = ConfirmationDialogFragment(this@UpdateManager) - dialog.show(settingFragment.fragmentManager, "ConfirmationDialogFragment") + Log.i(TAG, "versionCode $versionCode ${release?.c}") + if (release?.c != null) { + text = if (release?.c!! > versionCode) { + "最新版本:${release?.n}\n${release?.d ?: ""}" } else { - Toast.makeText(context, "不需要更新", Toast.LENGTH_LONG) - .show() + "已是最新版本,不需要更新" } } } catch (e: Exception) { Log.e(TAG, "Error occurred: ${e.message}", e) } + updateUI(text) } } - private fun updateUI(release: Release?) { - if (release?.data?.versionName.isNullOrEmpty()) { - settingFragment.setVersionName("版本获取失败") - } else { - settingFragment.setVersionName("最新版本:${release?.data?.versionName!!}") + private fun updateUI(text: String) { + val dialog = ConfirmationFragment(this@UpdateManager, text) + dialog.show(settingFragment.fragmentManager, TAG) + } + + private fun haveStoragePermission(): Boolean { + if (Build.VERSION.SDK_INT >= 23) { + if (checkSelfPermission(context, android.Manifest.permission.WRITE_EXTERNAL_STORAGE) + === PermissionChecker.PERMISSION_GRANTED + ) { + Log.e("Permission error", "You have permission") + return true + } else { + Log.e("Permission error", "You have asked for permission") + ActivityCompat.requestPermissions( + context as Activity, arrayOf( + android.Manifest.permission.WRITE_EXTERNAL_STORAGE + ), 1 + ) + return false + } + } else { //you don't need to worry about these stuff below api level 23 + Log.e("Permission error", "You already have the permission") + return true } } - private fun startDownload(release: Release) { - val apkFileName = "my-tv-${release.data.versionName}.apk" + + private fun startDownload(release: ReleaseV2) { + val apkFileName = "my-tv-${release.n}.apk" Log.i(TAG, "apkFileName $apkFileName") val downloadManager = context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager - val request = Request(Uri.parse(release.data.downloadUrl)) - Log.i(TAG, "url ${Uri.parse(release.data.downloadUrl)}") - request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, apkFileName) - request.setTitle("New Version Download") + val request = Request(Uri.parse(release.u)) + Log.i(TAG, "url ${Uri.parse(release.u)}") + context.getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS)?.mkdirs() + request.setDestinationInExternalFilesDir( + context, + Environment.DIRECTORY_DOWNLOADS, + apkFileName + ) + request.setTitle("${settingFragment.resources.getString(R.string.app_name)} ${release.n}") request.setNotificationVisibility(Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED) + request.setAllowedOverRoaming(false) + request.setMimeType("application/vnd.android.package-archive") // 获取下载任务的引用 val downloadReference = downloadManager.enqueue(request) @@ -97,7 +129,11 @@ class UpdateManager( } } - private fun getDownloadProgress(context: Context, downloadId: Long, progressListener: (Int) -> Unit) { + private fun getDownloadProgress( + context: Context, + downloadId: Long, + progressListener: (Int) -> Unit + ) { val downloadManager = context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager val handler = Handler(Looper.getMainLooper()) val intervalMillis: Long = 1000 @@ -179,7 +215,6 @@ class UpdateManager( override fun onCancel() { } - fun destroy() { if (downloadReceiver != null) { context.unregisterReceiver(downloadReceiver) diff --git a/app/src/main/java/com/lizongying/mytv/api/ApiClient.kt b/app/src/main/java/com/lizongying/mytv/api/ApiClient.kt index 6833d58..a8fbb60 100644 --- a/app/src/main/java/com/lizongying/mytv/api/ApiClient.kt +++ b/app/src/main/java/com/lizongying/mytv/api/ApiClient.kt @@ -17,7 +17,6 @@ import javax.net.ssl.X509TrustManager class ApiClient { private val yspUrl = "https://player-api.yangshipin.cn/" private val myUrl = "https://lyrics.run/" - private val devUrl = "http://10.0.2.2:8081/" private val protoUrl = "https://capi.yangshipin.cn/" private val traceUrl = "https://btrace.yangshipin.cn/" private val fUrl = "https://m.fengshows.com/" diff --git a/app/src/main/java/com/lizongying/mytv/api/Info.kt b/app/src/main/java/com/lizongying/mytv/api/Info.kt index 0bec143..9fd7c4c 100644 --- a/app/src/main/java/com/lizongying/mytv/api/Info.kt +++ b/app/src/main/java/com/lizongying/mytv/api/Info.kt @@ -16,24 +16,19 @@ data class Token( ) data class InfoV2( + val o: String?, val f: String?, val t: String?, val e: Int?, val c: Int?, ) -data class Release( - val code: Int?, - val msg: String?, - val data: Data, -) { - data class Data( - val versionName: String, - val versionCode: Int, - val downloadUrl: String, - val updateTime: Int, - ) -} +data class ReleaseV2( + val n: String?, + val u: String?, + val d: String?, + val c: Int?, +) data class TimeResponse( val data: Time diff --git a/app/src/main/java/com/lizongying/mytv/api/ReleaseService.kt b/app/src/main/java/com/lizongying/mytv/api/ReleaseService.kt index d06eea1..8870d48 100644 --- a/app/src/main/java/com/lizongying/mytv/api/ReleaseService.kt +++ b/app/src/main/java/com/lizongying/mytv/api/ReleaseService.kt @@ -2,10 +2,11 @@ package com.lizongying.mytv.api import retrofit2.Call import retrofit2.http.GET - +import retrofit2.http.Path interface ReleaseService { - @GET("my-tv/v1/release") + @GET("my-tv/v2/release/{name}") fun getRelease( - ): Call + @Path("name") date: String = "2", + ): Call } \ No newline at end of file diff --git a/app/src/main/java/com/lizongying/mytv/api/YSP.kt b/app/src/main/java/com/lizongying/mytv/api/YSP.kt index 5a20ae6..20ba9c1 100644 --- a/app/src/main/java/com/lizongying/mytv/api/YSP.kt +++ b/app/src/main/java/com/lizongying/mytv/api/YSP.kt @@ -63,8 +63,8 @@ object YSP { } fun switch(tvModel: TVViewModel): String { - livepid = tvModel.pid.value!! - cnlid = tvModel.sid.value!! + livepid = tvModel.getTV().pid + cnlid = tvModel.getTV().sid defn = "fhd" randStr = getRand() @@ -82,7 +82,7 @@ object YSP { } fun getAuthData(tvModel: TVViewModel): String { - livepid = tvModel.pid.value!! + livepid = tvModel.getTV().pid randStr = getRand() diff --git a/app/src/main/java/com/lizongying/mytv/models/TVViewModel.kt b/app/src/main/java/com/lizongying/mytv/models/TVViewModel.kt index 5fff447..c2efea6 100644 --- a/app/src/main/java/com/lizongying/mytv/models/TVViewModel.kt +++ b/app/src/main/java/com/lizongying/mytv/models/TVViewModel.kt @@ -1,15 +1,9 @@ package com.lizongying.mytv.models -import android.net.Uri import android.util.Log -import androidx.annotation.OptIn import androidx.lifecycle.LiveData import androidx.lifecycle.MutableLiveData import androidx.lifecycle.ViewModel -import androidx.media3.common.MediaItem -import androidx.media3.common.util.UnstableApi -import androidx.media3.datasource.DefaultHttpDataSource -import androidx.media3.exoplayer.hls.HlsMediaSource import com.lizongying.mytv.TV import com.lizongying.mytv.api.FEPG import com.lizongying.mytv.proto.Ysp.cn.yangshipin.omstv.common.proto.programModel.Program @@ -26,7 +20,7 @@ class TVViewModel(private var tv: TV) : ViewModel() { var tokenYSPRetryTimes = 0 var tokenYSPRetryMaxTimes = 0 var tokenFHRetryTimes = 0 - var tokenFHRetryMaxTimes = 2 + var tokenFHRetryMaxTimes = 8 var needGetToken = false @@ -34,22 +28,10 @@ class TVViewModel(private var tv: TV) : ViewModel() { val errInfo: LiveData get() = _errInfo - private val _programId = MutableLiveData() - val programId: LiveData - get() = _programId - private var _epg = MutableLiveData>() val epg: LiveData> get() = _epg - private val _id = MutableLiveData() - val id: LiveData - get() = _id - - private val _title = MutableLiveData() - val title: LiveData - get() = _title - private val _videoUrl = MutableLiveData>() val videoUrl: LiveData> get() = _videoUrl @@ -58,18 +40,6 @@ class TVViewModel(private var tv: TV) : ViewModel() { val videoIndex: LiveData get() = _videoIndex - private val _logo = MutableLiveData() - val logo: LiveData - get() = _logo - - private val _pid = MutableLiveData() - val pid: LiveData - get() = _pid - - private val _sid = MutableLiveData() - val sid: LiveData - get() = _sid - private val _change = MutableLiveData() val change: LiveData get() = _change @@ -90,9 +60,8 @@ class TVViewModel(private var tv: TV) : ViewModel() { } else { tv.videoUrl = tv.videoUrl + listOf(url) } - tv.videoIndex = tv.videoUrl.lastIndex _videoUrl.value = tv.videoUrl - _videoIndex.value = tv.videoIndex + _videoIndex.value = tv.videoUrl.lastIndex } fun firstSource() { @@ -117,14 +86,8 @@ class TVViewModel(private var tv: TV) : ViewModel() { } init { - _id.value = tv.id - _title.value = tv.title _videoUrl.value = tv.videoUrl - _videoIndex.value = tv.videoIndex - _logo.value = tv.logo - _programId.value = tv.programId - _pid.value = tv.pid - _sid.value = tv.sid + _videoIndex.value = tv.videoUrl.lastIndex } fun getRowPosition(): Int { @@ -147,10 +110,6 @@ class TVViewModel(private var tv: TV) : ViewModel() { _errInfo.value = info } - fun update(t: TV) { - tv = t - } - fun getTV(): TV { return tv } @@ -173,21 +132,6 @@ class TVViewModel(private var tv: TV) : ViewModel() { _epg.value = p.map { EPG(it.title, formatFTime(it.event_time)) }.toMutableList() } - /** - * (playerView?.player as ExoPlayer).setMediaSource(tvViewModel.buildSource()) - */ - @OptIn(UnstableApi::class) - fun buildSource(): HlsMediaSource { - val httpDataSource = DefaultHttpDataSource.Factory() -// mHeaders?.let { httpDataSource.setDefaultRequestProperties(it) } - - return HlsMediaSource.Factory(httpDataSource).createMediaSource( - MediaItem.fromUri( - Uri.parse(getVideoUrlCurrent()) - ) - ) - } - fun getVideoUrlCurrent(): String { return _videoUrl.value!![_videoIndex.value!!] } diff --git a/app/src/main/java/com/lizongying/mytv/requests/MyRequest.kt b/app/src/main/java/com/lizongying/mytv/requests/MyRequest.kt index 412c504..17022eb 100644 --- a/app/src/main/java/com/lizongying/mytv/requests/MyRequest.kt +++ b/app/src/main/java/com/lizongying/mytv/requests/MyRequest.kt @@ -1,7 +1,7 @@ package com.lizongying.mytv.requests import com.lizongying.mytv.api.ApiClient -import com.lizongying.mytv.api.Release +import com.lizongying.mytv.api.ReleaseV2 import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.withContext import retrofit2.Call @@ -13,17 +13,17 @@ import kotlin.coroutines.suspendCoroutine class MyRequest { private var releaseService = ApiClient().releaseService - suspend fun getRelease(): Release? { + suspend fun getRelease(): ReleaseV2? { return withContext(Dispatchers.IO) { fetchRelease() } } - private suspend fun fetchRelease(): Release? { + private suspend fun fetchRelease(): ReleaseV2? { return suspendCoroutine { continuation -> releaseService.getRelease() - .enqueue(object : Callback { - override fun onResponse(call: Call, response: Response) { + .enqueue(object : Callback { + override fun onResponse(call: Call, response: Response) { if (response.isSuccessful) { continuation.resume(response.body()) } else { @@ -31,7 +31,7 @@ class MyRequest { } } - override fun onFailure(call: Call, t: Throwable) { + override fun onFailure(call: Call, t: Throwable) { continuation.resume(null) } }) diff --git a/app/src/main/res/layout/dialog.xml b/app/src/main/res/layout/setting.xml similarity index 100% rename from app/src/main/res/layout/dialog.xml rename to app/src/main/res/layout/setting.xml diff --git a/build.gradle b/build.gradle index 479de02..40928c7 100644 --- a/build.gradle +++ b/build.gradle @@ -1,6 +1,6 @@ // Top-level build file where you can add configuration options common to all sub-projects/modules. plugins { - id 'com.android.application' version '8.2.2' apply false - id 'com.android.library' version '8.2.2' apply false + id 'com.android.application' version '8.3.0' apply false + id 'com.android.library' version '8.3.0' apply false id 'org.jetbrains.kotlin.android' version '1.9.22' apply false } \ No newline at end of file diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index 99e680a..65a247e 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,6 @@ #Fri Dec 01 13:53:24 HKT 2023 distributionBase=GRADLE_USER_HOME -distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-rc-1-bin.zip distributionPath=wrapper/dists zipStorePath=wrapper/dists zipStoreBase=GRADLE_USER_HOME