Compare commits
202 Commits
feature-ca
...
fix-videoC
| Author | SHA1 | Date | |
|---|---|---|---|
| c1008e0162 | |||
| d1e8068e51 | |||
| 6de9b1977c | |||
| 3c0f54bfd7 | |||
| 8950658f08 | |||
| 7a78729a44 | |||
| 03e5e22fef | |||
| aa93ce0b89 | |||
| 0c365ad049 | |||
| 3d5c578fef | |||
| 0a22f0f543 | |||
| 5bf7b69d79 | |||
| d57f84a1d7 | |||
| 32b2f0ceff | |||
| bae871cfa1 | |||
| d95fe9fe14 | |||
| eb006e4c55 | |||
| cb88d0c9ae | |||
| 3efad736ae | |||
| 42ad959155 | |||
| cdf800c49f | |||
| 569277572a | |||
| 19b84571c1 | |||
| 0812b8339e | |||
| b817a0c807 | |||
| 3da70d7e27 | |||
| 5e59db85be | |||
| 77477ff4dd | |||
| 89026e671c | |||
| 1c8e7e53a5 | |||
| b264427be6 | |||
| d5134f972d | |||
| e2fd01a6d5 | |||
| 289cc99bc2 | |||
| 3d5ebe7e99 | |||
| d9964d37a4 | |||
| 5da39a9c52 | |||
| 44a162762c | |||
| d0f036ec35 | |||
| 10b928474b | |||
| 94f3b7c1e4 | |||
| fb8b2de115 | |||
| 0d5d33a365 | |||
| c39e91073b | |||
| d258474a5a | |||
| b0c56feef5 | |||
| 191472d0c4 | |||
| 40c666e3d1 | |||
| 083739e562 | |||
| 71ccb9c0e5 | |||
| 4a5f4ca2ca | |||
| 78ade4a193 | |||
| ae14653e72 | |||
| 01ac2c13e1 | |||
| 9e471b83d9 | |||
| a560d66567 | |||
| 80b39daaff | |||
| fb32388536 | |||
| 3de009ac43 | |||
| b29256f598 | |||
| e7cf472a0f | |||
| 03c59d23b8 | |||
| b6f805f0e4 | |||
| e23c2469ed | |||
| 387c799de1 | |||
| 230dd81342 | |||
| 47bdfec8c2 | |||
| 6a844da259 | |||
| 18bb58d293 | |||
| 045186b3c8 | |||
| b531599893 | |||
| 1da84508d8 | |||
| 4c44fab217 | |||
| 5c3d438a7e | |||
| 92a8efdee1 | |||
| eb1e2ca5f4 | |||
| 5b1022628c | |||
| 33f61ac0fa | |||
| 0b349e102e | |||
| 81371c5a31 | |||
| 85a59e11b9 | |||
| e24ccc16fa | |||
| 89a43b1285 | |||
| ea8af28828 | |||
| 8a2c023343 | |||
| a86fe76e59 | |||
| d703e38c3f | |||
| 9e93b50860 | |||
| 9907967a0a | |||
| 331969cc8d | |||
| 7157f89245 | |||
| 163bb3c8da | |||
| 33f71c62df | |||
| 365c367cc7 | |||
| 7ea95e550b | |||
| 699361e04c | |||
| e835821f3c | |||
| 76d5f6333e | |||
| 91899a9537 | |||
| 5591bb3dbf | |||
| 1adbdf127f | |||
| 8169f5739c | |||
| 127c6734f8 | |||
| a78ce4f6d4 | |||
| 22a2628513 | |||
| e972d17f1c | |||
| e603942b5f | |||
| 0c4bad406e | |||
| 791047753d | |||
| 3784f1f87b | |||
| c0162892ef | |||
| 10d2995429 | |||
| b0d8f5d0b6 | |||
| a64a49df22 | |||
| 349de75dfd | |||
| 1014c26d29 | |||
| d1bacf8950 | |||
| 0595648f4c | |||
| 23c8b34189 | |||
| 932be48125 | |||
| 791eed8a01 | |||
| e791210039 | |||
| 01fa3c1cb3 | |||
| 9663278916 | |||
| 1b03f3f31f | |||
| a73f2974e1 | |||
| bf8ae0f317 | |||
| 545def36e6 | |||
| aaeecc9e53 | |||
| 16895b5c32 | |||
| a68c04001b | |||
| 1dd70f482f | |||
| 103423abf7 | |||
| 569184a507 | |||
| e052c6eafe | |||
| dad4a28eb8 | |||
| 7428cde108 | |||
| 9e40e162ac | |||
| 27c954ec95 | |||
| ec98e5c73c | |||
| 538b3d88aa | |||
| aa4e251295 | |||
| 32f84dc703 | |||
| c8e157b2d6 | |||
| 9122dd7f3a | |||
| 41ddeab41a | |||
| 5ca841de4e | |||
| ca6091b90d | |||
| 5169bc360c | |||
| 4673f6dc5b | |||
| 5da6f2b021 | |||
| 53ce81673b | |||
| e991f36853 | |||
| 211b7812de | |||
| 70cf27789f | |||
| 931a513ac5 | |||
| c4bf7d3a3b | |||
| b9cfcf9c9e | |||
| a418f457f5 | |||
| 83e68c8ee3 | |||
| d26066ff84 | |||
| 4000e1b9dc | |||
| 02bdb46625 | |||
| 96737ded5b | |||
| 6654094480 | |||
| 0cc25203b1 | |||
| 9294c8bcdf | |||
| 8f661337f5 | |||
| db6662c980 | |||
| 2dc7cf28c9 | |||
| c9fd6304fd | |||
| 4d1e4511a3 | |||
| 449aa69033 | |||
| 5fa32f1e2b | |||
| 71bb4b30d2 | |||
| 8b0cb4c909 | |||
| 79729e4b30 | |||
| 7cf9e66a5b | |||
| 917cff6311 | |||
| 7c120aa0cc | |||
| 2c7ce60f42 | |||
| e4ebe6e145 | |||
| 5d0ca3f84c | |||
| 49dfb2605c | |||
| cfddcd79bd | |||
| a07a1697c2 | |||
| 21259e260d | |||
| 438b392cfc | |||
| ac69896f9d | |||
| f5263b090a | |||
| bde7e35623 | |||
| a0488f2c75 | |||
| e8f7995b32 | |||
| 042a0a848d | |||
| a98d8537c7 | |||
| 8ebb4cc70e | |||
| fa8fd42e9a | |||
| aa94bf27ff | |||
| a2524e0a60 | |||
| 9a9644c3eb | |||
| d0f8d55c9f | |||
| 367d8b844a |
201
.github/workflows/main.yml
vendored
201
.github/workflows/main.yml
vendored
@ -1,84 +1,157 @@
|
|||||||
name: build_apk
|
name: Pilipala Release
|
||||||
|
|
||||||
# action事件触发
|
# action事件触发
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
# push tag时触发
|
# push tag时触发
|
||||||
tags:
|
tags:
|
||||||
- 'v*.*.*'
|
- "v*.*.*"
|
||||||
|
|
||||||
# 可以有多个jobs
|
# 可以有多个jobs
|
||||||
jobs:
|
jobs:
|
||||||
build_apk:
|
android:
|
||||||
# 运行环境 ubuntu-latest window-latest mac-latest
|
# 运行环境 ubuntu-latest window-latest mac-latest
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
# 每个jobs中可以有多个steps
|
# 每个jobs中可以有多个steps
|
||||||
steps:
|
steps:
|
||||||
- name: 代码迁出
|
- name: 代码迁出
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
- name: 构建Java环境
|
- name: 构建Java环境
|
||||||
uses: actions/setup-java@v3
|
uses: actions/setup-java@v3
|
||||||
with:
|
with:
|
||||||
distribution: "zulu"
|
distribution: "zulu"
|
||||||
java-version: "17"
|
java-version: "17"
|
||||||
token: ${{secrets.GIT_TOKEN}}
|
token: ${{secrets.GIT_TOKEN}}
|
||||||
|
|
||||||
- name: 检查缓存
|
- name: 检查缓存
|
||||||
uses: actions/cache@v2
|
uses: actions/cache@v2
|
||||||
id: cache-flutter
|
id: cache-flutter
|
||||||
with:
|
with:
|
||||||
path: /root/flutter-sdk # Flutter SDK 的路径
|
path: /root/flutter-sdk # Flutter SDK 的路径
|
||||||
key: ${{ runner.os }}-flutter-${{ hashFiles('**/pubspec.lock') }}
|
key: ${{ runner.os }}-flutter-${{ hashFiles('**/pubspec.lock') }}
|
||||||
|
|
||||||
- name: 安装Flutter
|
- name: 安装Flutter
|
||||||
if: steps.cache-flutter.outputs.cache-hit != 'true'
|
if: steps.cache-flutter.outputs.cache-hit != 'true'
|
||||||
uses: subosito/flutter-action@v2
|
uses: subosito/flutter-action@v2
|
||||||
with:
|
with:
|
||||||
flutter-version: 3.16.4
|
flutter-version: 3.16.5
|
||||||
channel: any
|
channel: any
|
||||||
|
|
||||||
- name: 下载项目依赖
|
- name: 下载项目依赖
|
||||||
run: flutter pub get
|
run: flutter pub get
|
||||||
|
|
||||||
- name: 解码生成 jks
|
- name: 解码生成 jks
|
||||||
run: echo $KEYSTORE_BASE64 | base64 -di > android/app/vvex.jks
|
run: echo $KEYSTORE_BASE64 | base64 -di > android/app/vvex.jks
|
||||||
env:
|
env:
|
||||||
KEYSTORE_BASE64: ${{ secrets.KEYSTORE_BASE64 }}
|
KEYSTORE_BASE64: ${{ secrets.KEYSTORE_BASE64 }}
|
||||||
|
|
||||||
- name: flutter build apk
|
- name: flutter build apk
|
||||||
# 对应 android/app/build.gradle signingConfigs中的配置项
|
run: flutter build apk --release --split-per-abi
|
||||||
run: flutter build apk --release --split-per-abi
|
env:
|
||||||
env:
|
KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }}
|
||||||
KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }}
|
KEY_ALIAS: ${{ secrets.KEY_ALIAS }}
|
||||||
KEY_ALIAS: ${{ secrets.KEY_ALIAS }}
|
KEY_PASSWORD: ${{ secrets.KEY_PASSWORD}}
|
||||||
KEY_PASSWORD: ${{ secrets.KEY_PASSWORD}}
|
|
||||||
|
|
||||||
- name: 获取版本号
|
- name: flutter build apk
|
||||||
id: version
|
run: flutter build apk --release
|
||||||
run: echo "version=${GITHUB_REF#refs/tags/v}" >>$GITHUB_OUTPUT
|
env:
|
||||||
|
KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }}
|
||||||
|
KEY_ALIAS: ${{ secrets.KEY_ALIAS }}
|
||||||
|
KEY_PASSWORD: ${{ secrets.KEY_PASSWORD}}
|
||||||
|
|
||||||
# - name: 获取当前日期
|
- name: 获取版本号
|
||||||
# id: date
|
id: version
|
||||||
# run: echo "date=$(date +'%m%d')" >>$GITHUB_OUTPUT
|
run: echo "version=${GITHUB_REF#refs/tags/v}" >>$GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: 重命名应用 Pili-arm64-v8a-*.*.*.0101.apk
|
# - name: 获取当前日期
|
||||||
run: |
|
# id: date
|
||||||
# DATE=${{ steps.date.outputs.date }}
|
# run: echo "date=$(date +'%m%d')" >>$GITHUB_OUTPUT
|
||||||
for file in build/app/outputs/flutter-apk/app-*-release.apk; do
|
|
||||||
if [[ $file =~ app-(.*)-release.apk ]]; then
|
|
||||||
new_file_name="build/app/outputs/flutter-apk/Pili-${BASH_REMATCH[1]}-${{ steps.version.outputs.version }}.apk"
|
|
||||||
mv "$file" "$new_file_name"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
- name: 构建和发布release
|
- name: 重命名应用
|
||||||
uses: ncipollo/release-action@v1
|
run: |
|
||||||
with:
|
# DATE=${{ steps.date.outputs.date }}
|
||||||
# release title
|
for file in build/app/outputs/flutter-apk/app-*.apk; do
|
||||||
name: v${{ steps.version.outputs.version }}
|
if [[ $file =~ app-(.?*)release.apk ]]; then
|
||||||
artifacts: "build/app/outputs/flutter-apk/Pili-*.apk"
|
new_file_name="build/app/outputs/flutter-apk/Pili-${BASH_REMATCH[1]}${{ steps.version.outputs.version }}.apk"
|
||||||
bodyFile: "change_log/${{steps.version.outputs.version}}.md"
|
mv "$file" "$new_file_name"
|
||||||
token: ${{ secrets.GIT_TOKEN }}
|
fi
|
||||||
allowUpdates: true
|
done
|
||||||
|
|
||||||
|
- name: 上传
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Pilipala-Release
|
||||||
|
path: |
|
||||||
|
build/app/outputs/flutter-apk/Pili-*.apk
|
||||||
|
|
||||||
|
iOS:
|
||||||
|
runs-on: macos-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: 代码迁出
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 安装Flutter
|
||||||
|
if: steps.cache-flutter.outputs.cache-hit != 'true'
|
||||||
|
uses: subosito/flutter-action@v2.10.0
|
||||||
|
with:
|
||||||
|
cache: true
|
||||||
|
flutter-version: 3.16.5
|
||||||
|
|
||||||
|
- name: flutter build ipa
|
||||||
|
run: |
|
||||||
|
flutter build ios --release --no-codesign
|
||||||
|
ln -sf ./build/ios/iphoneos Payload
|
||||||
|
zip -r9 app.ipa Payload/runner.app
|
||||||
|
|
||||||
|
- name: 获取版本号
|
||||||
|
id: version
|
||||||
|
run: echo "version=${GITHUB_REF#refs/tags/v}" >>$GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: 重命名应用
|
||||||
|
run: |
|
||||||
|
DATE=${{ steps.date.outputs.date }}
|
||||||
|
for file in app.ipa; do
|
||||||
|
new_file_name="build/Pili-${{ steps.version.outputs.version }}.ipa"
|
||||||
|
mv "$file" "$new_file_name"
|
||||||
|
done
|
||||||
|
|
||||||
|
- name: 上传
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
if-no-files-found: error
|
||||||
|
name: Pilipala-Release
|
||||||
|
path: |
|
||||||
|
build/Pili-*.ipa
|
||||||
|
|
||||||
|
upload:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
needs:
|
||||||
|
- android
|
||||||
|
- iOS
|
||||||
|
steps:
|
||||||
|
- uses: actions/download-artifact@v3
|
||||||
|
with:
|
||||||
|
name: Pilipala-Release
|
||||||
|
path: ./Pilipala-Release
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: sudo apt-get install tree -y
|
||||||
|
|
||||||
|
- name: Get version
|
||||||
|
id: version
|
||||||
|
run: echo "version=${GITHUB_REF#refs/tags/v}" >>$GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Upload Release
|
||||||
|
uses: ncipollo/release-action@v1
|
||||||
|
with:
|
||||||
|
name: v${{ steps.version.outputs.version }}
|
||||||
|
token: ${{ secrets.GIT_TOKEN }}
|
||||||
|
omitBodyDuringUpdate: true
|
||||||
|
omitNameDuringUpdate: true
|
||||||
|
omitPrereleaseDuringUpdate: true
|
||||||
|
allowUpdates: true
|
||||||
|
artifacts: Pilipala-Release/*
|
||||||
|
|||||||
106
.gitignore
vendored
106
.gitignore
vendored
@ -21,6 +21,29 @@ migrate_working_dir/
|
|||||||
# is commented out by default.
|
# is commented out by default.
|
||||||
#.vscode/
|
#.vscode/
|
||||||
|
|
||||||
|
# Flutter repo-specific
|
||||||
|
/bin/cache/
|
||||||
|
/bin/internal/bootstrap.bat
|
||||||
|
/bin/internal/bootstrap.sh
|
||||||
|
/bin/mingit/
|
||||||
|
/dev/benchmarks/mega_gallery/
|
||||||
|
/dev/bots/.recipe_deps
|
||||||
|
/dev/bots/android_tools/
|
||||||
|
/dev/devicelab/ABresults*.json
|
||||||
|
/dev/docs/doc/
|
||||||
|
/dev/docs/api_docs.zip
|
||||||
|
/dev/docs/flutter.docs.zip
|
||||||
|
/dev/docs/lib/
|
||||||
|
/dev/docs/pubspec.yaml
|
||||||
|
/dev/integration_tests/**/xcuserdata
|
||||||
|
/dev/integration_tests/**/Pods
|
||||||
|
/packages/flutter/coverage/
|
||||||
|
version
|
||||||
|
analysis_benchmark.json
|
||||||
|
|
||||||
|
# packages file containing multi-root paths
|
||||||
|
.packages.generated
|
||||||
|
|
||||||
# Flutter/Dart/Pub related
|
# Flutter/Dart/Pub related
|
||||||
**/doc/api/
|
**/doc/api/
|
||||||
**/ios/Flutter/.last_build_id
|
**/ios/Flutter/.last_build_id
|
||||||
@ -31,14 +54,83 @@ migrate_working_dir/
|
|||||||
.pub-cache/
|
.pub-cache/
|
||||||
.pub/
|
.pub/
|
||||||
/build/
|
/build/
|
||||||
|
flutter_*.png
|
||||||
# Symbolication related
|
linked_*.ds
|
||||||
app.*.symbols
|
unlinked.ds
|
||||||
|
unlinked_spec.ds
|
||||||
|
|
||||||
# Obfuscation related
|
# Obfuscation related
|
||||||
app.*.map.json
|
app.*.map.json
|
||||||
|
|
||||||
# Android Studio will place build artifacts here
|
# Android related
|
||||||
/android/app/debug
|
**/android/**/gradle-wrapper.jar
|
||||||
/android/app/profile
|
.gradle/
|
||||||
/android/app/release
|
**/android/captures/
|
||||||
|
**/android/gradlew
|
||||||
|
**/android/gradlew.bat
|
||||||
|
**/android/local.properties
|
||||||
|
**/android/**/GeneratedPluginRegistrant.java
|
||||||
|
**/android/key.properties
|
||||||
|
*.jks
|
||||||
|
|
||||||
|
# iOS/XCode related
|
||||||
|
**/ios/**/*.mode1v3
|
||||||
|
**/ios/**/*.mode2v3
|
||||||
|
**/ios/**/*.moved-aside
|
||||||
|
**/ios/**/*.pbxuser
|
||||||
|
**/ios/**/*.perspectivev3
|
||||||
|
**/ios/**/*sync/
|
||||||
|
**/ios/**/.sconsign.dblite
|
||||||
|
**/ios/**/.tags*
|
||||||
|
**/ios/**/.vagrant/
|
||||||
|
**/ios/**/DerivedData/
|
||||||
|
**/ios/**/Icon?
|
||||||
|
**/ios/**/Pods/
|
||||||
|
**/ios/**/.symlinks/
|
||||||
|
**/ios/**/profile
|
||||||
|
**/ios/**/xcuserdata
|
||||||
|
**/ios/.generated/
|
||||||
|
**/ios/Flutter/.last_build_id
|
||||||
|
**/ios/Flutter/App.framework
|
||||||
|
**/ios/Flutter/Flutter.framework
|
||||||
|
**/ios/Flutter/Flutter.podspec
|
||||||
|
**/ios/Flutter/Generated.xcconfig
|
||||||
|
**/ios/Flutter/ephemeral
|
||||||
|
**/ios/Flutter/app.flx
|
||||||
|
**/ios/Flutter/app.zip
|
||||||
|
**/ios/Flutter/flutter_assets/
|
||||||
|
**/ios/Flutter/flutter_export_environment.sh
|
||||||
|
**/ios/ServiceDefinitions.json
|
||||||
|
**/ios/Runner/GeneratedPluginRegistrant.*
|
||||||
|
|
||||||
|
# macOS
|
||||||
|
**/Flutter/ephemeral/
|
||||||
|
**/Pods/
|
||||||
|
**/macos/Flutter/GeneratedPluginRegistrant.swift
|
||||||
|
**/macos/Flutter/ephemeral
|
||||||
|
**/xcuserdata/
|
||||||
|
|
||||||
|
# Windows
|
||||||
|
**/windows/flutter/generated_plugin_registrant.cc
|
||||||
|
**/windows/flutter/generated_plugin_registrant.h
|
||||||
|
**/windows/flutter/generated_plugins.cmake
|
||||||
|
|
||||||
|
# Linux
|
||||||
|
**/linux/flutter/generated_plugin_registrant.cc
|
||||||
|
**/linux/flutter/generated_plugin_registrant.h
|
||||||
|
**/linux/flutter/generated_plugins.cmake
|
||||||
|
|
||||||
|
# Coverage
|
||||||
|
coverage/
|
||||||
|
|
||||||
|
# Symbols
|
||||||
|
app.*.symbols
|
||||||
|
|
||||||
|
# Exceptions to above rules.
|
||||||
|
!**/ios/**/default.mode1v3
|
||||||
|
!**/ios/**/default.mode2v3
|
||||||
|
!**/ios/**/default.pbxuser
|
||||||
|
!**/ios/**/default.perspectivev3
|
||||||
|
!/packages/flutter_tools/test/data/dart_dependencies_test/**/.packages
|
||||||
|
!/dev/ci/**/Gemfile.lock
|
||||||
|
!.vscode/settings.json
|
||||||
@ -58,11 +58,10 @@ android {
|
|||||||
applicationId "com.guozhigq.pilipala"
|
applicationId "com.guozhigq.pilipala"
|
||||||
// You can update the following values to match your application needs.
|
// You can update the following values to match your application needs.
|
||||||
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration.
|
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration.
|
||||||
// minSdkVersion flutter.minSdkVersion
|
|
||||||
targetSdkVersion flutter.targetSdkVersion
|
targetSdkVersion flutter.targetSdkVersion
|
||||||
versionCode flutterVersionCode.toInteger()
|
versionCode flutterVersionCode.toInteger()
|
||||||
versionName flutterVersionName
|
versionName flutterVersionName
|
||||||
minSdkVersion 19
|
minSdkVersion 21
|
||||||
multiDexEnabled true
|
multiDexEnabled true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,3 +94,14 @@ flutter {
|
|||||||
dependencies {
|
dependencies {
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
|
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ext.abiCodes = ["x86_64": 1, "armeabi-v7a": 2, "arm64-v8a": 3]
|
||||||
|
import com.android.build.OutputFile
|
||||||
|
android.applicationVariants.all { variant ->
|
||||||
|
variant.outputs.each { output ->
|
||||||
|
def abiVersionCode = project.ext.abiCodes.get(output.getFilter(OutputFile.ABI))
|
||||||
|
if (abiVersionCode != null) {
|
||||||
|
output.versionCodeOverride = variant.versionCode * 10 + abiVersionCode
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -65,11 +65,13 @@
|
|||||||
/>
|
/>
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.intent.action.MAIN"/>
|
<action android:name="android.intent.action.MAIN"/>
|
||||||
|
<category android:name="android.intent.category.LAUNCHER"/>
|
||||||
|
</intent-filter>
|
||||||
|
<intent-filter>
|
||||||
<action android:name="android.intent.action.VIEW" />
|
<action android:name="android.intent.action.VIEW" />
|
||||||
<action android:name="android.intent.action.SEARCH" />
|
<action android:name="android.intent.action.SEARCH" />
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
<category android:name="android.intent.category.BROWSABLE" />
|
<category android:name="android.intent.category.BROWSABLE" />
|
||||||
<category android:name="android.intent.category.LAUNCHER"/>
|
|
||||||
<data android:scheme="bilibili" android:host="forward" />
|
<data android:scheme="bilibili" android:host="forward" />
|
||||||
<data android:scheme="bilibili" android:host="comment"
|
<data android:scheme="bilibili" android:host="comment"
|
||||||
android:pathPattern="/detail/.*/.*/.*" />
|
android:pathPattern="/detail/.*/.*/.*" />
|
||||||
|
|||||||
BIN
assets/images/live/default_bg.webp
Normal file
BIN
assets/images/live/default_bg.webp
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 16 KiB |
39
change_log/1.0.17.0125.md
Normal file
39
change_log/1.0.17.0125.md
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
## 1.0.17
|
||||||
|
|
||||||
|
|
||||||
|
### 功能
|
||||||
|
+ 视频全屏时隐藏进度条
|
||||||
|
+ 动态内容增加投稿跳转
|
||||||
|
+ 未开启自动播放时点击封面播放
|
||||||
|
+ 弹幕发送标识
|
||||||
|
+ 定时关闭
|
||||||
|
+ 推荐视频卡片拉黑up功能
|
||||||
|
+ 首页tabbar编辑排序
|
||||||
|
|
||||||
|
### 修复
|
||||||
|
+ 连续跳转搜索页未刷新
|
||||||
|
+ 搜索结果为空时页面异常
|
||||||
|
+ 评论区链接解析
|
||||||
|
+ 视频全屏状态栏背景色
|
||||||
|
+ 私信对话气泡位置
|
||||||
|
+ 设置up关注分组样式
|
||||||
|
+ 每次推荐请求数据相同
|
||||||
|
+ iOS代理网络异常
|
||||||
|
+ 双击切换播放状态无声
|
||||||
|
+ 设置自定义倍速白屏
|
||||||
|
+ 免登录查看1080p
|
||||||
|
|
||||||
|
### 优化
|
||||||
|
+ 首页web端推荐观看数展示
|
||||||
|
+ 首页web端推荐接口更新
|
||||||
|
+ 首页样式
|
||||||
|
+ 搜索页跳转
|
||||||
|
+ 弹幕资源优化
|
||||||
|
+ 图片渲染占用内存优化(部分)
|
||||||
|
+ 两次返回退出应用
|
||||||
|
+ schame 补充
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
更多更新日志可在Github上查看
|
||||||
|
问题反馈、功能建议请查看「关于」页面。
|
||||||
16
change_log/1.0.18.0130.md
Normal file
16
change_log/1.0.18.0130.md
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
## 1.0.18
|
||||||
|
|
||||||
|
|
||||||
|
### 功能
|
||||||
|
|
||||||
|
|
||||||
|
### 修复
|
||||||
|
|
||||||
|
|
||||||
|
### 优化
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
更多更新日志可在Github上查看
|
||||||
|
问题反馈、功能建议请查看「关于」页面。
|
||||||
15
change_log/1.0.19.0131.md
Normal file
15
change_log/1.0.19.0131.md
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
## 1.0.19
|
||||||
|
|
||||||
|
|
||||||
|
### 修复
|
||||||
|
+ 视频404、评论加载错误
|
||||||
|
+ bvav转换
|
||||||
|
|
||||||
|
### 优化
|
||||||
|
+ 视频详情页内存占用
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
更多更新日志可在Github上查看
|
||||||
|
问题反馈、功能建议请查看「关于」页面。
|
||||||
@ -13,8 +13,13 @@ PODS:
|
|||||||
- device_info_plus (0.0.1):
|
- device_info_plus (0.0.1):
|
||||||
- Flutter
|
- Flutter
|
||||||
- Flutter (1.0.0)
|
- Flutter (1.0.0)
|
||||||
|
- flutter_mailer (0.0.1):
|
||||||
|
- Flutter
|
||||||
- flutter_volume_controller (0.0.1):
|
- flutter_volume_controller (0.0.1):
|
||||||
- Flutter
|
- Flutter
|
||||||
|
- fluttertoast (0.0.2):
|
||||||
|
- Flutter
|
||||||
|
- Toast
|
||||||
- FMDB (2.7.5):
|
- FMDB (2.7.5):
|
||||||
- FMDB/standard (= 2.7.5)
|
- FMDB/standard (= 2.7.5)
|
||||||
- FMDB/standard (2.7.5)
|
- FMDB/standard (2.7.5)
|
||||||
@ -49,6 +54,7 @@ PODS:
|
|||||||
- Flutter
|
- Flutter
|
||||||
- system_proxy (0.0.1):
|
- system_proxy (0.0.1):
|
||||||
- Flutter
|
- Flutter
|
||||||
|
- Toast (4.1.0)
|
||||||
- url_launcher_ios (0.0.1):
|
- url_launcher_ios (0.0.1):
|
||||||
- Flutter
|
- Flutter
|
||||||
- volume_controller (0.0.1):
|
- volume_controller (0.0.1):
|
||||||
@ -68,7 +74,9 @@ DEPENDENCIES:
|
|||||||
- connectivity_plus (from `.symlinks/plugins/connectivity_plus/ios`)
|
- connectivity_plus (from `.symlinks/plugins/connectivity_plus/ios`)
|
||||||
- device_info_plus (from `.symlinks/plugins/device_info_plus/ios`)
|
- device_info_plus (from `.symlinks/plugins/device_info_plus/ios`)
|
||||||
- Flutter (from `Flutter`)
|
- Flutter (from `Flutter`)
|
||||||
|
- flutter_mailer (from `.symlinks/plugins/flutter_mailer/ios`)
|
||||||
- flutter_volume_controller (from `.symlinks/plugins/flutter_volume_controller/ios`)
|
- flutter_volume_controller (from `.symlinks/plugins/flutter_volume_controller/ios`)
|
||||||
|
- fluttertoast (from `.symlinks/plugins/fluttertoast/ios`)
|
||||||
- gt3_flutter_plugin (from `.symlinks/plugins/gt3_flutter_plugin/ios`)
|
- gt3_flutter_plugin (from `.symlinks/plugins/gt3_flutter_plugin/ios`)
|
||||||
- media_kit_libs_ios_video (from `.symlinks/plugins/media_kit_libs_ios_video/ios`)
|
- media_kit_libs_ios_video (from `.symlinks/plugins/media_kit_libs_ios_video/ios`)
|
||||||
- media_kit_native_event_loop (from `.symlinks/plugins/media_kit_native_event_loop/ios`)
|
- media_kit_native_event_loop (from `.symlinks/plugins/media_kit_native_event_loop/ios`)
|
||||||
@ -93,6 +101,7 @@ SPEC REPOS:
|
|||||||
- FMDB
|
- FMDB
|
||||||
- GT3Captcha-iOS
|
- GT3Captcha-iOS
|
||||||
- ReachabilitySwift
|
- ReachabilitySwift
|
||||||
|
- Toast
|
||||||
|
|
||||||
EXTERNAL SOURCES:
|
EXTERNAL SOURCES:
|
||||||
appscheme:
|
appscheme:
|
||||||
@ -109,8 +118,12 @@ EXTERNAL SOURCES:
|
|||||||
:path: ".symlinks/plugins/device_info_plus/ios"
|
:path: ".symlinks/plugins/device_info_plus/ios"
|
||||||
Flutter:
|
Flutter:
|
||||||
:path: Flutter
|
:path: Flutter
|
||||||
|
flutter_mailer:
|
||||||
|
:path: ".symlinks/plugins/flutter_mailer/ios"
|
||||||
flutter_volume_controller:
|
flutter_volume_controller:
|
||||||
:path: ".symlinks/plugins/flutter_volume_controller/ios"
|
:path: ".symlinks/plugins/flutter_volume_controller/ios"
|
||||||
|
fluttertoast:
|
||||||
|
:path: ".symlinks/plugins/fluttertoast/ios"
|
||||||
gt3_flutter_plugin:
|
gt3_flutter_plugin:
|
||||||
:path: ".symlinks/plugins/gt3_flutter_plugin/ios"
|
:path: ".symlinks/plugins/gt3_flutter_plugin/ios"
|
||||||
media_kit_libs_ios_video:
|
media_kit_libs_ios_video:
|
||||||
@ -156,7 +169,9 @@ SPEC CHECKSUMS:
|
|||||||
connectivity_plus: 07c49e96d7fc92bc9920617b83238c4d178b446a
|
connectivity_plus: 07c49e96d7fc92bc9920617b83238c4d178b446a
|
||||||
device_info_plus: c6fb39579d0f423935b0c9ce7ee2f44b71b9fce6
|
device_info_plus: c6fb39579d0f423935b0c9ce7ee2f44b71b9fce6
|
||||||
Flutter: f04841e97a9d0b0a8025694d0796dd46242b2854
|
Flutter: f04841e97a9d0b0a8025694d0796dd46242b2854
|
||||||
|
flutter_mailer: 2ef5a67087bc8c6c4cefd04a178bf1ae2c94cd83
|
||||||
flutter_volume_controller: e4d5832f08008180f76e30faf671ffd5a425e529
|
flutter_volume_controller: e4d5832f08008180f76e30faf671ffd5a425e529
|
||||||
|
fluttertoast: 31b00dabfa7fb7bacd9e7dbee580d7a2ff4bf265
|
||||||
FMDB: 2ce00b547f966261cd18927a3ddb07cb6f3db82a
|
FMDB: 2ce00b547f966261cd18927a3ddb07cb6f3db82a
|
||||||
gt3_flutter_plugin: bfa1f26e9a09dc00401514be5ed437f964cabf23
|
gt3_flutter_plugin: bfa1f26e9a09dc00401514be5ed437f964cabf23
|
||||||
GT3Captcha-iOS: 5e3b1077834d8a9d6f4d64a447a30af3e14affe6
|
GT3Captcha-iOS: 5e3b1077834d8a9d6f4d64a447a30af3e14affe6
|
||||||
@ -173,6 +188,7 @@ SPEC CHECKSUMS:
|
|||||||
sqflite: 31f7eba61e3074736dff8807a9b41581e4f7f15a
|
sqflite: 31f7eba61e3074736dff8807a9b41581e4f7f15a
|
||||||
status_bar_control: 7c84146799e6a076315cc1550f78ef53aae3e446
|
status_bar_control: 7c84146799e6a076315cc1550f78ef53aae3e446
|
||||||
system_proxy: bec1a5c5af67dd3e3ebf43979400a8756c04cc44
|
system_proxy: bec1a5c5af67dd3e3ebf43979400a8756c04cc44
|
||||||
|
Toast: ec33c32b8688982cecc6348adeae667c1b9938da
|
||||||
url_launcher_ios: bf5ce03e0e2088bad9cc378ea97fa0ed5b49673b
|
url_launcher_ios: bf5ce03e0e2088bad9cc378ea97fa0ed5b49673b
|
||||||
volume_controller: 531ddf792994285c9b17f9d8a7e4dcdd29b3eae9
|
volume_controller: 531ddf792994285c9b17f9d8a7e4dcdd29b3eae9
|
||||||
wakelock_plus: 8b09852c8876491e4b6d179e17dfe2a0b5f60d47
|
wakelock_plus: 8b09852c8876491e4b6d179e17dfe2a0b5f60d47
|
||||||
|
|||||||
@ -20,7 +20,7 @@ class ContentContainer extends StatelessWidget {
|
|||||||
builder: (BuildContext context, BoxConstraints constraints) {
|
builder: (BuildContext context, BoxConstraints constraints) {
|
||||||
return SingleChildScrollView(
|
return SingleChildScrollView(
|
||||||
clipBehavior: childClipBehavior ?? Clip.hardEdge,
|
clipBehavior: childClipBehavior ?? Clip.hardEdge,
|
||||||
physics: isScrollable ? null : NeverScrollableScrollPhysics(),
|
physics: isScrollable ? null : const NeverScrollableScrollPhysics(),
|
||||||
child: ConstrainedBox(
|
child: ConstrainedBox(
|
||||||
constraints: constraints.copyWith(
|
constraints: constraints.copyWith(
|
||||||
minHeight: constraints.maxHeight,
|
minHeight: constraints.maxHeight,
|
||||||
@ -34,7 +34,7 @@ class ContentContainer extends StatelessWidget {
|
|||||||
child: contentWidget!,
|
child: contentWidget!,
|
||||||
)
|
)
|
||||||
else
|
else
|
||||||
Spacer(),
|
const Spacer(),
|
||||||
if (bottomWidget != null) bottomWidget!,
|
if (bottomWidget != null) bottomWidget!,
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@ -2,16 +2,17 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
|
|
||||||
Box setting = GStrorage.setting;
|
Box<dynamic> setting = GStrorage.setting;
|
||||||
|
|
||||||
class CustomToast extends StatelessWidget {
|
class CustomToast extends StatelessWidget {
|
||||||
|
const CustomToast({super.key, required this.msg});
|
||||||
|
|
||||||
final String msg;
|
final String msg;
|
||||||
const CustomToast({Key? key, required this.msg}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
double toastOpacity =
|
final double toastOpacity =
|
||||||
setting.get(SettingBoxKey.defaultToastOp, defaultValue: 1.0);
|
setting.get(SettingBoxKey.defaultToastOp, defaultValue: 1.0) as double;
|
||||||
return Container(
|
return Container(
|
||||||
margin:
|
margin:
|
||||||
EdgeInsets.only(bottom: MediaQuery.of(context).padding.bottom + 30),
|
EdgeInsets.only(bottom: MediaQuery.of(context).padding.bottom + 30),
|
||||||
|
|||||||
@ -1,45 +1,46 @@
|
|||||||
import 'package:get/get.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_html/flutter_html.dart';
|
import 'package:flutter_html/flutter_html.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'network_img_layer.dart';
|
||||||
|
|
||||||
// ignore: must_be_immutable
|
// ignore: must_be_immutable
|
||||||
class HtmlRender extends StatelessWidget {
|
class HtmlRender extends StatelessWidget {
|
||||||
String? htmlContent;
|
const HtmlRender({
|
||||||
final int? imgCount;
|
|
||||||
final List? imgList;
|
|
||||||
|
|
||||||
HtmlRender({
|
|
||||||
this.htmlContent,
|
this.htmlContent,
|
||||||
this.imgCount,
|
this.imgCount,
|
||||||
this.imgList,
|
this.imgList,
|
||||||
super.key,
|
super.key,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
final String? htmlContent;
|
||||||
|
final int? imgCount;
|
||||||
|
final List<String>? imgList;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Html(
|
return Html(
|
||||||
data: htmlContent,
|
data: htmlContent,
|
||||||
onLinkTap: (url, buildContext, attributes) => {},
|
onLinkTap: (String? url, Map<String, String> buildContext, attributes) {},
|
||||||
extensions: [
|
extensions: [
|
||||||
TagExtension(
|
TagExtension(
|
||||||
tagsToExtend: {"img"},
|
tagsToExtend: <String>{'img'},
|
||||||
builder: (extensionContext) {
|
builder: (ExtensionContext extensionContext) {
|
||||||
try {
|
try {
|
||||||
Map attributes = extensionContext.attributes;
|
final Map<String, dynamic> attributes =
|
||||||
List key = attributes.keys.toList();
|
extensionContext.attributes;
|
||||||
String? imgUrl = key.contains('src')
|
final List<dynamic> key = attributes.keys.toList();
|
||||||
? attributes['src']
|
String imgUrl = key.contains('src')
|
||||||
: attributes['data-src'];
|
? attributes['src'] as String
|
||||||
if (imgUrl!.startsWith('//')) {
|
: attributes['data-src'] as String;
|
||||||
|
if (imgUrl.startsWith('//')) {
|
||||||
imgUrl = 'https:$imgUrl';
|
imgUrl = 'https:$imgUrl';
|
||||||
}
|
}
|
||||||
if (imgUrl.startsWith('http://')) {
|
if (imgUrl.startsWith('http://')) {
|
||||||
imgUrl = imgUrl.replaceAll('http://', 'https://');
|
imgUrl = imgUrl.replaceAll('http://', 'https://');
|
||||||
}
|
}
|
||||||
imgUrl = imgUrl.contains('@') ? imgUrl.split('@').first : imgUrl;
|
imgUrl = imgUrl.contains('@') ? imgUrl.split('@').first : imgUrl;
|
||||||
bool isEmote = imgUrl.contains('/emote/');
|
final bool isEmote = imgUrl.contains('/emote/');
|
||||||
bool isMall = imgUrl.contains('/mall/');
|
final bool isMall = imgUrl.contains('/mall/');
|
||||||
if (isMall) {
|
if (isMall) {
|
||||||
return const SizedBox();
|
return const SizedBox();
|
||||||
}
|
}
|
||||||
@ -58,38 +59,37 @@ class HtmlRender extends StatelessWidget {
|
|||||||
src: imgUrl,
|
src: imgUrl,
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
print(err);
|
|
||||||
return const SizedBox();
|
return const SizedBox();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
style: {
|
style: {
|
||||||
"html": Style(
|
'html': Style(
|
||||||
fontSize: FontSize.medium,
|
fontSize: FontSize.medium,
|
||||||
lineHeight: LineHeight.percent(140),
|
lineHeight: LineHeight.percent(140),
|
||||||
),
|
),
|
||||||
"body": Style(margin: Margins.zero, padding: HtmlPaddings.zero),
|
'body': Style(margin: Margins.zero, padding: HtmlPaddings.zero),
|
||||||
"a": Style(
|
'a': Style(
|
||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
textDecoration: TextDecoration.none,
|
textDecoration: TextDecoration.none,
|
||||||
),
|
),
|
||||||
"p": Style(
|
'p': Style(
|
||||||
margin: Margins.only(bottom: 10),
|
margin: Margins.only(bottom: 10),
|
||||||
),
|
),
|
||||||
"span": Style(
|
'span': Style(
|
||||||
fontSize: FontSize.medium,
|
fontSize: FontSize.medium,
|
||||||
height: Height(1.65),
|
height: Height(1.65),
|
||||||
),
|
),
|
||||||
"div": Style(height: Height.auto()),
|
'div': Style(height: Height.auto()),
|
||||||
"li > p": Style(
|
'li > p': Style(
|
||||||
display: Display.inline,
|
display: Display.inline,
|
||||||
),
|
),
|
||||||
"li": Style(
|
'li': Style(
|
||||||
padding: HtmlPaddings.only(bottom: 4),
|
padding: HtmlPaddings.only(bottom: 4),
|
||||||
textAlign: TextAlign.justify,
|
textAlign: TextAlign.justify,
|
||||||
),
|
),
|
||||||
"img": Style(margin: Margins.only(top: 4, bottom: 4)),
|
'img': Style(margin: Margins.only(top: 4, bottom: 4)),
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -22,20 +22,27 @@ class HttpError extends StatelessWidget {
|
|||||||
"assets/images/error.svg",
|
"assets/images/error.svg",
|
||||||
height: 200,
|
height: 200,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 20),
|
const SizedBox(height: 30),
|
||||||
Text(
|
Text(
|
||||||
errMsg ?? '请求异常',
|
errMsg ?? '请求异常',
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
style: Theme.of(context).textTheme.titleSmall,
|
style: Theme.of(context).textTheme.titleSmall,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 30),
|
const SizedBox(height: 20),
|
||||||
OutlinedButton.icon(
|
FilledButton.tonal(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
fn!();
|
fn!();
|
||||||
},
|
},
|
||||||
icon: const Icon(Icons.arrow_forward_outlined, size: 20),
|
style: ButtonStyle(
|
||||||
label: Text(btnText ?? '点击重试'),
|
backgroundColor: MaterialStateProperty.resolveWith((states) {
|
||||||
)
|
return Theme.of(context).colorScheme.primary.withAlpha(20);
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
child: Text(
|
||||||
|
btnText ?? '点击重试',
|
||||||
|
style: TextStyle(color: Theme.of(context).colorScheme.primary),
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../../utils/utils.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import '../constants.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'network_img_layer.dart';
|
||||||
|
|
||||||
class LiveCard extends StatelessWidget {
|
class LiveCard extends StatelessWidget {
|
||||||
// ignore: prefer_typing_uninitialized_variables
|
// ignore: prefer_typing_uninitialized_variables
|
||||||
final liveItem;
|
final dynamic liveItem;
|
||||||
|
|
||||||
const LiveCard({
|
const LiveCard({
|
||||||
Key? key,
|
Key? key,
|
||||||
@ -14,7 +14,7 @@ class LiveCard extends StatelessWidget {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
String heroTag = Utils.makeHeroTag(liveItem.roomid);
|
final String heroTag = Utils.makeHeroTag(liveItem.roomid);
|
||||||
|
|
||||||
return Card(
|
return Card(
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
@ -23,7 +23,6 @@ class LiveCard extends StatelessWidget {
|
|||||||
borderRadius: BorderRadius.circular(0),
|
borderRadius: BorderRadius.circular(0),
|
||||||
side: BorderSide(
|
side: BorderSide(
|
||||||
color: Theme.of(context).dividerColor.withOpacity(0.08),
|
color: Theme.of(context).dividerColor.withOpacity(0.08),
|
||||||
width: 1,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
margin: EdgeInsets.zero,
|
margin: EdgeInsets.zero,
|
||||||
@ -33,15 +32,16 @@ class LiveCard extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
AspectRatio(
|
AspectRatio(
|
||||||
aspectRatio: StyleString.aspectRatio,
|
aspectRatio: StyleString.aspectRatio,
|
||||||
child: LayoutBuilder(builder: (context, boxConstraints) {
|
child: LayoutBuilder(builder:
|
||||||
double maxWidth = boxConstraints.maxWidth;
|
(BuildContext context, BoxConstraints boxConstraints) {
|
||||||
double maxHeight = boxConstraints.maxHeight;
|
final double maxWidth = boxConstraints.maxWidth;
|
||||||
|
final double maxHeight = boxConstraints.maxHeight;
|
||||||
return Stack(
|
return Stack(
|
||||||
children: [
|
children: [
|
||||||
Hero(
|
Hero(
|
||||||
tag: heroTag,
|
tag: heroTag,
|
||||||
child: NetworkImgLayer(
|
child: NetworkImgLayer(
|
||||||
src: liveItem.cover,
|
src: liveItem.cover as String,
|
||||||
type: 'emote',
|
type: 'emote',
|
||||||
width: maxWidth,
|
width: maxWidth,
|
||||||
height: maxHeight,
|
height: maxHeight,
|
||||||
@ -58,7 +58,7 @@ class LiveCard extends StatelessWidget {
|
|||||||
// view: liveItem.stat.view,
|
// view: liveItem.stat.view,
|
||||||
// danmaku: liveItem.stat.danmaku,
|
// danmaku: liveItem.stat.danmaku,
|
||||||
// duration: liveItem.duration,
|
// duration: liveItem.duration,
|
||||||
online: liveItem.online,
|
online: liveItem.online as int,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -90,7 +90,7 @@ class LiveContent extends StatelessWidget {
|
|||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
liveItem.title,
|
liveItem.title as String,
|
||||||
textAlign: TextAlign.start,
|
textAlign: TextAlign.start,
|
||||||
style: const TextStyle(fontSize: 13, fontWeight: FontWeight.w500),
|
style: const TextStyle(fontSize: 13, fontWeight: FontWeight.w500),
|
||||||
maxLines: 2,
|
maxLines: 2,
|
||||||
@ -99,7 +99,7 @@ class LiveContent extends StatelessWidget {
|
|||||||
SizedBox(
|
SizedBox(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
child: Text(
|
child: Text(
|
||||||
liveItem.uname,
|
liveItem.uname as String,
|
||||||
maxLines: 1,
|
maxLines: 1,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: Theme.of(context).textTheme.labelMedium!.fontSize,
|
fontSize: Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
@ -114,9 +114,9 @@ class LiveContent extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class LiveStat extends StatelessWidget {
|
class LiveStat extends StatelessWidget {
|
||||||
final int? online;
|
const LiveStat({super.key, required this.online});
|
||||||
|
|
||||||
const LiveStat({Key? key, required this.online}) : super(key: key);
|
final int? online;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@ -136,7 +136,7 @@ class LiveStat extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: <Widget>[
|
||||||
// Row(
|
// Row(
|
||||||
// children: [
|
// children: [
|
||||||
// StatView(
|
// StatView(
|
||||||
|
|||||||
@ -1,78 +1,101 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart';
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import 'package:pilipala/utils/extension.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import '../../utils/storage.dart';
|
||||||
|
import '../constants.dart';
|
||||||
|
|
||||||
Box setting = GStrorage.setting;
|
Box<dynamic> setting = GStrorage.setting;
|
||||||
|
|
||||||
class NetworkImgLayer extends StatelessWidget {
|
class NetworkImgLayer extends StatelessWidget {
|
||||||
final String? src;
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final double? cacheW;
|
|
||||||
final double? cacheH;
|
|
||||||
final String? type;
|
|
||||||
final Duration? fadeOutDuration;
|
|
||||||
final Duration? fadeInDuration;
|
|
||||||
final int? quality;
|
|
||||||
|
|
||||||
const NetworkImgLayer({
|
const NetworkImgLayer({
|
||||||
Key? key,
|
super.key,
|
||||||
this.src,
|
this.src,
|
||||||
required this.width,
|
required this.width,
|
||||||
required this.height,
|
required this.height,
|
||||||
this.cacheW,
|
|
||||||
this.cacheH,
|
|
||||||
this.type,
|
this.type,
|
||||||
this.fadeOutDuration,
|
this.fadeOutDuration,
|
||||||
this.fadeInDuration,
|
this.fadeInDuration,
|
||||||
// 图片质量 默认1%
|
// 图片质量 默认1%
|
||||||
this.quality,
|
this.quality,
|
||||||
}) : super(key: key);
|
this.origAspectRatio,
|
||||||
|
});
|
||||||
|
|
||||||
|
final String? src;
|
||||||
|
final double width;
|
||||||
|
final double height;
|
||||||
|
final String? type;
|
||||||
|
final Duration? fadeOutDuration;
|
||||||
|
final Duration? fadeInDuration;
|
||||||
|
final int? quality;
|
||||||
|
final double? origAspectRatio;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
double pr = MediaQuery.of(context).devicePixelRatio;
|
final String imageUrl =
|
||||||
int picQuality = setting.get(SettingBoxKey.defaultPicQa, defaultValue: 10);
|
'${src!.startsWith('//') ? 'https:${src!}' : src!}@${quality ?? 100}q.webp';
|
||||||
|
int? memCacheWidth, memCacheHeight;
|
||||||
|
double aspectRatio = (width / height).toDouble();
|
||||||
|
|
||||||
// double pr = 2;
|
void setMemCacheSizes() {
|
||||||
return src != ''
|
if (aspectRatio > 1) {
|
||||||
|
memCacheHeight = height.cacheSize(context);
|
||||||
|
} else if (aspectRatio < 1) {
|
||||||
|
memCacheWidth = width.cacheSize(context);
|
||||||
|
} else {
|
||||||
|
if (origAspectRatio != null && origAspectRatio! > 1) {
|
||||||
|
memCacheWidth = width.cacheSize(context);
|
||||||
|
} else if (origAspectRatio != null && origAspectRatio! < 1) {
|
||||||
|
memCacheHeight = height.cacheSize(context);
|
||||||
|
} else {
|
||||||
|
memCacheWidth = width.cacheSize(context);
|
||||||
|
memCacheHeight = height.cacheSize(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setMemCacheSizes();
|
||||||
|
|
||||||
|
if (memCacheWidth == null && memCacheHeight == null) {
|
||||||
|
memCacheWidth = width.toInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
return src != '' && src != null
|
||||||
? ClipRRect(
|
? ClipRRect(
|
||||||
clipBehavior: Clip.hardEdge,
|
clipBehavior: Clip.antiAlias,
|
||||||
borderRadius: BorderRadius.circular(type == 'avatar'
|
borderRadius: BorderRadius.circular(
|
||||||
? 50
|
type == 'avatar'
|
||||||
: type == 'emote'
|
? 50
|
||||||
? 0
|
: type == 'emote'
|
||||||
: StyleString.imgRadius.x),
|
? 0
|
||||||
|
: StyleString.imgRadius.x,
|
||||||
|
),
|
||||||
child: CachedNetworkImage(
|
child: CachedNetworkImage(
|
||||||
imageUrl:
|
imageUrl: imageUrl,
|
||||||
'${src!.startsWith('//') ? 'https:${src!}' : src!}@${quality ?? picQuality}q.webp',
|
width: width,
|
||||||
width: width ?? double.infinity,
|
height: height,
|
||||||
height: height ?? double.infinity,
|
memCacheWidth: memCacheWidth,
|
||||||
alignment: Alignment.center,
|
memCacheHeight: memCacheHeight,
|
||||||
maxWidthDiskCache: ((cacheW ?? width!) * pr).toInt(),
|
|
||||||
// maxHeightDiskCache: (cacheH ?? height!).toInt(),
|
|
||||||
memCacheWidth: ((cacheW ?? width!) * pr).toInt(),
|
|
||||||
// memCacheHeight: (cacheH ?? height!).toInt(),
|
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
fadeOutDuration:
|
fadeOutDuration:
|
||||||
fadeOutDuration ?? const Duration(milliseconds: 200),
|
fadeOutDuration ?? const Duration(milliseconds: 120),
|
||||||
fadeInDuration:
|
fadeInDuration:
|
||||||
fadeInDuration ?? const Duration(milliseconds: 200),
|
fadeInDuration ?? const Duration(milliseconds: 120),
|
||||||
// filterQuality: FilterQuality.high,
|
filterQuality: FilterQuality.high,
|
||||||
errorWidget: (context, url, error) => placeholder(context),
|
errorWidget: (BuildContext context, String url, Object error) =>
|
||||||
placeholder: (context, url) => placeholder(context),
|
placeholder(context),
|
||||||
|
placeholder: (BuildContext context, String url) =>
|
||||||
|
placeholder(context),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
: placeholder(context);
|
: placeholder(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget placeholder(context) {
|
Widget placeholder(BuildContext context) {
|
||||||
return Container(
|
return Container(
|
||||||
width: width ?? double.infinity,
|
width: width,
|
||||||
height: height ?? double.infinity,
|
height: height,
|
||||||
clipBehavior: Clip.hardEdge,
|
clipBehavior: Clip.antiAlias,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Theme.of(context).colorScheme.onInverseSurface.withOpacity(0.4),
|
color: Theme.of(context).colorScheme.onInverseSurface.withOpacity(0.4),
|
||||||
borderRadius: BorderRadius.circular(type == 'avatar'
|
borderRadius: BorderRadius.circular(type == 'avatar'
|
||||||
@ -82,13 +105,16 @@ class NetworkImgLayer extends StatelessWidget {
|
|||||||
: StyleString.imgRadius.x),
|
: StyleString.imgRadius.x),
|
||||||
),
|
),
|
||||||
child: Center(
|
child: Center(
|
||||||
child: Image.asset(
|
child: Image.asset(
|
||||||
type == 'avatar'
|
type == 'avatar'
|
||||||
? 'assets/images/noface.jpeg'
|
? 'assets/images/noface.jpeg'
|
||||||
: 'assets/images/loading.png',
|
: 'assets/images/loading.png',
|
||||||
width: 300,
|
width: width,
|
||||||
height: 300,
|
height: height,
|
||||||
)),
|
cacheWidth: width.cacheSize(context),
|
||||||
|
cacheHeight: height.cacheSize(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,16 +1,17 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../../utils/download.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import '../constants.dart';
|
||||||
import 'package:pilipala/utils/download.dart';
|
import 'network_img_layer.dart';
|
||||||
|
|
||||||
class OverlayPop extends StatelessWidget {
|
class OverlayPop extends StatelessWidget {
|
||||||
|
const OverlayPop({super.key, this.videoItem, this.closeFn});
|
||||||
|
|
||||||
final dynamic videoItem;
|
final dynamic videoItem;
|
||||||
final Function? closeFn;
|
final Function? closeFn;
|
||||||
const OverlayPop({super.key, this.videoItem, this.closeFn});
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
double imgWidth = MediaQuery.of(context).size.width - 8 * 2;
|
final double imgWidth = MediaQuery.sizeOf(context).width - 8 * 2;
|
||||||
return Container(
|
return Container(
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 8),
|
margin: const EdgeInsets.symmetric(horizontal: 8),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
@ -19,7 +20,6 @@ class OverlayPop extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Stack(
|
Stack(
|
||||||
@ -27,7 +27,7 @@ class OverlayPop extends StatelessWidget {
|
|||||||
NetworkImgLayer(
|
NetworkImgLayer(
|
||||||
width: imgWidth,
|
width: imgWidth,
|
||||||
height: imgWidth / StyleString.aspectRatio,
|
height: imgWidth / StyleString.aspectRatio,
|
||||||
src: videoItem.pic!,
|
src: videoItem.pic! as String,
|
||||||
quality: 100,
|
quality: 100,
|
||||||
),
|
),
|
||||||
Positioned(
|
Positioned(
|
||||||
@ -61,7 +61,7 @@ class OverlayPop extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Text(
|
child: Text(
|
||||||
videoItem.title!,
|
videoItem.title! as String,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(width: 4),
|
const SizedBox(width: 4),
|
||||||
@ -69,7 +69,10 @@ class OverlayPop extends StatelessWidget {
|
|||||||
tooltip: '保存封面图',
|
tooltip: '保存封面图',
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
await DownloadUtils.downloadImg(
|
await DownloadUtils.downloadImg(
|
||||||
videoItem.pic ?? videoItem.cover);
|
videoItem.pic != null
|
||||||
|
? videoItem.pic as String
|
||||||
|
: videoItem.cover as String,
|
||||||
|
);
|
||||||
// closeFn!();
|
// closeFn!();
|
||||||
},
|
},
|
||||||
icon: const Icon(Icons.download, size: 20),
|
icon: const Icon(Icons.download, size: 20),
|
||||||
|
|||||||
@ -17,8 +17,8 @@ class PullToRefreshHeader extends StatelessWidget {
|
|||||||
this.info,
|
this.info,
|
||||||
this.lastRefreshTime, {
|
this.lastRefreshTime, {
|
||||||
this.color,
|
this.color,
|
||||||
Key? key,
|
super.key,
|
||||||
}) : super(key: key);
|
});
|
||||||
|
|
||||||
final PullToRefreshScrollNotificationInfo? info;
|
final PullToRefreshScrollNotificationInfo? info;
|
||||||
final DateTime? lastRefreshTime;
|
final DateTime? lastRefreshTime;
|
||||||
@ -28,7 +28,7 @@ class PullToRefreshHeader extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final PullToRefreshScrollNotificationInfo? infos = info;
|
final PullToRefreshScrollNotificationInfo? infos = info;
|
||||||
if (infos == null) {
|
if (infos == null) {
|
||||||
return Container();
|
return const SizedBox();
|
||||||
}
|
}
|
||||||
String text = '';
|
String text = '';
|
||||||
if (infos.mode == PullToRefreshIndicatorMode.armed) {
|
if (infos.mode == PullToRefreshIndicatorMode.armed) {
|
||||||
@ -65,7 +65,6 @@ class PullToRefreshHeader extends StatelessWidget {
|
|||||||
top: top,
|
top: top,
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Container(
|
child: Container(
|
||||||
|
|||||||
@ -3,7 +3,7 @@ import 'package:pilipala/utils/utils.dart';
|
|||||||
|
|
||||||
class StatDanMu extends StatelessWidget {
|
class StatDanMu extends StatelessWidget {
|
||||||
final String? theme;
|
final String? theme;
|
||||||
final int? danmu;
|
final dynamic danmu;
|
||||||
final String? size;
|
final String? size;
|
||||||
|
|
||||||
const StatDanMu({Key? key, this.theme, this.danmu, this.size})
|
const StatDanMu({Key? key, this.theme, this.danmu, this.size})
|
||||||
|
|||||||
@ -3,7 +3,7 @@ import 'package:pilipala/utils/utils.dart';
|
|||||||
|
|
||||||
class StatView extends StatelessWidget {
|
class StatView extends StatelessWidget {
|
||||||
final String? theme;
|
final String? theme;
|
||||||
final int? view;
|
final dynamic view;
|
||||||
final String? size;
|
final String? size;
|
||||||
|
|
||||||
const StatView({Key? key, this.theme, this.view, this.size})
|
const StatView({Key? key, this.theme, this.view, this.size})
|
||||||
|
|||||||
@ -1,17 +1,29 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:flutter/material.dart';
|
import '../../http/search.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../../http/user.dart';
|
||||||
import 'package:pilipala/common/widgets/badge.dart';
|
import '../../http/video.dart';
|
||||||
import 'package:pilipala/common/widgets/stat/danmu.dart';
|
import '../../utils/utils.dart';
|
||||||
import 'package:pilipala/common/widgets/stat/view.dart';
|
import '../constants.dart';
|
||||||
import 'package:pilipala/http/search.dart';
|
import 'badge.dart';
|
||||||
import 'package:pilipala/http/user.dart';
|
import 'network_img_layer.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'stat/danmu.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'stat/view.dart';
|
||||||
|
|
||||||
// 视频卡片 - 水平布局
|
// 视频卡片 - 水平布局
|
||||||
class VideoCardH extends StatelessWidget {
|
class VideoCardH extends StatelessWidget {
|
||||||
|
const VideoCardH({
|
||||||
|
super.key,
|
||||||
|
required this.videoItem,
|
||||||
|
this.longPress,
|
||||||
|
this.longPressEnd,
|
||||||
|
this.source = 'normal',
|
||||||
|
this.showOwner = true,
|
||||||
|
this.showView = true,
|
||||||
|
this.showDanmaku = true,
|
||||||
|
this.showPubdate = false,
|
||||||
|
});
|
||||||
// ignore: prefer_typing_uninitialized_variables
|
// ignore: prefer_typing_uninitialized_variables
|
||||||
final videoItem;
|
final videoItem;
|
||||||
final Function()? longPress;
|
final Function()? longPress;
|
||||||
@ -22,23 +34,11 @@ class VideoCardH extends StatelessWidget {
|
|||||||
final bool showDanmaku;
|
final bool showDanmaku;
|
||||||
final bool showPubdate;
|
final bool showPubdate;
|
||||||
|
|
||||||
const VideoCardH({
|
|
||||||
Key? key,
|
|
||||||
required this.videoItem,
|
|
||||||
this.longPress,
|
|
||||||
this.longPressEnd,
|
|
||||||
this.source = 'normal',
|
|
||||||
this.showOwner = true,
|
|
||||||
this.showView = true,
|
|
||||||
this.showDanmaku = true,
|
|
||||||
this.showPubdate = false,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
int aid = videoItem.aid;
|
final int aid = videoItem.aid;
|
||||||
String bvid = videoItem.bvid;
|
final String bvid = videoItem.bvid;
|
||||||
String heroTag = Utils.makeHeroTag(aid);
|
final String heroTag = Utils.makeHeroTag(aid);
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onLongPress: () {
|
onLongPress: () {
|
||||||
if (longPress != null) {
|
if (longPress != null) {
|
||||||
@ -53,7 +53,7 @@ class VideoCardH extends StatelessWidget {
|
|||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
try {
|
try {
|
||||||
int cid =
|
final int cid =
|
||||||
videoItem.cid ?? await SearchHttp.ab2c(aid: aid, bvid: bvid);
|
videoItem.cid ?? await SearchHttp.ab2c(aid: aid, bvid: bvid);
|
||||||
Get.toNamed('/video?bvid=$bvid&cid=$cid',
|
Get.toNamed('/video?bvid=$bvid&cid=$cid',
|
||||||
arguments: {'videoItem': videoItem, 'heroTag': heroTag});
|
arguments: {'videoItem': videoItem, 'heroTag': heroTag});
|
||||||
@ -65,11 +65,11 @@ class VideoCardH extends StatelessWidget {
|
|||||||
padding: const EdgeInsets.fromLTRB(
|
padding: const EdgeInsets.fromLTRB(
|
||||||
StyleString.safeSpace, 5, StyleString.safeSpace, 5),
|
StyleString.safeSpace, 5, StyleString.safeSpace, 5),
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, boxConstraints) {
|
builder: (BuildContext context, BoxConstraints boxConstraints) {
|
||||||
double width = (boxConstraints.maxWidth -
|
final double width = (boxConstraints.maxWidth -
|
||||||
StyleString.cardSpace *
|
StyleString.cardSpace *
|
||||||
6 /
|
6 /
|
||||||
MediaQuery.of(context).textScaleFactor) /
|
MediaQuery.textScalerOf(context).scale(1.0)) /
|
||||||
2;
|
2;
|
||||||
return Container(
|
return Container(
|
||||||
constraints: const BoxConstraints(minHeight: 88),
|
constraints: const BoxConstraints(minHeight: 88),
|
||||||
@ -77,29 +77,28 @@ class VideoCardH extends StatelessWidget {
|
|||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: <Widget>[
|
||||||
AspectRatio(
|
AspectRatio(
|
||||||
aspectRatio: StyleString.aspectRatio,
|
aspectRatio: StyleString.aspectRatio,
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, boxConstraints) {
|
builder: (BuildContext context,
|
||||||
double maxWidth = boxConstraints.maxWidth;
|
BoxConstraints boxConstraints) {
|
||||||
double maxHeight = boxConstraints.maxHeight;
|
final double maxWidth = boxConstraints.maxWidth;
|
||||||
|
final double maxHeight = boxConstraints.maxHeight;
|
||||||
return Stack(
|
return Stack(
|
||||||
children: [
|
children: [
|
||||||
Hero(
|
Hero(
|
||||||
tag: heroTag,
|
tag: heroTag,
|
||||||
child: NetworkImgLayer(
|
child: NetworkImgLayer(
|
||||||
src: videoItem.pic,
|
src: videoItem.pic as String,
|
||||||
width: maxWidth,
|
width: maxWidth,
|
||||||
height: maxHeight,
|
height: maxHeight,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
PBadge(
|
PBadge(
|
||||||
text: Utils.timeFormat(videoItem.duration!),
|
text: Utils.timeFormat(videoItem.duration!),
|
||||||
top: null,
|
|
||||||
right: 6.0,
|
right: 6.0,
|
||||||
bottom: 6.0,
|
bottom: 6.0,
|
||||||
left: null,
|
|
||||||
type: 'gray',
|
type: 'gray',
|
||||||
),
|
),
|
||||||
// if (videoItem.rcmdReason != null &&
|
// if (videoItem.rcmdReason != null &&
|
||||||
@ -159,7 +158,7 @@ class VideoContent extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
if (videoItem.title is String) ...[
|
if (videoItem.title is String) ...[
|
||||||
Text(
|
Text(
|
||||||
videoItem.title,
|
videoItem.title as String,
|
||||||
textAlign: TextAlign.start,
|
textAlign: TextAlign.start,
|
||||||
style: const TextStyle(
|
style: const TextStyle(
|
||||||
fontWeight: FontWeight.w500,
|
fontWeight: FontWeight.w500,
|
||||||
@ -172,9 +171,9 @@ class VideoContent extends StatelessWidget {
|
|||||||
maxLines: 2,
|
maxLines: 2,
|
||||||
text: TextSpan(
|
text: TextSpan(
|
||||||
children: [
|
children: [
|
||||||
for (var i in videoItem.title) ...[
|
for (final i in videoItem.title) ...[
|
||||||
TextSpan(
|
TextSpan(
|
||||||
text: i['text'],
|
text: i['text'] as String,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontWeight: FontWeight.w500,
|
fontWeight: FontWeight.w500,
|
||||||
letterSpacing: 0.3,
|
letterSpacing: 0.3,
|
||||||
@ -216,7 +215,7 @@ class VideoContent extends StatelessWidget {
|
|||||||
Row(
|
Row(
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
videoItem.owner.name,
|
videoItem.owner.name as String,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize:
|
fontSize:
|
||||||
Theme.of(context).textTheme.labelMedium!.fontSize,
|
Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
@ -230,14 +229,14 @@ class VideoContent extends StatelessWidget {
|
|||||||
if (showView) ...[
|
if (showView) ...[
|
||||||
StatView(
|
StatView(
|
||||||
theme: 'gray',
|
theme: 'gray',
|
||||||
view: videoItem.stat.view,
|
view: videoItem.stat.view as int,
|
||||||
),
|
),
|
||||||
const SizedBox(width: 8),
|
const SizedBox(width: 8),
|
||||||
],
|
],
|
||||||
if (showDanmaku)
|
if (showDanmaku)
|
||||||
StatDanMu(
|
StatDanMu(
|
||||||
theme: 'gray',
|
theme: 'gray',
|
||||||
danmu: videoItem.stat.danmaku,
|
danmu: videoItem.stat.danmaku as int,
|
||||||
),
|
),
|
||||||
|
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
@ -267,7 +266,6 @@ class VideoContent extends StatelessWidget {
|
|||||||
height: 24,
|
height: 24,
|
||||||
child: PopupMenuButton<String>(
|
child: PopupMenuButton<String>(
|
||||||
padding: EdgeInsets.zero,
|
padding: EdgeInsets.zero,
|
||||||
tooltip: '稍后再看',
|
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
Icons.more_vert_outlined,
|
Icons.more_vert_outlined,
|
||||||
color: Theme.of(context).colorScheme.outline,
|
color: Theme.of(context).colorScheme.outline,
|
||||||
@ -281,11 +279,11 @@ class VideoContent extends StatelessWidget {
|
|||||||
PopupMenuItem<String>(
|
PopupMenuItem<String>(
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
var res = await UserHttp.toViewLater(
|
var res = await UserHttp.toViewLater(
|
||||||
bvid: videoItem.bvid);
|
bvid: videoItem.bvid as String);
|
||||||
SmartDialog.showToast(res['msg']);
|
SmartDialog.showToast(res['msg']);
|
||||||
},
|
},
|
||||||
value: 'pause',
|
value: 'pause',
|
||||||
height: 35,
|
height: 40,
|
||||||
child: const Row(
|
child: const Row(
|
||||||
children: [
|
children: [
|
||||||
Icon(Icons.watch_later_outlined, size: 16),
|
Icon(Icons.watch_later_outlined, size: 16),
|
||||||
@ -294,6 +292,60 @@ class VideoContent extends StatelessWidget {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
const PopupMenuDivider(),
|
||||||
|
PopupMenuItem<String>(
|
||||||
|
onTap: () async {
|
||||||
|
SmartDialog.show(
|
||||||
|
useSystem: true,
|
||||||
|
animationType:
|
||||||
|
SmartAnimationType.centerFade_otherSlide,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('提示'),
|
||||||
|
content: Text(
|
||||||
|
'确定拉黑:${videoItem.owner.name}(${videoItem.owner.mid})?'
|
||||||
|
'\n\n注:被拉黑的Up可以在隐私设置-黑名单管理中解除'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => SmartDialog.dismiss(),
|
||||||
|
child: Text(
|
||||||
|
'点错了',
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.outline),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
var res = await VideoHttp.relationMod(
|
||||||
|
mid: videoItem.owner.mid,
|
||||||
|
act: 5,
|
||||||
|
reSrc: 11,
|
||||||
|
);
|
||||||
|
SmartDialog.dismiss();
|
||||||
|
SmartDialog.showToast(res['code'] == 0
|
||||||
|
? '成功'
|
||||||
|
: res['msg']);
|
||||||
|
},
|
||||||
|
child: const Text('确认'),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
value: 'pause',
|
||||||
|
height: 40,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const Icon(Icons.block, size: 16),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
Text('拉黑:${videoItem.owner.name}',
|
||||||
|
style: const TextStyle(fontSize: 13))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@ -1,17 +1,19 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:flutter/material.dart';
|
import '../../models/model_rec_video_item.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import 'stat/danmu.dart';
|
||||||
import 'package:pilipala/common/widgets/badge.dart';
|
import 'stat/view.dart';
|
||||||
import 'package:pilipala/common/widgets/stat/danmu.dart';
|
import '../../http/dynamics.dart';
|
||||||
import 'package:pilipala/common/widgets/stat/view.dart';
|
import '../../http/search.dart';
|
||||||
import 'package:pilipala/http/dynamics.dart';
|
import '../../http/user.dart';
|
||||||
import 'package:pilipala/http/search.dart';
|
import '../../http/video.dart';
|
||||||
import 'package:pilipala/http/user.dart';
|
import '../../models/common/search_type.dart';
|
||||||
import 'package:pilipala/models/common/search_type.dart';
|
import '../../utils/id_utils.dart';
|
||||||
import 'package:pilipala/utils/id_utils.dart';
|
import '../../utils/utils.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import '../constants.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'badge.dart';
|
||||||
|
import 'network_img_layer.dart';
|
||||||
|
|
||||||
// 视频卡片 - 垂直布局
|
// 视频卡片 - 垂直布局
|
||||||
class VideoCardV extends StatelessWidget {
|
class VideoCardV extends StatelessWidget {
|
||||||
@ -159,12 +161,12 @@ class VideoCardV extends StatelessWidget {
|
|||||||
height: maxHeight,
|
height: maxHeight,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
if (videoItem.duration != null)
|
if (videoItem.duration > 0)
|
||||||
if (crossAxisCount == 1) ...[
|
if (crossAxisCount == 1) ...[
|
||||||
PBadge(
|
PBadge(
|
||||||
bottom: 10,
|
bottom: 10,
|
||||||
right: 10,
|
right: 10,
|
||||||
text: videoItem.duration,
|
text: Utils.timeFormat(videoItem.duration),
|
||||||
)
|
)
|
||||||
] else ...[
|
] else ...[
|
||||||
PBadge(
|
PBadge(
|
||||||
@ -172,7 +174,7 @@ class VideoCardV extends StatelessWidget {
|
|||||||
right: 7,
|
right: 7,
|
||||||
size: 'small',
|
size: 'small',
|
||||||
type: 'gray',
|
type: 'gray',
|
||||||
text: videoItem.duration,
|
text: Utils.timeFormat(videoItem.duration),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
@ -217,24 +219,17 @@ class VideoContent extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
if (videoItem.goto == 'av' && crossAxisCount == 1) ...[
|
if (videoItem.goto == 'av' && crossAxisCount == 1) ...[
|
||||||
const SizedBox(width: 10),
|
const SizedBox(width: 10),
|
||||||
WatchLater(
|
VideoPopupMenu(
|
||||||
size: 32,
|
size: 32,
|
||||||
iconSize: 18,
|
iconSize: 18,
|
||||||
callFn: () async {
|
videoItem: videoItem,
|
||||||
int aid = videoItem.param;
|
|
||||||
var res =
|
|
||||||
await UserHttp.toViewLater(bvid: IdUtils.av2bv(aid));
|
|
||||||
SmartDialog.showToast(res['msg']);
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
if (crossAxisCount > 1) ...[
|
if (crossAxisCount > 1) ...[
|
||||||
const SizedBox(height: 2),
|
const SizedBox(height: 2),
|
||||||
VideoStat(
|
VideoStat(videoItem: videoItem, crossAxisCount: crossAxisCount),
|
||||||
videoItem: videoItem,
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
if (crossAxisCount == 1) const SizedBox(height: 4),
|
if (crossAxisCount == 1) const SizedBox(height: 4),
|
||||||
Row(
|
Row(
|
||||||
@ -295,21 +290,20 @@ class VideoContent extends StatelessWidget {
|
|||||||
color: Theme.of(context).colorScheme.outline,
|
color: Theme.of(context).colorScheme.outline,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
VideoStat(
|
Expanded(
|
||||||
videoItem: videoItem,
|
flex: 1,
|
||||||
|
child: VideoStat(
|
||||||
|
videoItem: videoItem,
|
||||||
|
crossAxisCount: crossAxisCount,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
const Spacer(),
|
// const Spacer(),
|
||||||
],
|
],
|
||||||
if (videoItem.goto == 'av' && crossAxisCount != 1) ...[
|
if (videoItem.goto == 'av' && crossAxisCount != 1) ...[
|
||||||
WatchLater(
|
VideoPopupMenu(
|
||||||
size: 24,
|
size: 24,
|
||||||
iconSize: 14,
|
iconSize: 14,
|
||||||
callFn: () async {
|
videoItem: videoItem,
|
||||||
int aid = videoItem.param;
|
|
||||||
var res =
|
|
||||||
await UserHttp.toViewLater(bvid: IdUtils.av2bv(aid));
|
|
||||||
SmartDialog.showToast(res['msg']);
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
] else ...[
|
] else ...[
|
||||||
const SizedBox(height: 24)
|
const SizedBox(height: 24)
|
||||||
@ -325,42 +319,55 @@ class VideoContent extends StatelessWidget {
|
|||||||
|
|
||||||
class VideoStat extends StatelessWidget {
|
class VideoStat extends StatelessWidget {
|
||||||
final dynamic videoItem;
|
final dynamic videoItem;
|
||||||
|
final int crossAxisCount;
|
||||||
|
|
||||||
const VideoStat({
|
const VideoStat({
|
||||||
Key? key,
|
Key? key,
|
||||||
required this.videoItem,
|
required this.videoItem,
|
||||||
|
required this.crossAxisCount,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return RichText(
|
return Row(
|
||||||
maxLines: 1,
|
children: [
|
||||||
text: TextSpan(
|
StatView(
|
||||||
style: TextStyle(
|
theme: 'gray',
|
||||||
fontSize: Theme.of(context).textTheme.labelSmall!.fontSize,
|
view: videoItem.stat.view,
|
||||||
color: Theme.of(context).colorScheme.outline,
|
|
||||||
),
|
),
|
||||||
children: [
|
const SizedBox(width: 8),
|
||||||
if (videoItem.stat.view != '-')
|
StatDanMu(
|
||||||
TextSpan(text: '${videoItem.stat.view}观看'),
|
theme: 'gray',
|
||||||
if (videoItem.stat.danmu != '-')
|
danmu: videoItem.stat.danmu,
|
||||||
TextSpan(text: ' • ${videoItem.stat.danmu}弹幕'),
|
),
|
||||||
],
|
if (videoItem is RecVideoItemModel) ...<Widget>[
|
||||||
),
|
crossAxisCount > 1 ? const Spacer() : const SizedBox(width: 8),
|
||||||
|
RichText(
|
||||||
|
maxLines: 1,
|
||||||
|
text: TextSpan(
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: Theme.of(context).textTheme.labelSmall!.fontSize,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
text: Utils.formatTimestampToRelativeTime(videoItem.pubdate)),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 4),
|
||||||
|
]
|
||||||
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class WatchLater extends StatelessWidget {
|
class VideoPopupMenu extends StatelessWidget {
|
||||||
final double? size;
|
final double? size;
|
||||||
final double? iconSize;
|
final double? iconSize;
|
||||||
final Function? callFn;
|
final dynamic videoItem;
|
||||||
|
|
||||||
const WatchLater({
|
const VideoPopupMenu({
|
||||||
Key? key,
|
Key? key,
|
||||||
required this.size,
|
required this.size,
|
||||||
required this.iconSize,
|
required this.iconSize,
|
||||||
this.callFn,
|
required this.videoItem,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -370,7 +377,6 @@ class WatchLater extends StatelessWidget {
|
|||||||
height: size,
|
height: size,
|
||||||
child: PopupMenuButton<String>(
|
child: PopupMenuButton<String>(
|
||||||
padding: EdgeInsets.zero,
|
padding: EdgeInsets.zero,
|
||||||
tooltip: '稍后再看',
|
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
Icons.more_vert_outlined,
|
Icons.more_vert_outlined,
|
||||||
color: Theme.of(context).colorScheme.outline,
|
color: Theme.of(context).colorScheme.outline,
|
||||||
@ -381,9 +387,13 @@ class WatchLater extends StatelessWidget {
|
|||||||
onSelected: (String type) {},
|
onSelected: (String type) {},
|
||||||
itemBuilder: (BuildContext context) => <PopupMenuEntry<String>>[
|
itemBuilder: (BuildContext context) => <PopupMenuEntry<String>>[
|
||||||
PopupMenuItem<String>(
|
PopupMenuItem<String>(
|
||||||
onTap: () => callFn!(),
|
onTap: () async {
|
||||||
|
var res =
|
||||||
|
await UserHttp.toViewLater(bvid: videoItem.bvid as String);
|
||||||
|
SmartDialog.showToast(res['msg']);
|
||||||
|
},
|
||||||
value: 'pause',
|
value: 'pause',
|
||||||
height: 35,
|
height: 40,
|
||||||
child: const Row(
|
child: const Row(
|
||||||
children: [
|
children: [
|
||||||
Icon(Icons.watch_later_outlined, size: 16),
|
Icon(Icons.watch_later_outlined, size: 16),
|
||||||
@ -392,6 +402,55 @@ class WatchLater extends StatelessWidget {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
const PopupMenuDivider(),
|
||||||
|
PopupMenuItem<String>(
|
||||||
|
onTap: () async {
|
||||||
|
SmartDialog.show(
|
||||||
|
useSystem: true,
|
||||||
|
animationType: SmartAnimationType.centerFade_otherSlide,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('提示'),
|
||||||
|
content: Text(
|
||||||
|
'确定拉黑:${videoItem.owner.name}(${videoItem.owner.mid})?'
|
||||||
|
'\n\n注:被拉黑的Up可以在隐私设置-黑名单管理中解除'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => SmartDialog.dismiss(),
|
||||||
|
child: Text(
|
||||||
|
'点错了',
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context).colorScheme.outline),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
var res = await VideoHttp.relationMod(
|
||||||
|
mid: videoItem.owner.mid,
|
||||||
|
act: 5,
|
||||||
|
reSrc: 11,
|
||||||
|
);
|
||||||
|
SmartDialog.dismiss();
|
||||||
|
SmartDialog.showToast(res['msg'] ?? '成功');
|
||||||
|
},
|
||||||
|
child: const Text('确认'),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
value: 'pause',
|
||||||
|
height: 40,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const Icon(Icons.block, size: 16),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
Text('拉黑:${videoItem.owner.name}',
|
||||||
|
style: const TextStyle(fontSize: 13))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@ -185,7 +185,7 @@ class Api {
|
|||||||
static const String searchDefault = '/x/web-interface/wbi/search/default';
|
static const String searchDefault = '/x/web-interface/wbi/search/default';
|
||||||
|
|
||||||
// 搜索关键词
|
// 搜索关键词
|
||||||
static const String serachSuggest =
|
static const String searchSuggest =
|
||||||
'https://s.search.bilibili.com/main/suggest';
|
'https://s.search.bilibili.com/main/suggest';
|
||||||
|
|
||||||
// 分类搜索
|
// 分类搜索
|
||||||
@ -214,6 +214,9 @@ class Api {
|
|||||||
// https://api.bilibili.com/x/relation/tags
|
// https://api.bilibili.com/x/relation/tags
|
||||||
static const String followingsClass = '/x/relation/tags';
|
static const String followingsClass = '/x/relation/tags';
|
||||||
|
|
||||||
|
// 搜索follow
|
||||||
|
static const followSearch = '/x/relation/followings/search';
|
||||||
|
|
||||||
// 粉丝
|
// 粉丝
|
||||||
// vmid 用户id pn 页码 ps 每页个数,最大50 order: desc
|
// vmid 用户id pn 页码 ps 每页个数,最大50 order: desc
|
||||||
// order_type 排序规则 最近访问传空,最常访问传 attention
|
// order_type 排序规则 最近访问传空,最常访问传 attention
|
||||||
@ -230,6 +233,10 @@ class Api {
|
|||||||
static const String liveRoomInfo =
|
static const String liveRoomInfo =
|
||||||
'${HttpString.liveBaseUrl}/xlive/web-room/v2/index/getRoomPlayInfo';
|
'${HttpString.liveBaseUrl}/xlive/web-room/v2/index/getRoomPlayInfo';
|
||||||
|
|
||||||
|
// 直播间详情 H5
|
||||||
|
static const String liveRoomInfoH5 =
|
||||||
|
'${HttpString.liveBaseUrl}/xlive/web-room/v1/index/getH5InfoByRoom';
|
||||||
|
|
||||||
// 用户信息 需要Wbi签名
|
// 用户信息 需要Wbi签名
|
||||||
// https://api.bilibili.com/x/space/wbi/acc/info?mid=503427686&token=&platform=web&web_location=1550101&w_rid=d709892496ce93e3d94d6d37c95bde91&wts=1689301482
|
// https://api.bilibili.com/x/space/wbi/acc/info?mid=503427686&token=&platform=web&web_location=1550101&w_rid=d709892496ce93e3d94d6d37c95bde91&wts=1689301482
|
||||||
static const String memberInfo = '/x/space/wbi/acc/info';
|
static const String memberInfo = '/x/space/wbi/acc/info';
|
||||||
@ -467,4 +474,7 @@ class Api {
|
|||||||
/// page_size
|
/// page_size
|
||||||
static const getSeasonDetailApi =
|
static const getSeasonDetailApi =
|
||||||
'/x/polymer/web-space/seasons_archives_list';
|
'/x/polymer/web-space/seasons_archives_list';
|
||||||
|
|
||||||
|
/// 获取未读动态数
|
||||||
|
static const getUnreadDynamic = '/x/web-interface/dynamic/entrance';
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import 'package:pilipala/http/index.dart';
|
import '../models/bangumi/list.dart';
|
||||||
import 'package:pilipala/models/bangumi/list.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class BangumiHttp {
|
class BangumiHttp {
|
||||||
static Future bangumiList({int? page}) async {
|
static Future bangumiList({int? page}) async {
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import 'package:pilipala/http/index.dart';
|
import '../models/user/black.dart';
|
||||||
import 'package:pilipala/models/user/black.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class BlackHttp {
|
class BlackHttp {
|
||||||
static Future blackList({required int pn, int? ps}) async {
|
static Future blackList({required int pn, int? ps}) async {
|
||||||
|
|||||||
17
lib/http/common.dart
Normal file
17
lib/http/common.dart
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import 'index.dart';
|
||||||
|
|
||||||
|
class CommonHttp {
|
||||||
|
static Future unReadDynamic() async {
|
||||||
|
var res = await Request().get(Api.getUnreadDynamic,
|
||||||
|
data: {'alltype_offset': 0, 'video_offset': '', 'article_offset': 0});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {'status': true, 'data': res.data['data']['dyn_basic_infos']};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'data': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,9 +1,6 @@
|
|||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import '../models/danmaku/dm.pb.dart';
|
||||||
import 'package:pilipala/http/index.dart';
|
import 'index.dart';
|
||||||
import 'package:pilipala/models/danmaku/dm.pb.dart';
|
|
||||||
|
|
||||||
import 'constants.dart';
|
|
||||||
|
|
||||||
class DanmakaHttp {
|
class DanmakaHttp {
|
||||||
// 获取视频弹幕
|
// 获取视频弹幕
|
||||||
@ -24,21 +21,23 @@ class DanmakaHttp {
|
|||||||
);
|
);
|
||||||
return DmSegMobileReply.fromBuffer(response.data);
|
return DmSegMobileReply.fromBuffer(response.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future shootDanmaku({
|
static Future shootDanmaku({
|
||||||
int type = 1,//弹幕类选择(1:视频弹幕 2:漫画弹幕)
|
int type = 1, //弹幕类选择(1:视频弹幕 2:漫画弹幕)
|
||||||
required int oid,// 视频cid
|
required int oid, // 视频cid
|
||||||
required String msg,//弹幕文本(长度小于 100 字符)
|
required String msg, //弹幕文本(长度小于 100 字符)
|
||||||
int mode = 1,// 弹幕类型(1:滚动弹幕 4:底端弹幕 5:顶端弹幕 6:逆向弹幕(不能使用) 7:高级弹幕 8:代码弹幕(不能使用) 9:BAS弹幕(pool必须为2))
|
int mode =
|
||||||
|
1, // 弹幕类型(1:滚动弹幕 4:底端弹幕 5:顶端弹幕 6:逆向弹幕(不能使用) 7:高级弹幕 8:代码弹幕(不能使用) 9:BAS弹幕(pool必须为2))
|
||||||
// String? aid,// 稿件avid
|
// String? aid,// 稿件avid
|
||||||
// String? bvid,// bvid与aid必须有一个
|
// String? bvid,// bvid与aid必须有一个
|
||||||
required String bvid,
|
required String bvid,
|
||||||
int? progress,// 弹幕出现在视频内的时间(单位为毫秒,默认为0)
|
int? progress, // 弹幕出现在视频内的时间(单位为毫秒,默认为0)
|
||||||
int? color,// 弹幕颜色(默认白色,16777215)
|
int? color, // 弹幕颜色(默认白色,16777215)
|
||||||
int? fontsize,// 弹幕字号(默认25)
|
int? fontsize, // 弹幕字号(默认25)
|
||||||
int? pool,// 弹幕池选择(0:普通池 1:字幕池 2:特殊池(代码/BAS弹幕)默认普通池,0)
|
int? pool, // 弹幕池选择(0:普通池 1:字幕池 2:特殊池(代码/BAS弹幕)默认普通池,0)
|
||||||
//int? rnd,// 当前时间戳*1000000(若无此项,则发送弹幕冷却时间限制为90s;若有此项,则发送弹幕冷却时间限制为5s)
|
//int? rnd,// 当前时间戳*1000000(若无此项,则发送弹幕冷却时间限制为90s;若有此项,则发送弹幕冷却时间限制为5s)
|
||||||
int? colorful,//60001:专属渐变彩色(需要会员)
|
int? colorful, //60001:专属渐变彩色(需要会员)
|
||||||
int? checkbox_type,//是否带 UP 身份标识(0:普通;4:带有标识)
|
int? checkbox_type, //是否带 UP 身份标识(0:普通;4:带有标识)
|
||||||
// String? csrf,//CSRF Token(位于 Cookie) Cookie 方式必要
|
// String? csrf,//CSRF Token(位于 Cookie) Cookie 方式必要
|
||||||
// String? access_key,// APP 登录 Token APP 方式必要
|
// String? access_key,// APP 登录 Token APP 方式必要
|
||||||
}) async {
|
}) async {
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import 'package:pilipala/http/index.dart';
|
import '../models/dynamics/result.dart';
|
||||||
import 'package:pilipala/models/dynamics/result.dart';
|
import '../models/dynamics/up.dart';
|
||||||
import 'package:pilipala/models/dynamics/up.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class DynamicsHttp {
|
class DynamicsHttp {
|
||||||
static Future followDynamic({
|
static Future followDynamic({
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import 'package:pilipala/http/index.dart';
|
import '../models/fans/result.dart';
|
||||||
import 'package:pilipala/models/fans/result.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class FanHttp {
|
class FanHttp {
|
||||||
static Future fans({int? vmid, int? pn, int? ps, String? orderType}) async {
|
static Future fans({int? vmid, int? pn, int? ps, String? orderType}) async {
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import 'package:pilipala/http/index.dart';
|
import '../models/follow/result.dart';
|
||||||
import 'package:pilipala/models/follow/result.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class FollowHttp {
|
class FollowHttp {
|
||||||
static Future followings(
|
static Future followings(
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import 'package:html/dom.dart';
|
import 'package:html/dom.dart';
|
||||||
import 'package:html/parser.dart';
|
import 'package:html/parser.dart';
|
||||||
import 'package:pilipala/http/index.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class HtmlHttp {
|
class HtmlHttp {
|
||||||
// article
|
// article
|
||||||
@ -15,7 +15,7 @@ class HtmlHttp {
|
|||||||
Match match = regex.firstMatch(response.data)!;
|
Match match = regex.firstMatch(response.data)!;
|
||||||
String matchedString = match.group(0)!;
|
String matchedString = match.group(0)!;
|
||||||
response = await Request().get(
|
response = await Request().get(
|
||||||
'https:$matchedString' + '/',
|
'https:$matchedString/',
|
||||||
extra: {'ua': 'pc'},
|
extra: {'ua': 'pc'},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,17 +1,18 @@
|
|||||||
// ignore_for_file: avoid_print
|
// ignore_for_file: avoid_print
|
||||||
|
import 'dart:async';
|
||||||
import 'dart:developer';
|
import 'dart:developer';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
import 'dart:async';
|
|
||||||
import 'package:dio/dio.dart';
|
|
||||||
import 'package:cookie_jar/cookie_jar.dart';
|
import 'package:cookie_jar/cookie_jar.dart';
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
import 'package:dio/io.dart';
|
import 'package:dio/io.dart';
|
||||||
import 'package:dio_http2_adapter/dio_http2_adapter.dart';
|
|
||||||
import 'package:hive/hive.dart';
|
|
||||||
import 'package:pilipala/utils/storage.dart';
|
|
||||||
import 'package:pilipala/utils/utils.dart';
|
|
||||||
import 'package:pilipala/http/constants.dart';
|
|
||||||
import 'package:pilipala/http/interceptor.dart';
|
|
||||||
import 'package:dio_cookie_manager/dio_cookie_manager.dart';
|
import 'package:dio_cookie_manager/dio_cookie_manager.dart';
|
||||||
|
// import 'package:dio_http2_adapter/dio_http2_adapter.dart';
|
||||||
|
import 'package:hive/hive.dart';
|
||||||
|
import 'package:pilipala/utils/id_utils.dart';
|
||||||
|
import '../utils/storage.dart';
|
||||||
|
import '../utils/utils.dart';
|
||||||
|
import 'constants.dart';
|
||||||
|
import 'interceptor.dart';
|
||||||
|
|
||||||
class Request {
|
class Request {
|
||||||
static final Request _instance = Request._internal();
|
static final Request _instance = Request._internal();
|
||||||
@ -20,25 +21,25 @@ class Request {
|
|||||||
factory Request() => _instance;
|
factory Request() => _instance;
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
static Box localCache = GStrorage.localCache;
|
static Box localCache = GStrorage.localCache;
|
||||||
late dynamic enableSystemProxy;
|
late bool enableSystemProxy;
|
||||||
late String systemProxyHost;
|
late String systemProxyHost;
|
||||||
late String systemProxyPort;
|
late String systemProxyPort;
|
||||||
|
|
||||||
/// 设置cookie
|
/// 设置cookie
|
||||||
static setCookie() async {
|
static setCookie() async {
|
||||||
Box userInfoCache = GStrorage.userInfo;
|
Box userInfoCache = GStrorage.userInfo;
|
||||||
var cookiePath = await Utils.getCookiePath();
|
final String cookiePath = await Utils.getCookiePath();
|
||||||
var cookieJar = PersistCookieJar(
|
final PersistCookieJar cookieJar = PersistCookieJar(
|
||||||
ignoreExpires: true,
|
ignoreExpires: true,
|
||||||
storage: FileStorage(cookiePath),
|
storage: FileStorage(cookiePath),
|
||||||
);
|
);
|
||||||
cookieManager = CookieManager(cookieJar);
|
cookieManager = CookieManager(cookieJar);
|
||||||
dio.interceptors.add(cookieManager);
|
dio.interceptors.add(cookieManager);
|
||||||
var cookie = await cookieManager.cookieJar
|
final List<Cookie> cookie = await cookieManager.cookieJar
|
||||||
.loadForRequest(Uri.parse(HttpString.baseUrl));
|
.loadForRequest(Uri.parse(HttpString.baseUrl));
|
||||||
var userInfo = userInfoCache.get('userInfoCache');
|
final userInfo = userInfoCache.get('userInfoCache');
|
||||||
if (userInfo != null && userInfo.mid != null) {
|
if (userInfo != null && userInfo.mid != null) {
|
||||||
var cookie2 = await cookieManager.cookieJar
|
final List<Cookie> cookie2 = await cookieManager.cookieJar
|
||||||
.loadForRequest(Uri.parse(HttpString.tUrl));
|
.loadForRequest(Uri.parse(HttpString.tUrl));
|
||||||
if (cookie2.isEmpty) {
|
if (cookie2.isEmpty) {
|
||||||
try {
|
try {
|
||||||
@ -57,14 +58,15 @@ class Request {
|
|||||||
log("setCookie, ${e.toString()}");
|
log("setCookie, ${e.toString()}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var cookieString =
|
final String cookieString = cookie
|
||||||
cookie.map((cookie) => '${cookie.name}=${cookie.value}').join('; ');
|
.map((Cookie cookie) => '${cookie.name}=${cookie.value}')
|
||||||
|
.join('; ');
|
||||||
dio.options.headers['cookie'] = cookieString;
|
dio.options.headers['cookie'] = cookieString;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 从cookie中获取 csrf token
|
// 从cookie中获取 csrf token
|
||||||
static Future<String> getCsrf() async {
|
static Future<String> getCsrf() async {
|
||||||
var cookies = await cookieManager.cookieJar
|
List<Cookie> cookies = await cookieManager.cookieJar
|
||||||
.loadForRequest(Uri.parse(HttpString.apiBaseUrl));
|
.loadForRequest(Uri.parse(HttpString.apiBaseUrl));
|
||||||
String token = '';
|
String token = '';
|
||||||
if (cookies.where((e) => e.name == 'bili_jct').isNotEmpty) {
|
if (cookies.where((e) => e.name == 'bili_jct').isNotEmpty) {
|
||||||
@ -73,13 +75,14 @@ class Request {
|
|||||||
return token;
|
return token;
|
||||||
}
|
}
|
||||||
|
|
||||||
static setOptionsHeaders(userInfo, status) {
|
static setOptionsHeaders(userInfo, bool status) {
|
||||||
if (status) {
|
if (status) {
|
||||||
dio.options.headers['x-bili-mid'] = userInfo.mid.toString();
|
dio.options.headers['x-bili-mid'] = userInfo.mid.toString();
|
||||||
|
dio.options.headers['x-bili-aurora-eid'] =
|
||||||
|
IdUtils.genAuroraEid(userInfo.mid);
|
||||||
}
|
}
|
||||||
dio.options.headers['env'] = 'prod';
|
dio.options.headers['env'] = 'prod';
|
||||||
dio.options.headers['app-key'] = 'android64';
|
dio.options.headers['app-key'] = 'android64';
|
||||||
dio.options.headers['x-bili-aurora-eid'] = 'UlMFQVcABlAH';
|
|
||||||
dio.options.headers['x-bili-aurora-zone'] = 'sh001';
|
dio.options.headers['x-bili-aurora-zone'] = 'sh001';
|
||||||
dio.options.headers['referer'] = 'https://www.bilibili.com/';
|
dio.options.headers['referer'] = 'https://www.bilibili.com/';
|
||||||
}
|
}
|
||||||
@ -100,30 +103,31 @@ class Request {
|
|||||||
headers: {},
|
headers: {},
|
||||||
);
|
);
|
||||||
|
|
||||||
enableSystemProxy =
|
enableSystemProxy = setting.get(SettingBoxKey.enableSystemProxy,
|
||||||
setting.get(SettingBoxKey.enableSystemProxy, defaultValue: false);
|
defaultValue: false) as bool;
|
||||||
systemProxyHost =
|
systemProxyHost =
|
||||||
localCache.get(LocalCacheKey.systemProxyHost, defaultValue: '');
|
localCache.get(LocalCacheKey.systemProxyHost, defaultValue: '');
|
||||||
systemProxyPort =
|
systemProxyPort =
|
||||||
localCache.get(LocalCacheKey.systemProxyPort, defaultValue: '');
|
localCache.get(LocalCacheKey.systemProxyPort, defaultValue: '');
|
||||||
|
|
||||||
dio = Dio(options)
|
dio = Dio(options);
|
||||||
|
|
||||||
/// fix 第三方登录 302重定向 跟iOS代理问题冲突
|
/// fix 第三方登录 302重定向 跟iOS代理问题冲突
|
||||||
..httpClientAdapter = Http2Adapter(
|
// ..httpClientAdapter = Http2Adapter(
|
||||||
ConnectionManager(
|
// ConnectionManager(
|
||||||
idleTimeout: const Duration(milliseconds: 10000),
|
// idleTimeout: const Duration(milliseconds: 10000),
|
||||||
onClientCreate: (_, config) => config.onBadCertificate = (_) => true,
|
// onClientCreate: (_, ClientSetting config) =>
|
||||||
),
|
// config.onBadCertificate = (_) => true,
|
||||||
);
|
// ),
|
||||||
|
// );
|
||||||
|
|
||||||
/// 设置代理
|
/// 设置代理
|
||||||
if (enableSystemProxy) {
|
if (enableSystemProxy) {
|
||||||
dio.httpClientAdapter = IOHttpClientAdapter(
|
dio.httpClientAdapter = IOHttpClientAdapter(
|
||||||
createHttpClient: () {
|
createHttpClient: () {
|
||||||
final client = HttpClient();
|
final HttpClient client = HttpClient();
|
||||||
// Config the client.
|
// Config the client.
|
||||||
client.findProxy = (uri) {
|
client.findProxy = (Uri uri) {
|
||||||
// return 'PROXY host:port';
|
// return 'PROXY host:port';
|
||||||
return 'PROXY $systemProxyHost:$systemProxyPort';
|
return 'PROXY $systemProxyHost:$systemProxyPort';
|
||||||
};
|
};
|
||||||
@ -145,7 +149,7 @@ class Request {
|
|||||||
));
|
));
|
||||||
|
|
||||||
dio.transformer = BackgroundTransformer();
|
dio.transformer = BackgroundTransformer();
|
||||||
dio.options.validateStatus = (status) {
|
dio.options.validateStatus = (int? status) {
|
||||||
return status! >= 200 && status < 300 ||
|
return status! >= 200 && status < 300 ||
|
||||||
HttpString.validateStatusCodes.contains(status);
|
HttpString.validateStatusCodes.contains(status);
|
||||||
};
|
};
|
||||||
@ -156,7 +160,7 @@ class Request {
|
|||||||
*/
|
*/
|
||||||
get(url, {data, options, cancelToken, extra}) async {
|
get(url, {data, options, cancelToken, extra}) async {
|
||||||
Response response;
|
Response response;
|
||||||
Options options = Options();
|
final Options options = Options();
|
||||||
ResponseType resType = ResponseType.json;
|
ResponseType resType = ResponseType.json;
|
||||||
if (extra != null) {
|
if (extra != null) {
|
||||||
resType = extra!['resType'] ?? ResponseType.json;
|
resType = extra!['resType'] ?? ResponseType.json;
|
||||||
@ -175,8 +179,14 @@ class Request {
|
|||||||
);
|
);
|
||||||
return response;
|
return response;
|
||||||
} on DioException catch (e) {
|
} on DioException catch (e) {
|
||||||
print('get error: $e');
|
Response errResponse = Response(
|
||||||
return Future.error(await ApiInterceptor.dioError(e));
|
data: {
|
||||||
|
'message': await ApiInterceptor.dioError(e)
|
||||||
|
}, // 将自定义 Map 数据赋值给 Response 的 data 属性
|
||||||
|
statusCode: 200,
|
||||||
|
requestOptions: RequestOptions(),
|
||||||
|
);
|
||||||
|
return errResponse;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -197,8 +207,14 @@ class Request {
|
|||||||
// print('post success: ${response.data}');
|
// print('post success: ${response.data}');
|
||||||
return response;
|
return response;
|
||||||
} on DioException catch (e) {
|
} on DioException catch (e) {
|
||||||
print('post error: $e');
|
Response errResponse = Response(
|
||||||
return Future.error(await ApiInterceptor.dioError(e));
|
data: {
|
||||||
|
'message': await ApiInterceptor.dioError(e)
|
||||||
|
}, // 将自定义 Map 数据赋值给 Response 的 data 属性
|
||||||
|
statusCode: 200,
|
||||||
|
requestOptions: RequestOptions(),
|
||||||
|
);
|
||||||
|
return errResponse;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -1,11 +1,10 @@
|
|||||||
// ignore_for_file: avoid_print
|
// ignore_for_file: avoid_print
|
||||||
|
|
||||||
import 'package:dio/dio.dart';
|
|
||||||
import 'package:connectivity_plus/connectivity_plus.dart';
|
import 'package:connectivity_plus/connectivity_plus.dart';
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import '../utils/storage.dart';
|
||||||
// import 'package:get/get.dart' hide Response;
|
|
||||||
|
|
||||||
class ApiInterceptor extends Interceptor {
|
class ApiInterceptor extends Interceptor {
|
||||||
@override
|
@override
|
||||||
@ -21,16 +20,16 @@ class ApiInterceptor extends Interceptor {
|
|||||||
void onResponse(Response response, ResponseInterceptorHandler handler) {
|
void onResponse(Response response, ResponseInterceptorHandler handler) {
|
||||||
try {
|
try {
|
||||||
if (response.statusCode == 302) {
|
if (response.statusCode == 302) {
|
||||||
List<String> locations = response.headers['location']!;
|
final List<String> locations = response.headers['location']!;
|
||||||
if (locations.isNotEmpty) {
|
if (locations.isNotEmpty) {
|
||||||
if (locations.first.startsWith('https://www.mcbbs.net')) {
|
if (locations.first.startsWith('https://www.mcbbs.net')) {
|
||||||
final uri = Uri.parse(locations.first);
|
final Uri uri = Uri.parse(locations.first);
|
||||||
final accessKey = uri.queryParameters['access_key'];
|
final String? accessKey = uri.queryParameters['access_key'];
|
||||||
final mid = uri.queryParameters['mid'];
|
final String? mid = uri.queryParameters['mid'];
|
||||||
try {
|
try {
|
||||||
Box localCache = GStrorage.localCache;
|
Box localCache = GStrorage.localCache;
|
||||||
localCache.put(
|
localCache.put(LocalCacheKey.accessKey,
|
||||||
LocalCacheKey.accessKey, {'mid': mid, 'value': accessKey});
|
<String, String?>{'mid': mid, 'value': accessKey});
|
||||||
} catch (_) {}
|
} catch (_) {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -53,47 +52,46 @@ class ApiInterceptor extends Interceptor {
|
|||||||
super.onError(err, handler);
|
super.onError(err, handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future dioError(DioException error) async {
|
static Future<String> dioError(DioException error) async {
|
||||||
switch (error.type) {
|
switch (error.type) {
|
||||||
case DioExceptionType.badCertificate:
|
case DioExceptionType.badCertificate:
|
||||||
return '证书有误!';
|
return '证书有误!';
|
||||||
case DioExceptionType.badResponse:
|
case DioExceptionType.badResponse:
|
||||||
return '服务器异常,请稍后重试!';
|
return '服务器异常,请稍后重试!';
|
||||||
case DioExceptionType.cancel:
|
case DioExceptionType.cancel:
|
||||||
return "请求已被取消,请重新请求";
|
return '请求已被取消,请重新请求';
|
||||||
case DioExceptionType.connectionError:
|
case DioExceptionType.connectionError:
|
||||||
return '连接错误,请检查网络设置';
|
return '连接错误,请检查网络设置';
|
||||||
case DioExceptionType.connectionTimeout:
|
case DioExceptionType.connectionTimeout:
|
||||||
return "网络连接超时,请检查网络设置";
|
return '网络连接超时,请检查网络设置';
|
||||||
case DioExceptionType.receiveTimeout:
|
case DioExceptionType.receiveTimeout:
|
||||||
return "响应超时,请稍后重试!";
|
return '响应超时,请稍后重试!';
|
||||||
case DioExceptionType.sendTimeout:
|
case DioExceptionType.sendTimeout:
|
||||||
return "发送请求超时,请检查网络设置";
|
return '发送请求超时,请检查网络设置';
|
||||||
case DioExceptionType.unknown:
|
case DioExceptionType.unknown:
|
||||||
var res = await checkConect();
|
final String res = await checkConnect();
|
||||||
return res + " \n 网络异常,请稍后重试!";
|
return '$res,网络异常!';
|
||||||
default:
|
|
||||||
return "Dio异常";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future<dynamic> checkConect() async {
|
static Future<String> checkConnect() async {
|
||||||
final connectivityResult = await (Connectivity().checkConnectivity());
|
final ConnectivityResult connectivityResult =
|
||||||
if (connectivityResult == ConnectivityResult.mobile) {
|
await Connectivity().checkConnectivity();
|
||||||
return 'connected with mobile network';
|
switch (connectivityResult) {
|
||||||
} else if (connectivityResult == ConnectivityResult.wifi) {
|
case ConnectivityResult.mobile:
|
||||||
return 'connected with wifi network';
|
return '正在使用移动流量';
|
||||||
} else if (connectivityResult == ConnectivityResult.ethernet) {
|
case ConnectivityResult.wifi:
|
||||||
// I am connected to a ethernet network.
|
return '正在使用wifi';
|
||||||
} else if (connectivityResult == ConnectivityResult.vpn) {
|
case ConnectivityResult.ethernet:
|
||||||
// I am connected to a vpn network.
|
return '正在使用局域网';
|
||||||
// Note for iOS and macOS:
|
case ConnectivityResult.vpn:
|
||||||
// There is no separate network interface type for [vpn].
|
return '正在使用代理网络';
|
||||||
// It returns [other] on any device (also simulator)
|
case ConnectivityResult.other:
|
||||||
} else if (connectivityResult == ConnectivityResult.other) {
|
return '正在使用其他网络';
|
||||||
// I am connected to a network which is not in the above mentioned networks.
|
case ConnectivityResult.none:
|
||||||
} else if (connectivityResult == ConnectivityResult.none) {
|
return '未连接到任何网络';
|
||||||
return 'not connected to any network';
|
default:
|
||||||
|
return '';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,7 +1,8 @@
|
|||||||
import 'package:pilipala/http/api.dart';
|
import '../models/live/item.dart';
|
||||||
import 'package:pilipala/http/init.dart';
|
import '../models/live/room_info.dart';
|
||||||
import 'package:pilipala/models/live/item.dart';
|
import '../models/live/room_info_h5.dart';
|
||||||
import 'package:pilipala/models/live/room_info.dart';
|
import 'api.dart';
|
||||||
|
import 'init.dart';
|
||||||
|
|
||||||
class LiveHttp {
|
class LiveHttp {
|
||||||
static Future liveList(
|
static Future liveList(
|
||||||
@ -46,4 +47,22 @@ class LiveHttp {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Future liveRoomInfoH5({roomId, qn}) async {
|
||||||
|
var res = await Request().get(Api.liveRoomInfoH5, data: {
|
||||||
|
'room_id': roomId,
|
||||||
|
});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {
|
||||||
|
'status': true,
|
||||||
|
'data': RoomInfoH5Model.fromJson(res.data['data'])
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'data': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,13 +1,12 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
import 'dart:math';
|
import 'dart:math';
|
||||||
import 'package:crypto/crypto.dart';
|
import 'package:crypto/crypto.dart';
|
||||||
|
|
||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:encrypt/encrypt.dart';
|
import 'package:encrypt/encrypt.dart';
|
||||||
import 'package:pilipala/http/index.dart';
|
|
||||||
import 'package:pilipala/models/login/index.dart';
|
|
||||||
import 'package:pilipala/utils/login.dart';
|
|
||||||
import 'package:uuid/uuid.dart';
|
import 'package:uuid/uuid.dart';
|
||||||
|
import '../models/login/index.dart';
|
||||||
|
import '../utils/login.dart';
|
||||||
|
import 'index.dart';
|
||||||
|
|
||||||
class LoginHttp {
|
class LoginHttp {
|
||||||
static Future queryCaptcha() async {
|
static Future queryCaptcha() async {
|
||||||
|
|||||||
@ -1,17 +1,17 @@
|
|||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../common/constants.dart';
|
||||||
import 'package:pilipala/http/index.dart';
|
import '../models/dynamics/result.dart';
|
||||||
import 'package:pilipala/models/dynamics/result.dart';
|
import '../models/follow/result.dart';
|
||||||
import 'package:pilipala/models/follow/result.dart';
|
import '../models/member/archive.dart';
|
||||||
import 'package:pilipala/models/member/archive.dart';
|
import '../models/member/coin.dart';
|
||||||
import 'package:pilipala/models/member/coin.dart';
|
import '../models/member/info.dart';
|
||||||
import 'package:pilipala/models/member/info.dart';
|
import '../models/member/seasons.dart';
|
||||||
import 'package:pilipala/models/member/seasons.dart';
|
import '../models/member/tags.dart';
|
||||||
import 'package:pilipala/models/member/tags.dart';
|
import '../utils/storage.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import '../utils/utils.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import '../utils/wbi_sign.dart';
|
||||||
import 'package:pilipala/utils/wbi_sign.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class MemberHttp {
|
class MemberHttp {
|
||||||
static Future memberInfo({
|
static Future memberInfo({
|
||||||
@ -461,4 +461,41 @@ class MemberHttp {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 搜索follow
|
||||||
|
static Future getfollowSearch({
|
||||||
|
required int mid,
|
||||||
|
required int ps,
|
||||||
|
required int pn,
|
||||||
|
required String name,
|
||||||
|
}) async {
|
||||||
|
Map<String, dynamic> data = {
|
||||||
|
'vmid': mid,
|
||||||
|
'pn': pn,
|
||||||
|
'ps': ps,
|
||||||
|
'order': 'desc',
|
||||||
|
'order_type': 'attention',
|
||||||
|
'gaia_source': 'main_web',
|
||||||
|
'name': name,
|
||||||
|
'web_location': 333.999,
|
||||||
|
};
|
||||||
|
Map params = await WbiSign().makSign(data);
|
||||||
|
var res = await Request().get(Api.followSearch, data: {
|
||||||
|
...data,
|
||||||
|
'w_rid': params['w_rid'],
|
||||||
|
'wts': params['wts'],
|
||||||
|
});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {
|
||||||
|
'status': true,
|
||||||
|
'data': FollowDataModel.fromJson(res.data['data'])
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'data': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,8 +1,8 @@
|
|||||||
import 'package:pilipala/http/api.dart';
|
import '../models/msg/account.dart';
|
||||||
import 'package:pilipala/http/init.dart';
|
import '../models/msg/session.dart';
|
||||||
import 'package:pilipala/models/msg/account.dart';
|
import '../utils/wbi_sign.dart';
|
||||||
import 'package:pilipala/models/msg/session.dart';
|
import 'api.dart';
|
||||||
import 'package:pilipala/utils/wbi_sign.dart';
|
import 'init.dart';
|
||||||
|
|
||||||
class MsgHttp {
|
class MsgHttp {
|
||||||
// 会话列表
|
// 会话列表
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import 'package:pilipala/http/api.dart';
|
import '../models/video/reply/data.dart';
|
||||||
import 'package:pilipala/http/init.dart';
|
import 'api.dart';
|
||||||
import 'package:pilipala/models/video/reply/data.dart';
|
import 'init.dart';
|
||||||
|
|
||||||
class ReplyHttp {
|
class ReplyHttp {
|
||||||
static Future replyList({
|
static Future replyList({
|
||||||
|
|||||||
@ -1,13 +1,12 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
|
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/http/index.dart';
|
import '../models/bangumi/info.dart';
|
||||||
import 'package:pilipala/models/bangumi/info.dart';
|
import '../models/common/search_type.dart';
|
||||||
import 'package:pilipala/models/common/search_type.dart';
|
import '../models/search/hot.dart';
|
||||||
import 'package:pilipala/models/search/hot.dart';
|
import '../models/search/result.dart';
|
||||||
import 'package:pilipala/models/search/result.dart';
|
import '../models/search/suggest.dart';
|
||||||
import 'package:pilipala/models/search/suggest.dart';
|
import '../utils/storage.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'index.dart';
|
||||||
|
|
||||||
class SearchHttp {
|
class SearchHttp {
|
||||||
static Box setting = GStrorage.setting;
|
static Box setting = GStrorage.setting;
|
||||||
@ -37,7 +36,7 @@ class SearchHttp {
|
|||||||
|
|
||||||
// 获取搜索建议
|
// 获取搜索建议
|
||||||
static Future searchSuggest({required term}) async {
|
static Future searchSuggest({required term}) async {
|
||||||
var res = await Request().get(Api.serachSuggest,
|
var res = await Request().get(Api.searchSuggest,
|
||||||
data: {'term': term, 'main_ver': 'v1', 'highlight': term});
|
data: {'term': term, 'main_ver': 'v1', 'highlight': term});
|
||||||
if (res.data is String) {
|
if (res.data is String) {
|
||||||
Map<String, dynamic> resultMap = json.decode(res.data);
|
Map<String, dynamic> resultMap = json.decode(res.data);
|
||||||
@ -129,25 +128,28 @@ class SearchHttp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future ab2c({int? aid, String? bvid}) async {
|
static Future<int> ab2c({int? aid, String? bvid}) async {
|
||||||
Map<String, dynamic> data = {};
|
Map<String, dynamic> data = {};
|
||||||
if (aid != null) {
|
if (aid != null) {
|
||||||
data['aid'] = aid;
|
data['aid'] = aid;
|
||||||
} else if (bvid != null) {
|
} else if (bvid != null) {
|
||||||
data['bvid'] = bvid;
|
data['bvid'] = bvid;
|
||||||
}
|
}
|
||||||
var res = await Request().get(Api.ab2c, data: {...data});
|
final dynamic res =
|
||||||
|
await Request().get(Api.ab2c, data: <String, dynamic>{...data});
|
||||||
return res.data['data'].first['cid'];
|
return res.data['data'].first['cid'];
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future bangumiInfo({int? seasonId, int? epId}) async {
|
static Future<Map<String, dynamic>> bangumiInfo(
|
||||||
Map<String, dynamic> data = {};
|
{int? seasonId, int? epId}) async {
|
||||||
|
final Map<String, dynamic> data = {};
|
||||||
if (seasonId != null) {
|
if (seasonId != null) {
|
||||||
data['season_id'] = seasonId;
|
data['season_id'] = seasonId;
|
||||||
} else if (epId != null) {
|
} else if (epId != null) {
|
||||||
data['ep_id'] = epId;
|
data['ep_id'] = epId;
|
||||||
}
|
}
|
||||||
var res = await Request().get(Api.bangumiInfo, data: {...data});
|
final dynamic res =
|
||||||
|
await Request().get(Api.bangumiInfo, data: <String, dynamic>{...data});
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {
|
return {
|
||||||
'status': true,
|
'status': true,
|
||||||
|
|||||||
@ -1,14 +1,13 @@
|
|||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../common/constants.dart';
|
||||||
import 'package:pilipala/http/api.dart';
|
import '../models/model_hot_video_item.dart';
|
||||||
import 'package:pilipala/http/init.dart';
|
import '../models/user/fav_detail.dart';
|
||||||
import 'package:pilipala/models/model_hot_video_item.dart';
|
import '../models/user/fav_folder.dart';
|
||||||
import 'package:pilipala/models/user/fav_detail.dart';
|
import '../models/user/history.dart';
|
||||||
import 'package:pilipala/models/user/fav_folder.dart';
|
import '../models/user/info.dart';
|
||||||
import 'package:pilipala/models/user/history.dart';
|
import '../models/user/stat.dart';
|
||||||
import 'package:pilipala/models/user/info.dart';
|
import 'api.dart';
|
||||||
import 'package:pilipala/models/user/stat.dart';
|
import 'init.dart';
|
||||||
import 'package:pilipala/utils/wbi_sign.dart';
|
|
||||||
|
|
||||||
class UserHttp {
|
class UserHttp {
|
||||||
static Future<dynamic> userStat({required int mid}) async {
|
static Future<dynamic> userStat({required int mid}) async {
|
||||||
|
|||||||
@ -1,19 +1,19 @@
|
|||||||
import 'dart:developer';
|
import 'dart:developer';
|
||||||
|
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import '../common/constants.dart';
|
||||||
import 'package:pilipala/http/api.dart';
|
import '../models/common/reply_type.dart';
|
||||||
import 'package:pilipala/http/init.dart';
|
import '../models/home/rcmd/result.dart';
|
||||||
import 'package:pilipala/models/common/reply_type.dart';
|
import '../models/model_hot_video_item.dart';
|
||||||
import 'package:pilipala/models/home/rcmd/result.dart';
|
import '../models/model_rec_video_item.dart';
|
||||||
import 'package:pilipala/models/model_hot_video_item.dart';
|
import '../models/user/fav_folder.dart';
|
||||||
import 'package:pilipala/models/model_rec_video_item.dart';
|
import '../models/video/ai.dart';
|
||||||
import 'package:pilipala/models/user/fav_folder.dart';
|
import '../models/video/play/url.dart';
|
||||||
import 'package:pilipala/models/video/ai.dart';
|
import '../models/video_detail_res.dart';
|
||||||
import 'package:pilipala/models/video/play/url.dart';
|
import '../utils/recommend_filter.dart';
|
||||||
import 'package:pilipala/models/video_detail_res.dart';
|
import '../utils/storage.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import '../utils/wbi_sign.dart';
|
||||||
import 'package:pilipala/utils/wbi_sign.dart';
|
import 'api.dart';
|
||||||
|
import 'init.dart';
|
||||||
|
|
||||||
/// res.data['code'] == 0 请求正常返回结果
|
/// res.data['code'] == 0 请求正常返回结果
|
||||||
/// res.data['data'] 为结果
|
/// res.data['data'] 为结果
|
||||||
@ -33,27 +33,41 @@ class VideoHttp {
|
|||||||
Api.recommendListWeb,
|
Api.recommendListWeb,
|
||||||
data: {
|
data: {
|
||||||
'version': 1,
|
'version': 1,
|
||||||
'feed_version': 'V3',
|
'feed_version': 'V8',
|
||||||
|
'homepage_ver': 1,
|
||||||
'ps': ps,
|
'ps': ps,
|
||||||
'fresh_idx': freshIdx,
|
'fresh_idx': freshIdx,
|
||||||
'fresh_type': 999999
|
'brush': freshIdx,
|
||||||
|
'fresh_type': 4
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
List<RecVideoItemModel> list = [];
|
List<RecVideoItemModel> list = [];
|
||||||
|
List<int> blackMidsList =
|
||||||
|
setting.get(SettingBoxKey.blackMidsList, defaultValue: [-1]);
|
||||||
for (var i in res.data['data']['item']) {
|
for (var i in res.data['data']['item']) {
|
||||||
list.add(RecVideoItemModel.fromJson(i));
|
//过滤掉live与ad,以及拉黑用户
|
||||||
|
if (i['goto'] == 'av' &&
|
||||||
|
(i['owner'] != null &&
|
||||||
|
!blackMidsList.contains(i['owner']['mid']))) {
|
||||||
|
RecVideoItemModel videoItem = RecVideoItemModel.fromJson(i);
|
||||||
|
if (!RecommendFilter.filter(videoItem)) {
|
||||||
|
list.add(videoItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return {'status': true, 'data': list};
|
return {'status': true, 'data': list};
|
||||||
} else {
|
} else {
|
||||||
return {'status': false, 'data': [], 'msg': ''};
|
return {'status': false, 'data': [], 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return {'status': false, 'data': [], 'msg': err.toString()};
|
return {'status': false, 'data': [], 'msg': err.toString()};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future rcmdVideoListApp({int? ps, required int freshIdx}) async {
|
// 添加额外的loginState变量模拟未登录状态
|
||||||
|
static Future rcmdVideoListApp(
|
||||||
|
{bool loginStatus = true, required int freshIdx}) async {
|
||||||
try {
|
try {
|
||||||
var res = await Request().get(
|
var res = await Request().get(
|
||||||
Api.recommendListApp,
|
Api.recommendListApp,
|
||||||
@ -66,9 +80,11 @@ class VideoHttp {
|
|||||||
'device_name': 'vivo',
|
'device_name': 'vivo',
|
||||||
'pull': freshIdx == 0 ? 'true' : 'false',
|
'pull': freshIdx == 0 ? 'true' : 'false',
|
||||||
'appkey': Constants.appKey,
|
'appkey': Constants.appKey,
|
||||||
'access_key': localCache
|
'access_key': loginStatus
|
||||||
.get(LocalCacheKey.accessKey, defaultValue: {})['value'] ??
|
? (localCache.get(LocalCacheKey.accessKey,
|
||||||
''
|
defaultValue: {})['value'] ??
|
||||||
|
'')
|
||||||
|
: ''
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
@ -81,12 +97,15 @@ class VideoHttp {
|
|||||||
(!enableRcmdDynamic ? i['card_goto'] != 'picture' : true) &&
|
(!enableRcmdDynamic ? i['card_goto'] != 'picture' : true) &&
|
||||||
(i['args'] != null &&
|
(i['args'] != null &&
|
||||||
!blackMidsList.contains(i['args']['up_mid']))) {
|
!blackMidsList.contains(i['args']['up_mid']))) {
|
||||||
list.add(RecVideoItemAppModel.fromJson(i));
|
RecVideoItemAppModel videoItem = RecVideoItemAppModel.fromJson(i);
|
||||||
|
if (!RecommendFilter.filter(videoItem)) {
|
||||||
|
list.add(videoItem);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return {'status': true, 'data': list};
|
return {'status': true, 'data': list};
|
||||||
} else {
|
} else {
|
||||||
return {'status': false, 'data': [], 'msg': ''};
|
return {'status': false, 'data': [], 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return {'status': false, 'data': [], 'msg': err.toString()};
|
return {'status': false, 'data': [], 'msg': err.toString()};
|
||||||
@ -111,7 +130,7 @@ class VideoHttp {
|
|||||||
}
|
}
|
||||||
return {'status': true, 'data': list};
|
return {'status': true, 'data': list};
|
||||||
} else {
|
} else {
|
||||||
return {'status': false, 'data': []};
|
return {'status': false, 'data': [], 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return {'status': false, 'data': [], 'msg': err};
|
return {'status': false, 'data': [], 'msg': err};
|
||||||
@ -134,6 +153,12 @@ class VideoHttp {
|
|||||||
data['bvid'] = bvid;
|
data['bvid'] = bvid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 免登录查看1080p
|
||||||
|
if (userInfoCache.get('userInfoCache') == null &&
|
||||||
|
setting.get(SettingBoxKey.p1080, defaultValue: true)) {
|
||||||
|
data['try_look'] = 1;
|
||||||
|
}
|
||||||
|
|
||||||
Map params = await WbiSign().makSign({
|
Map params = await WbiSign().makSign({
|
||||||
...data,
|
...data,
|
||||||
'fourk': 1,
|
'fourk': 1,
|
||||||
@ -142,11 +167,6 @@ class VideoHttp {
|
|||||||
'web_location': 1550101,
|
'web_location': 1550101,
|
||||||
});
|
});
|
||||||
|
|
||||||
// 免登录查看1080p
|
|
||||||
if (userInfoCache.get('userInfoCache') == null &&
|
|
||||||
setting.get(SettingBoxKey.p1080, defaultValue: true)) {
|
|
||||||
data['try_look'] = 1;
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
var res = await Request().get(Api.videoUrl, data: params);
|
var res = await Request().get(Api.videoUrl, data: params);
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
@ -196,7 +216,10 @@ class VideoHttp {
|
|||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
List<HotVideoItemModel> list = [];
|
List<HotVideoItemModel> list = [];
|
||||||
for (var i in res.data['data']) {
|
for (var i in res.data['data']) {
|
||||||
list.add(HotVideoItemModel.fromJson(i));
|
HotVideoItemModel videoItem = HotVideoItemModel.fromJson(i);
|
||||||
|
if (!RecommendFilter.filter(videoItem, relatedVideos: true)) {
|
||||||
|
list.add(videoItem);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return {'status': true, 'data': list};
|
return {'status': true, 'data': list};
|
||||||
} else {
|
} else {
|
||||||
@ -217,10 +240,11 @@ class VideoHttp {
|
|||||||
// 获取投币状态
|
// 获取投币状态
|
||||||
static Future hasCoinVideo({required String bvid}) async {
|
static Future hasCoinVideo({required String bvid}) async {
|
||||||
var res = await Request().get(Api.hasCoinVideo, data: {'bvid': bvid});
|
var res = await Request().get(Api.hasCoinVideo, data: {'bvid': bvid});
|
||||||
|
print('res: $res');
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
} else {
|
} else {
|
||||||
return {'status': true, 'data': []};
|
return {'status': false, 'data': []};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -298,7 +322,7 @@ class VideoHttp {
|
|||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
} else {
|
} else {
|
||||||
return {'status': false, 'data': []};
|
return {'status': false, 'data': [], 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -354,7 +378,7 @@ class VideoHttp {
|
|||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
} else {
|
} else {
|
||||||
return {'status': true, 'data': []};
|
return {'status': false, 'data': []};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -370,7 +394,7 @@ class VideoHttp {
|
|||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
} else {
|
} else {
|
||||||
return {'status': true, 'data': []};
|
return {'status': false, 'data': []};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -426,6 +450,8 @@ class VideoHttp {
|
|||||||
});
|
});
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
|
} else {
|
||||||
|
return {'status': false, 'data': null, 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -440,11 +466,13 @@ class VideoHttp {
|
|||||||
'up_mid': upMid,
|
'up_mid': upMid,
|
||||||
});
|
});
|
||||||
var res = await Request().get(Api.aiConclusion, data: params);
|
var res = await Request().get(Api.aiConclusion, data: params);
|
||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0 && res.data['data']['code'] == 0) {
|
||||||
return {
|
return {
|
||||||
'status': true,
|
'status': true,
|
||||||
'data': AiConclusionModel.fromJson(res.data['data']),
|
'data': AiConclusionModel.fromJson(res.data['data']),
|
||||||
};
|
};
|
||||||
|
} else {
|
||||||
|
return {'status': false, 'data': []};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -21,6 +21,9 @@ import 'package:pilipala/utils/app_scheme.dart';
|
|||||||
import 'package:pilipala/utils/data.dart';
|
import 'package:pilipala/utils/data.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
import 'package:media_kit/media_kit.dart'; // Provides [Player], [Media], [Playlist] etc.
|
import 'package:media_kit/media_kit.dart'; // Provides [Player], [Media], [Playlist] etc.
|
||||||
|
import 'package:pilipala/utils/recommend_filter.dart';
|
||||||
|
import 'package:catcher_2/catcher_2.dart';
|
||||||
|
import './services/loggeer.dart';
|
||||||
|
|
||||||
void main() async {
|
void main() async {
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
@ -30,7 +33,35 @@ void main() async {
|
|||||||
.then((_) async {
|
.then((_) async {
|
||||||
await GStrorage.init();
|
await GStrorage.init();
|
||||||
await setupServiceLocator();
|
await setupServiceLocator();
|
||||||
runApp(const MyApp());
|
Request();
|
||||||
|
await Request.setCookie();
|
||||||
|
RecommendFilter();
|
||||||
|
|
||||||
|
// 异常捕获 logo记录
|
||||||
|
final Catcher2Options debugConfig = Catcher2Options(
|
||||||
|
SilentReportMode(),
|
||||||
|
[
|
||||||
|
FileHandler(await getLogsPath()),
|
||||||
|
ConsoleHandler(
|
||||||
|
enableDeviceParameters: false,
|
||||||
|
enableApplicationParameters: false,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
final Catcher2Options releaseConfig = Catcher2Options(
|
||||||
|
SilentReportMode(),
|
||||||
|
[FileHandler(await getLogsPath())],
|
||||||
|
);
|
||||||
|
|
||||||
|
Catcher2(
|
||||||
|
debugConfig: debugConfig,
|
||||||
|
releaseConfig: releaseConfig,
|
||||||
|
runAppFunction: () {
|
||||||
|
runApp(const MyApp());
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
// 小白条、导航栏沉浸
|
// 小白条、导航栏沉浸
|
||||||
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
||||||
SystemChrome.setSystemUIOverlayStyle(const SystemUiOverlayStyle(
|
SystemChrome.setSystemUIOverlayStyle(const SystemUiOverlayStyle(
|
||||||
@ -38,9 +69,7 @@ void main() async {
|
|||||||
systemNavigationBarDividerColor: Colors.transparent,
|
systemNavigationBarDividerColor: Colors.transparent,
|
||||||
statusBarColor: Colors.transparent,
|
statusBarColor: Colors.transparent,
|
||||||
));
|
));
|
||||||
await Request.setCookie();
|
|
||||||
Data.init();
|
Data.init();
|
||||||
GStrorage.lazyInit();
|
|
||||||
PiliSchame.init();
|
PiliSchame.init();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -112,6 +141,13 @@ class MyApp extends StatelessWidget {
|
|||||||
? darkColorScheme
|
? darkColorScheme
|
||||||
: lightColorScheme,
|
: lightColorScheme,
|
||||||
useMaterial3: true,
|
useMaterial3: true,
|
||||||
|
snackBarTheme: SnackBarThemeData(
|
||||||
|
actionTextColor: lightColorScheme.primary,
|
||||||
|
backgroundColor: lightColorScheme.secondaryContainer,
|
||||||
|
closeIconColor: lightColorScheme.secondary,
|
||||||
|
contentTextStyle: TextStyle(color: lightColorScheme.secondary),
|
||||||
|
elevation: 20,
|
||||||
|
),
|
||||||
pageTransitionsTheme: const PageTransitionsTheme(
|
pageTransitionsTheme: const PageTransitionsTheme(
|
||||||
builders: <TargetPlatform, PageTransitionsBuilder>{
|
builders: <TargetPlatform, PageTransitionsBuilder>{
|
||||||
TargetPlatform.android: ZoomPageTransitionsBuilder(
|
TargetPlatform.android: ZoomPageTransitionsBuilder(
|
||||||
@ -126,10 +162,11 @@ class MyApp extends StatelessWidget {
|
|||||||
? lightColorScheme
|
? lightColorScheme
|
||||||
: darkColorScheme,
|
: darkColorScheme,
|
||||||
useMaterial3: true,
|
useMaterial3: true,
|
||||||
snackBarTheme: const SnackBarThemeData(
|
snackBarTheme: SnackBarThemeData(
|
||||||
actionTextColor: Colors.white,
|
actionTextColor: darkColorScheme.primary,
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: darkColorScheme.secondaryContainer,
|
||||||
contentTextStyle: TextStyle(color: Colors.white),
|
closeIconColor: darkColorScheme.secondary,
|
||||||
|
contentTextStyle: TextStyle(color: darkColorScheme.secondary),
|
||||||
elevation: 20,
|
elevation: 20,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -147,9 +184,8 @@ class MyApp extends StatelessWidget {
|
|||||||
return FlutterSmartDialog(
|
return FlutterSmartDialog(
|
||||||
toastBuilder: (String msg) => CustomToast(msg: msg),
|
toastBuilder: (String msg) => CustomToast(msg: msg),
|
||||||
child: MediaQuery(
|
child: MediaQuery(
|
||||||
data: MediaQuery.of(context).copyWith(
|
data: MediaQuery.of(context)
|
||||||
textScaleFactor:
|
.copyWith(textScaler: TextScaler.linear(textScale)),
|
||||||
MediaQuery.of(context).textScaleFactor * textScale),
|
|
||||||
child: child!,
|
child: child!,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
9
lib/models/common/dynamic_badge_mode.dart
Normal file
9
lib/models/common/dynamic_badge_mode.dart
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
enum DynamicBadgeMode { hidden, point, number }
|
||||||
|
|
||||||
|
extension DynamicBadgeModeDesc on DynamicBadgeMode {
|
||||||
|
String get description => ['隐藏', '红点', '数字'][index];
|
||||||
|
}
|
||||||
|
|
||||||
|
extension DynamicBadgeModeCode on DynamicBadgeMode {
|
||||||
|
int get code => [0, 1, 2][index];
|
||||||
|
}
|
||||||
@ -1,7 +1,7 @@
|
|||||||
// 首页推荐类型
|
// 首页推荐类型
|
||||||
enum RcmdType { web, app }
|
enum RcmdType { web, app, notLogin }
|
||||||
|
|
||||||
extension RcmdTypeExtension on RcmdType {
|
extension RcmdTypeExtension on RcmdType {
|
||||||
String get values => ['web', 'app'][index];
|
String get values => ['web', 'app', 'notLogin'][index];
|
||||||
String get labels => ['web端', 'app端'][index];
|
String get labels => ['web端', 'app端', '游客模式'][index];
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
enum ReplySortType { time, like, reply }
|
enum ReplySortType { time, like }
|
||||||
|
|
||||||
extension ReplySortTypeExtension on ReplySortType {
|
extension ReplySortTypeExtension on ReplySortType {
|
||||||
String get titles => ['最新评论', '最热评论', '回复最多'][index];
|
String get titles => ['最新评论', '最热评论'][index];
|
||||||
String get labels => ['最新', '最热', '最多回复'][index];
|
String get labels => ['最新', '最热'][index];
|
||||||
}
|
}
|
||||||
|
|||||||
@ -9,6 +9,7 @@ enum TabType { live, rcmd, hot, bangumi }
|
|||||||
|
|
||||||
extension TabTypeDesc on TabType {
|
extension TabTypeDesc on TabType {
|
||||||
String get description => ['直播', '推荐', '热门', '番剧'][index];
|
String get description => ['直播', '推荐', '热门', '番剧'][index];
|
||||||
|
String get id => ['live', 'rcmd', 'hot', 'bangumi'][index];
|
||||||
}
|
}
|
||||||
|
|
||||||
List tabsConfig = [
|
List tabsConfig = [
|
||||||
|
|||||||
@ -17,8 +17,9 @@ class LatestDataModel {
|
|||||||
url = json['url'];
|
url = json['url'];
|
||||||
tagName = json['tag_name'];
|
tagName = json['tag_name'];
|
||||||
createdAt = json['created_at'];
|
createdAt = json['created_at'];
|
||||||
assets =
|
assets = json['assets'] != null
|
||||||
json['assets'].map<AssetItem>((e) => AssetItem.fromJson(e)).toList();
|
? json['assets'].map<AssetItem>((e) => AssetItem.fromJson(e)).toList()
|
||||||
|
: [];
|
||||||
body = json['body'];
|
body = json['body'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,8 +1,5 @@
|
|||||||
import 'package:hive/hive.dart';
|
import 'package:pilipala/utils/id_utils.dart';
|
||||||
|
|
||||||
part 'result.g.dart';
|
|
||||||
|
|
||||||
@HiveType(typeId: 0)
|
|
||||||
class RecVideoItemAppModel {
|
class RecVideoItemAppModel {
|
||||||
RecVideoItemAppModel({
|
RecVideoItemAppModel({
|
||||||
this.id,
|
this.id,
|
||||||
@ -27,47 +24,27 @@ class RecVideoItemAppModel {
|
|||||||
this.adInfo,
|
this.adInfo,
|
||||||
});
|
});
|
||||||
|
|
||||||
@HiveField(0)
|
|
||||||
int? id;
|
int? id;
|
||||||
@HiveField(1)
|
|
||||||
int? aid;
|
int? aid;
|
||||||
@HiveField(2)
|
|
||||||
String? bvid;
|
String? bvid;
|
||||||
@HiveField(3)
|
|
||||||
int? cid;
|
int? cid;
|
||||||
@HiveField(4)
|
|
||||||
String? pic;
|
String? pic;
|
||||||
@HiveField(5)
|
|
||||||
RcmdStat? stat;
|
RcmdStat? stat;
|
||||||
@HiveField(6)
|
int? duration;
|
||||||
String? duration;
|
|
||||||
@HiveField(7)
|
|
||||||
String? title;
|
String? title;
|
||||||
@HiveField(8)
|
|
||||||
int? isFollowed;
|
int? isFollowed;
|
||||||
@HiveField(9)
|
|
||||||
RcmdOwner? owner;
|
RcmdOwner? owner;
|
||||||
@HiveField(10)
|
|
||||||
RcmdReason? rcmdReason;
|
RcmdReason? rcmdReason;
|
||||||
@HiveField(11)
|
|
||||||
String? goto;
|
String? goto;
|
||||||
@HiveField(12)
|
|
||||||
int? param;
|
int? param;
|
||||||
@HiveField(13)
|
|
||||||
String? uri;
|
String? uri;
|
||||||
@HiveField(14)
|
|
||||||
String? talkBack;
|
String? talkBack;
|
||||||
// 番剧
|
// 番剧
|
||||||
@HiveField(15)
|
|
||||||
String? bangumiView;
|
String? bangumiView;
|
||||||
@HiveField(16)
|
|
||||||
String? bangumiFollow;
|
String? bangumiFollow;
|
||||||
@HiveField(17)
|
|
||||||
String? bangumiBadge;
|
String? bangumiBadge;
|
||||||
|
|
||||||
@HiveField(18)
|
|
||||||
String? cardType;
|
String? cardType;
|
||||||
@HiveField(19)
|
|
||||||
Map? adInfo;
|
Map? adInfo;
|
||||||
|
|
||||||
RecVideoItemAppModel.fromJson(Map<String, dynamic> json) {
|
RecVideoItemAppModel.fromJson(Map<String, dynamic> json) {
|
||||||
@ -75,17 +52,32 @@ class RecVideoItemAppModel {
|
|||||||
? json['player_args']['aid']
|
? json['player_args']['aid']
|
||||||
: int.parse(json['param'] ?? '-1');
|
: int.parse(json['param'] ?? '-1');
|
||||||
aid = json['player_args'] != null ? json['player_args']['aid'] : -1;
|
aid = json['player_args'] != null ? json['player_args']['aid'] : -1;
|
||||||
bvid = null;
|
bvid = json['player_args'] != null
|
||||||
|
? IdUtils.av2bv(json['player_args']['aid'])
|
||||||
|
: '';
|
||||||
cid = json['player_args'] != null ? json['player_args']['cid'] : -1;
|
cid = json['player_args'] != null ? json['player_args']['cid'] : -1;
|
||||||
pic = json['cover'];
|
pic = json['cover'];
|
||||||
stat = RcmdStat.fromJson(json);
|
stat = RcmdStat.fromJson(json);
|
||||||
duration = json['cover_right_text'];
|
// 改用player_args中的duration作为原始数据(秒数)
|
||||||
|
duration =
|
||||||
|
json['player_args'] != null ? json['player_args']['duration'] : -1;
|
||||||
|
//duration = json['cover_right_text'];
|
||||||
title = json['title'];
|
title = json['title'];
|
||||||
isFollowed = 0;
|
|
||||||
owner = RcmdOwner.fromJson(json);
|
owner = RcmdOwner.fromJson(json);
|
||||||
rcmdReason = json['rcmd_reason_style'] != null
|
rcmdReason = json['rcmd_reason_style'] != null
|
||||||
? RcmdReason.fromJson(json['rcmd_reason_style'])
|
? RcmdReason.fromJson(json['rcmd_reason_style'])
|
||||||
: null;
|
: null;
|
||||||
|
// 由于app端api并不会直接返回与owner的关注状态
|
||||||
|
// 所以借用推荐原因是否为“已关注”、“新关注”等判别关注状态,从而与web端接口等效
|
||||||
|
isFollowed = rcmdReason != null &&
|
||||||
|
rcmdReason!.content != null &&
|
||||||
|
rcmdReason!.content!.contains('关注')
|
||||||
|
? 1
|
||||||
|
: 0;
|
||||||
|
// 如果是,就无需再显示推荐原因,交由view统一处理即可
|
||||||
|
if (isFollowed == 1) {
|
||||||
|
rcmdReason = null;
|
||||||
|
}
|
||||||
goto = json['goto'];
|
goto = json['goto'];
|
||||||
param = int.parse(json['param']);
|
param = int.parse(json['param']);
|
||||||
uri = json['uri'];
|
uri = json['uri'];
|
||||||
@ -102,18 +94,14 @@ class RecVideoItemAppModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@HiveType(typeId: 1)
|
|
||||||
class RcmdStat {
|
class RcmdStat {
|
||||||
RcmdStat({
|
RcmdStat({
|
||||||
this.view,
|
this.view,
|
||||||
this.like,
|
this.like,
|
||||||
this.danmu,
|
this.danmu,
|
||||||
});
|
});
|
||||||
@HiveField(0)
|
|
||||||
String? view;
|
String? view;
|
||||||
@HiveField(1)
|
|
||||||
String? like;
|
String? like;
|
||||||
@HiveField(2)
|
|
||||||
String? danmu;
|
String? danmu;
|
||||||
|
|
||||||
RcmdStat.fromJson(Map<String, dynamic> json) {
|
RcmdStat.fromJson(Map<String, dynamic> json) {
|
||||||
@ -122,13 +110,10 @@ class RcmdStat {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@HiveType(typeId: 2)
|
|
||||||
class RcmdOwner {
|
class RcmdOwner {
|
||||||
RcmdOwner({this.name, this.mid});
|
RcmdOwner({this.name, this.mid});
|
||||||
|
|
||||||
@HiveField(0)
|
|
||||||
String? name;
|
String? name;
|
||||||
@HiveField(1)
|
|
||||||
int? mid;
|
int? mid;
|
||||||
|
|
||||||
RcmdOwner.fromJson(Map<String, dynamic> json) {
|
RcmdOwner.fromJson(Map<String, dynamic> json) {
|
||||||
@ -141,13 +126,11 @@ class RcmdOwner {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@HiveType(typeId: 8)
|
|
||||||
class RcmdReason {
|
class RcmdReason {
|
||||||
RcmdReason({
|
RcmdReason({
|
||||||
this.content,
|
this.content,
|
||||||
});
|
});
|
||||||
|
|
||||||
@HiveField(0)
|
|
||||||
String? content;
|
String? content;
|
||||||
|
|
||||||
RcmdReason.fromJson(Map<String, dynamic> json) {
|
RcmdReason.fromJson(Map<String, dynamic> json) {
|
||||||
|
|||||||
@ -1,209 +0,0 @@
|
|||||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
|
||||||
|
|
||||||
part of 'result.dart';
|
|
||||||
|
|
||||||
// **************************************************************************
|
|
||||||
// TypeAdapterGenerator
|
|
||||||
// **************************************************************************
|
|
||||||
|
|
||||||
class RecVideoItemAppModelAdapter extends TypeAdapter<RecVideoItemAppModel> {
|
|
||||||
@override
|
|
||||||
final int typeId = 0;
|
|
||||||
|
|
||||||
@override
|
|
||||||
RecVideoItemAppModel read(BinaryReader reader) {
|
|
||||||
final numOfFields = reader.readByte();
|
|
||||||
final fields = <int, dynamic>{
|
|
||||||
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
|
||||||
};
|
|
||||||
return RecVideoItemAppModel(
|
|
||||||
id: fields[0] as int?,
|
|
||||||
aid: fields[1] as int?,
|
|
||||||
bvid: fields[2] as String?,
|
|
||||||
cid: fields[3] as int?,
|
|
||||||
pic: fields[4] as String?,
|
|
||||||
stat: fields[5] as RcmdStat?,
|
|
||||||
duration: fields[6] as String?,
|
|
||||||
title: fields[7] as String?,
|
|
||||||
isFollowed: fields[8] as int?,
|
|
||||||
owner: fields[9] as RcmdOwner?,
|
|
||||||
rcmdReason: fields[10] as RcmdReason?,
|
|
||||||
goto: fields[11] as String?,
|
|
||||||
param: fields[12] as int?,
|
|
||||||
uri: fields[13] as String?,
|
|
||||||
talkBack: fields[14] as String?,
|
|
||||||
bangumiView: fields[15] as String?,
|
|
||||||
bangumiFollow: fields[16] as String?,
|
|
||||||
bangumiBadge: fields[17] as String?,
|
|
||||||
cardType: fields[18] as String?,
|
|
||||||
adInfo: (fields[19] as Map?)?.cast<dynamic, dynamic>(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void write(BinaryWriter writer, RecVideoItemAppModel obj) {
|
|
||||||
writer
|
|
||||||
..writeByte(20)
|
|
||||||
..writeByte(0)
|
|
||||||
..write(obj.id)
|
|
||||||
..writeByte(1)
|
|
||||||
..write(obj.aid)
|
|
||||||
..writeByte(2)
|
|
||||||
..write(obj.bvid)
|
|
||||||
..writeByte(3)
|
|
||||||
..write(obj.cid)
|
|
||||||
..writeByte(4)
|
|
||||||
..write(obj.pic)
|
|
||||||
..writeByte(5)
|
|
||||||
..write(obj.stat)
|
|
||||||
..writeByte(6)
|
|
||||||
..write(obj.duration)
|
|
||||||
..writeByte(7)
|
|
||||||
..write(obj.title)
|
|
||||||
..writeByte(8)
|
|
||||||
..write(obj.isFollowed)
|
|
||||||
..writeByte(9)
|
|
||||||
..write(obj.owner)
|
|
||||||
..writeByte(10)
|
|
||||||
..write(obj.rcmdReason)
|
|
||||||
..writeByte(11)
|
|
||||||
..write(obj.goto)
|
|
||||||
..writeByte(12)
|
|
||||||
..write(obj.param)
|
|
||||||
..writeByte(13)
|
|
||||||
..write(obj.uri)
|
|
||||||
..writeByte(14)
|
|
||||||
..write(obj.talkBack)
|
|
||||||
..writeByte(15)
|
|
||||||
..write(obj.bangumiView)
|
|
||||||
..writeByte(16)
|
|
||||||
..write(obj.bangumiFollow)
|
|
||||||
..writeByte(17)
|
|
||||||
..write(obj.bangumiBadge)
|
|
||||||
..writeByte(18)
|
|
||||||
..write(obj.cardType)
|
|
||||||
..writeByte(19)
|
|
||||||
..write(obj.adInfo);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode => typeId.hashCode;
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) =>
|
|
||||||
identical(this, other) ||
|
|
||||||
other is RecVideoItemAppModelAdapter &&
|
|
||||||
runtimeType == other.runtimeType &&
|
|
||||||
typeId == other.typeId;
|
|
||||||
}
|
|
||||||
|
|
||||||
class RcmdStatAdapter extends TypeAdapter<RcmdStat> {
|
|
||||||
@override
|
|
||||||
final int typeId = 1;
|
|
||||||
|
|
||||||
@override
|
|
||||||
RcmdStat read(BinaryReader reader) {
|
|
||||||
final numOfFields = reader.readByte();
|
|
||||||
final fields = <int, dynamic>{
|
|
||||||
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
|
||||||
};
|
|
||||||
return RcmdStat(
|
|
||||||
view: fields[0] as String?,
|
|
||||||
like: fields[1] as String?,
|
|
||||||
danmu: fields[2] as String?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void write(BinaryWriter writer, RcmdStat obj) {
|
|
||||||
writer
|
|
||||||
..writeByte(3)
|
|
||||||
..writeByte(0)
|
|
||||||
..write(obj.view)
|
|
||||||
..writeByte(1)
|
|
||||||
..write(obj.like)
|
|
||||||
..writeByte(2)
|
|
||||||
..write(obj.danmu);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode => typeId.hashCode;
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) =>
|
|
||||||
identical(this, other) ||
|
|
||||||
other is RcmdStatAdapter &&
|
|
||||||
runtimeType == other.runtimeType &&
|
|
||||||
typeId == other.typeId;
|
|
||||||
}
|
|
||||||
|
|
||||||
class RcmdOwnerAdapter extends TypeAdapter<RcmdOwner> {
|
|
||||||
@override
|
|
||||||
final int typeId = 2;
|
|
||||||
|
|
||||||
@override
|
|
||||||
RcmdOwner read(BinaryReader reader) {
|
|
||||||
final numOfFields = reader.readByte();
|
|
||||||
final fields = <int, dynamic>{
|
|
||||||
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
|
||||||
};
|
|
||||||
return RcmdOwner(
|
|
||||||
name: fields[0] as String?,
|
|
||||||
mid: fields[1] as int?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void write(BinaryWriter writer, RcmdOwner obj) {
|
|
||||||
writer
|
|
||||||
..writeByte(2)
|
|
||||||
..writeByte(0)
|
|
||||||
..write(obj.name)
|
|
||||||
..writeByte(1)
|
|
||||||
..write(obj.mid);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode => typeId.hashCode;
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) =>
|
|
||||||
identical(this, other) ||
|
|
||||||
other is RcmdOwnerAdapter &&
|
|
||||||
runtimeType == other.runtimeType &&
|
|
||||||
typeId == other.typeId;
|
|
||||||
}
|
|
||||||
|
|
||||||
class RcmdReasonAdapter extends TypeAdapter<RcmdReason> {
|
|
||||||
@override
|
|
||||||
final int typeId = 8;
|
|
||||||
|
|
||||||
@override
|
|
||||||
RcmdReason read(BinaryReader reader) {
|
|
||||||
final numOfFields = reader.readByte();
|
|
||||||
final fields = <int, dynamic>{
|
|
||||||
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
|
||||||
};
|
|
||||||
return RcmdReason(
|
|
||||||
content: fields[0] as String?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void write(BinaryWriter writer, RcmdReason obj) {
|
|
||||||
writer
|
|
||||||
..writeByte(1)
|
|
||||||
..writeByte(0)
|
|
||||||
..write(obj.content);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode => typeId.hashCode;
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) =>
|
|
||||||
identical(this, other) ||
|
|
||||||
other is RcmdReasonAdapter &&
|
|
||||||
runtimeType == other.runtimeType &&
|
|
||||||
typeId == other.typeId;
|
|
||||||
}
|
|
||||||
130
lib/models/live/room_info_h5.dart
Normal file
130
lib/models/live/room_info_h5.dart
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
class RoomInfoH5Model {
|
||||||
|
RoomInfoH5Model({
|
||||||
|
this.roomInfo,
|
||||||
|
this.anchorInfo,
|
||||||
|
this.isRoomFeed,
|
||||||
|
this.watchedShow,
|
||||||
|
this.likeInfoV3,
|
||||||
|
this.blockInfo,
|
||||||
|
});
|
||||||
|
|
||||||
|
RoomInfo? roomInfo;
|
||||||
|
AnchorInfo? anchorInfo;
|
||||||
|
int? isRoomFeed;
|
||||||
|
Map? watchedShow;
|
||||||
|
LikeInfoV3? likeInfoV3;
|
||||||
|
Map? blockInfo;
|
||||||
|
|
||||||
|
RoomInfoH5Model.fromJson(Map<String, dynamic> json) {
|
||||||
|
roomInfo = RoomInfo.fromJson(json['room_info']);
|
||||||
|
anchorInfo = AnchorInfo.fromJson(json['anchor_info']);
|
||||||
|
isRoomFeed = json['is_room_feed'];
|
||||||
|
watchedShow = json['watched_show'];
|
||||||
|
likeInfoV3 = LikeInfoV3.fromJson(json['like_info_v3']);
|
||||||
|
blockInfo = json['block_info'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RoomInfo {
|
||||||
|
RoomInfo({
|
||||||
|
this.uid,
|
||||||
|
this.roomId,
|
||||||
|
this.title,
|
||||||
|
this.cover,
|
||||||
|
this.description,
|
||||||
|
this.liveStatus,
|
||||||
|
this.liveStartTime,
|
||||||
|
this.areaId,
|
||||||
|
this.areaName,
|
||||||
|
this.parentAreaId,
|
||||||
|
this.parentAreaName,
|
||||||
|
this.online,
|
||||||
|
this.background,
|
||||||
|
this.appBackground,
|
||||||
|
this.liveId,
|
||||||
|
});
|
||||||
|
|
||||||
|
int? uid;
|
||||||
|
int? roomId;
|
||||||
|
String? title;
|
||||||
|
String? cover;
|
||||||
|
String? description;
|
||||||
|
int? liveStatus;
|
||||||
|
int? liveStartTime;
|
||||||
|
int? areaId;
|
||||||
|
String? areaName;
|
||||||
|
int? parentAreaId;
|
||||||
|
String? parentAreaName;
|
||||||
|
int? online;
|
||||||
|
String? background;
|
||||||
|
String? appBackground;
|
||||||
|
String? liveId;
|
||||||
|
|
||||||
|
RoomInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
uid = json['uid'];
|
||||||
|
roomId = json['room_id'];
|
||||||
|
title = json['title'];
|
||||||
|
cover = json['cover'];
|
||||||
|
description = json['description'];
|
||||||
|
liveStatus = json['liveS_satus'];
|
||||||
|
liveStartTime = json['live_start_time'];
|
||||||
|
areaId = json['area_id'];
|
||||||
|
areaName = json['area_name'];
|
||||||
|
parentAreaId = json['parent_area_id'];
|
||||||
|
parentAreaName = json['parent_area_name'];
|
||||||
|
online = json['online'];
|
||||||
|
background = json['background'];
|
||||||
|
appBackground = json['app_background'];
|
||||||
|
liveId = json['live_id'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class AnchorInfo {
|
||||||
|
AnchorInfo({
|
||||||
|
this.baseInfo,
|
||||||
|
this.relationInfo,
|
||||||
|
});
|
||||||
|
|
||||||
|
BaseInfo? baseInfo;
|
||||||
|
RelationInfo? relationInfo;
|
||||||
|
|
||||||
|
AnchorInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
baseInfo = BaseInfo.fromJson(json['base_info']);
|
||||||
|
relationInfo = RelationInfo.fromJson(json['relation_info']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class BaseInfo {
|
||||||
|
BaseInfo({
|
||||||
|
this.uname,
|
||||||
|
this.face,
|
||||||
|
});
|
||||||
|
|
||||||
|
String? uname;
|
||||||
|
String? face;
|
||||||
|
|
||||||
|
BaseInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
uname = json['uname'];
|
||||||
|
face = json['face'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RelationInfo {
|
||||||
|
RelationInfo({this.attention});
|
||||||
|
|
||||||
|
int? attention;
|
||||||
|
|
||||||
|
RelationInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
attention = json['attention'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class LikeInfoV3 {
|
||||||
|
LikeInfoV3({this.totalLikes});
|
||||||
|
|
||||||
|
int? totalLikes;
|
||||||
|
|
||||||
|
LikeInfoV3.fromJson(Map<String, dynamic> json) {
|
||||||
|
totalLikes = json['total_likes'];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,5 +1,3 @@
|
|||||||
import 'package:pilipala/utils/utils.dart';
|
|
||||||
|
|
||||||
import './model_owner.dart';
|
import './model_owner.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
|
|
||||||
@ -38,7 +36,7 @@ class RecVideoItemModel {
|
|||||||
@HiveField(6)
|
@HiveField(6)
|
||||||
String? title = '';
|
String? title = '';
|
||||||
@HiveField(7)
|
@HiveField(7)
|
||||||
String? duration = '';
|
int? duration = -1;
|
||||||
@HiveField(8)
|
@HiveField(8)
|
||||||
int? pubdate = -1;
|
int? pubdate = -1;
|
||||||
@HiveField(9)
|
@HiveField(9)
|
||||||
@ -58,7 +56,7 @@ class RecVideoItemModel {
|
|||||||
uri = json["uri"];
|
uri = json["uri"];
|
||||||
pic = json["pic"];
|
pic = json["pic"];
|
||||||
title = json["title"];
|
title = json["title"];
|
||||||
duration = Utils.tampToSeektime(json["duration"]);
|
duration = json["duration"];
|
||||||
pubdate = json["pubdate"];
|
pubdate = json["pubdate"];
|
||||||
owner = Owner.fromJson(json["owner"]);
|
owner = Owner.fromJson(json["owner"]);
|
||||||
stat = Stat.fromJson(json["stat"]);
|
stat = Stat.fromJson(json["stat"]);
|
||||||
@ -77,14 +75,15 @@ class Stat {
|
|||||||
this.danmu,
|
this.danmu,
|
||||||
});
|
});
|
||||||
@HiveField(0)
|
@HiveField(0)
|
||||||
String? view;
|
int? view;
|
||||||
@HiveField(1)
|
@HiveField(1)
|
||||||
int? like;
|
int? like;
|
||||||
@HiveField(2)
|
@HiveField(2)
|
||||||
int? danmu;
|
int? danmu;
|
||||||
|
|
||||||
Stat.fromJson(Map<String, dynamic> json) {
|
Stat.fromJson(Map<String, dynamic> json) {
|
||||||
view = Utils.numFormat(json["view"]);
|
// 无需在model中转换以保留原始数据,在view层处理即可
|
||||||
|
view = json["view"];
|
||||||
like = json["like"];
|
like = json["like"];
|
||||||
danmu = json['danmaku'];
|
danmu = json['danmaku'];
|
||||||
}
|
}
|
||||||
|
|||||||
@ -24,7 +24,7 @@ class RecVideoItemModelAdapter extends TypeAdapter<RecVideoItemModel> {
|
|||||||
uri: fields[4] as String?,
|
uri: fields[4] as String?,
|
||||||
pic: fields[5] as String?,
|
pic: fields[5] as String?,
|
||||||
title: fields[6] as String?,
|
title: fields[6] as String?,
|
||||||
duration: fields[7] as String?,
|
duration: fields[7] as int?,
|
||||||
pubdate: fields[8] as int?,
|
pubdate: fields[8] as int?,
|
||||||
owner: fields[9] as Owner?,
|
owner: fields[9] as Owner?,
|
||||||
stat: fields[10] as Stat?,
|
stat: fields[10] as Stat?,
|
||||||
@ -87,7 +87,7 @@ class StatAdapter extends TypeAdapter<Stat> {
|
|||||||
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
for (int i = 0; i < numOfFields; i++) reader.readByte(): reader.read(),
|
||||||
};
|
};
|
||||||
return Stat(
|
return Stat(
|
||||||
view: fields[0] as String?,
|
view: fields[0] as int?,
|
||||||
like: fields[1] as int?,
|
like: fields[1] as int?,
|
||||||
danmu: fields[2] as int?,
|
danmu: fields[2] as int?,
|
||||||
);
|
);
|
||||||
|
|||||||
@ -166,7 +166,7 @@ class SessionMsgDataModel {
|
|||||||
int? hasMore;
|
int? hasMore;
|
||||||
int? minSeqno;
|
int? minSeqno;
|
||||||
int? maxSeqno;
|
int? maxSeqno;
|
||||||
List? eInfos;
|
List<dynamic>? eInfos;
|
||||||
|
|
||||||
SessionMsgDataModel.fromJson(Map<String, dynamic> json) {
|
SessionMsgDataModel.fromJson(Map<String, dynamic> json) {
|
||||||
messages = json['messages']
|
messages = json['messages']
|
||||||
|
|||||||
@ -7,6 +7,7 @@ import 'package:pilipala/http/index.dart';
|
|||||||
import 'package:pilipala/models/github/latest.dart';
|
import 'package:pilipala/models/github/latest.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
import '../../utils/cache_manage.dart';
|
||||||
|
|
||||||
class AboutPage extends StatefulWidget {
|
class AboutPage extends StatefulWidget {
|
||||||
const AboutPage({super.key});
|
const AboutPage({super.key});
|
||||||
@ -17,10 +18,23 @@ class AboutPage extends StatefulWidget {
|
|||||||
|
|
||||||
class _AboutPageState extends State<AboutPage> {
|
class _AboutPageState extends State<AboutPage> {
|
||||||
final AboutController _aboutController = Get.put(AboutController());
|
final AboutController _aboutController = Get.put(AboutController());
|
||||||
|
String cacheSize = '';
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
// 读取缓存占用
|
||||||
|
getCacheSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> getCacheSize() async {
|
||||||
|
final res = await CacheManage().loadApplicationCache();
|
||||||
|
setState(() => cacheSize = res);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
Color outline = Theme.of(context).colorScheme.outline;
|
final Color outline = Theme.of(context).colorScheme.outline;
|
||||||
TextStyle subTitleStyle =
|
TextStyle subTitleStyle =
|
||||||
TextStyle(fontSize: 13, color: Theme.of(context).colorScheme.outline);
|
TextStyle(fontSize: 13, color: Theme.of(context).colorScheme.outline);
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
@ -29,7 +43,6 @@ class _AboutPageState extends State<AboutPage> {
|
|||||||
),
|
),
|
||||||
body: SingleChildScrollView(
|
body: SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
Image.asset(
|
Image.asset(
|
||||||
'assets/images/logo/logo_android_2.png',
|
'assets/images/logo/logo_android_2.png',
|
||||||
@ -47,7 +60,7 @@ class _AboutPageState extends State<AboutPage> {
|
|||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
Obx(
|
Obx(
|
||||||
() => ListTile(
|
() => ListTile(
|
||||||
title: const Text("当前版本"),
|
title: const Text('当前版本'),
|
||||||
trailing: Text(_aboutController.currentVersion.value,
|
trailing: Text(_aboutController.currentVersion.value,
|
||||||
style: subTitleStyle),
|
style: subTitleStyle),
|
||||||
),
|
),
|
||||||
@ -87,6 +100,14 @@ class _AboutPageState extends State<AboutPage> {
|
|||||||
style: subTitleStyle,
|
style: subTitleStyle,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
onTap: () => _aboutController.webSiteUrl(),
|
||||||
|
title: const Text('访问官网'),
|
||||||
|
trailing: Text(
|
||||||
|
'https://pilipalanet.mysxl.cn',
|
||||||
|
style: subTitleStyle,
|
||||||
|
),
|
||||||
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
onTap: () => _aboutController.panDownload(),
|
onTap: () => _aboutController.panDownload(),
|
||||||
title: const Text('网盘下载'),
|
title: const Text('网盘下载'),
|
||||||
@ -126,6 +147,22 @@ class _AboutPageState extends State<AboutPage> {
|
|||||||
title: const Text('赞助'),
|
title: const Text('赞助'),
|
||||||
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
onTap: () => _aboutController.logs(),
|
||||||
|
title: const Text('错误日志'),
|
||||||
|
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
onTap: () async {
|
||||||
|
var cleanStatus = await CacheManage().clearCacheAll();
|
||||||
|
if (cleanStatus) {
|
||||||
|
getCacheSize();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
title: const Text('清除缓存'),
|
||||||
|
subtitle: Text('图片及网络缓存 $cacheSize', style: subTitleStyle),
|
||||||
|
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -245,4 +282,17 @@ class AboutController extends GetxController {
|
|||||||
print(e);
|
print(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 官网
|
||||||
|
webSiteUrl() {
|
||||||
|
launchUrl(
|
||||||
|
Uri.parse('https://pilipalanet.mysxl.cn'),
|
||||||
|
mode: LaunchMode.externalApplication,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 日志
|
||||||
|
logs() {
|
||||||
|
Get.toNamed('/logs');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -63,8 +63,8 @@ class BangumiIntroController extends GetxController {
|
|||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
super.onInit();
|
super.onInit();
|
||||||
if (Get.arguments.isNotEmpty) {
|
if (Get.arguments.isNotEmpty as bool) {
|
||||||
if (Get.arguments.containsKey('bangumiItem')) {
|
if (Get.arguments.containsKey('bangumiItem') as bool) {
|
||||||
preRender = true;
|
preRender = true;
|
||||||
bangumiItem = Get.arguments['bangumiItem'];
|
bangumiItem = Get.arguments['bangumiItem'];
|
||||||
// bangumiItem!['pic'] = args.pic;
|
// bangumiItem!['pic'] = args.pic;
|
||||||
|
|||||||
@ -51,12 +51,11 @@ class _BangumiIntroPanelState extends State<BangumiIntroPanel>
|
|||||||
cid = widget.cid!;
|
cid = widget.cid!;
|
||||||
bangumiIntroController = Get.put(BangumiIntroController(), tag: heroTag);
|
bangumiIntroController = Get.put(BangumiIntroController(), tag: heroTag);
|
||||||
videoDetailCtr = Get.find<VideoDetailController>(tag: heroTag);
|
videoDetailCtr = Get.find<VideoDetailController>(tag: heroTag);
|
||||||
bangumiIntroController.bangumiDetail.listen((value) {
|
bangumiIntroController.bangumiDetail.listen((BangumiInfoModel value) {
|
||||||
bangumiDetail = value;
|
bangumiDetail = value;
|
||||||
});
|
});
|
||||||
_futureBuilderFuture = bangumiIntroController.queryBangumiIntro();
|
_futureBuilderFuture = bangumiIntroController.queryBangumiIntro();
|
||||||
videoDetailCtr.cid.listen((p0) {
|
videoDetailCtr.cid.listen((int p0) {
|
||||||
print('🐶🐶$p0');
|
|
||||||
cid = p0;
|
cid = p0;
|
||||||
setState(() {});
|
setState(() {});
|
||||||
});
|
});
|
||||||
@ -67,7 +66,7 @@ class _BangumiIntroPanelState extends State<BangumiIntroPanel>
|
|||||||
super.build(context);
|
super.build(context);
|
||||||
return FutureBuilder(
|
return FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.done) {
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
if (snapshot.data['status']) {
|
if (snapshot.data['status']) {
|
||||||
// 请求成功
|
// 请求成功
|
||||||
@ -83,7 +82,7 @@ class _BangumiIntroPanelState extends State<BangumiIntroPanel>
|
|||||||
// errMsg: snapshot.data['msg'],
|
// errMsg: snapshot.data['msg'],
|
||||||
// fn: () => Get.back(),
|
// fn: () => Get.back(),
|
||||||
// );
|
// );
|
||||||
return SizedBox();
|
return const SizedBox();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return BangumiInfo(
|
return BangumiInfo(
|
||||||
@ -98,16 +97,16 @@ class _BangumiIntroPanelState extends State<BangumiIntroPanel>
|
|||||||
}
|
}
|
||||||
|
|
||||||
class BangumiInfo extends StatefulWidget {
|
class BangumiInfo extends StatefulWidget {
|
||||||
final bool loadingStatus;
|
|
||||||
final BangumiInfoModel? bangumiDetail;
|
|
||||||
final int? cid;
|
|
||||||
|
|
||||||
const BangumiInfo({
|
const BangumiInfo({
|
||||||
Key? key,
|
super.key,
|
||||||
this.loadingStatus = false,
|
this.loadingStatus = false,
|
||||||
this.bangumiDetail,
|
this.bangumiDetail,
|
||||||
this.cid,
|
this.cid,
|
||||||
}) : super(key: key);
|
});
|
||||||
|
|
||||||
|
final bool loadingStatus;
|
||||||
|
final BangumiInfoModel? bangumiDetail;
|
||||||
|
final int? cid;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<BangumiInfo> createState() => _BangumiInfoState();
|
State<BangumiInfo> createState() => _BangumiInfoState();
|
||||||
@ -123,12 +122,15 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
int? cid;
|
int? cid;
|
||||||
bool isProcessing = false;
|
bool isProcessing = false;
|
||||||
void Function()? handleState(Future Function() action) {
|
void Function()? handleState(Future Function() action) {
|
||||||
return isProcessing ? null : () async {
|
return isProcessing
|
||||||
setState(() => isProcessing = true);
|
? null
|
||||||
await action();
|
: () async {
|
||||||
setState(() => isProcessing = false);
|
setState(() => isProcessing = true);
|
||||||
};
|
await action();
|
||||||
|
setState(() => isProcessing = false);
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
@ -155,7 +157,7 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
context: context,
|
context: context,
|
||||||
useRootNavigator: true,
|
useRootNavigator: true,
|
||||||
isScrollControlled: true,
|
isScrollControlled: true,
|
||||||
builder: (context) {
|
builder: (BuildContext context) {
|
||||||
return FavPanel(ctr: bangumiIntroController);
|
return FavPanel(ctr: bangumiIntroController);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -175,7 +177,7 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
ThemeData t = Theme.of(context);
|
final ThemeData t = Theme.of(context);
|
||||||
return SliverPadding(
|
return SliverPadding(
|
||||||
padding: const EdgeInsets.only(
|
padding: const EdgeInsets.only(
|
||||||
left: StyleString.safeSpace, right: StyleString.safeSpace, top: 20),
|
left: StyleString.safeSpace, right: StyleString.safeSpace, top: 20),
|
||||||
@ -185,7 +187,6 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Row(
|
Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Stack(
|
Stack(
|
||||||
@ -244,7 +245,7 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
EdgeInsets.zero),
|
EdgeInsets.zero),
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
MaterialStateProperty.resolveWith(
|
MaterialStateProperty.resolveWith(
|
||||||
(states) {
|
(Set<MaterialState> states) {
|
||||||
return t
|
return t
|
||||||
.colorScheme.primaryContainer
|
.colorScheme.primaryContainer
|
||||||
.withOpacity(0.7);
|
.withOpacity(0.7);
|
||||||
@ -386,7 +387,8 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget actionGrid(BuildContext context, bangumiIntroController) {
|
Widget actionGrid(BuildContext context, bangumiIntroController) {
|
||||||
return LayoutBuilder(builder: (context, constraints) {
|
return LayoutBuilder(
|
||||||
|
builder: (BuildContext context, BoxConstraints constraints) {
|
||||||
return Material(
|
return Material(
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.only(top: 16, bottom: 8),
|
padding: const EdgeInsets.only(top: 16, bottom: 8),
|
||||||
@ -394,7 +396,7 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
height: constraints.maxWidth / 5 * 0.8,
|
height: constraints.maxWidth / 5 * 0.8,
|
||||||
child: GridView.count(
|
child: GridView.count(
|
||||||
primary: false,
|
primary: false,
|
||||||
padding: const EdgeInsets.all(0),
|
padding: EdgeInsets.zero,
|
||||||
crossAxisCount: 5,
|
crossAxisCount: 5,
|
||||||
childAspectRatio: 1.25,
|
childAspectRatio: 1.25,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
@ -402,7 +404,8 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
() => ActionItem(
|
() => ActionItem(
|
||||||
icon: const Icon(FontAwesomeIcons.thumbsUp),
|
icon: const Icon(FontAwesomeIcons.thumbsUp),
|
||||||
selectIcon: const Icon(FontAwesomeIcons.solidThumbsUp),
|
selectIcon: const Icon(FontAwesomeIcons.solidThumbsUp),
|
||||||
onTap: handleState(bangumiIntroController.actionLikeVideo),
|
onTap:
|
||||||
|
handleState(bangumiIntroController.actionLikeVideo),
|
||||||
selectStatus: bangumiIntroController.hasLike.value,
|
selectStatus: bangumiIntroController.hasLike.value,
|
||||||
loadingStatus: false,
|
loadingStatus: false,
|
||||||
text: !widget.loadingStatus
|
text: !widget.loadingStatus
|
||||||
@ -413,7 +416,8 @@ class _BangumiInfoState extends State<BangumiInfo> {
|
|||||||
() => ActionItem(
|
() => ActionItem(
|
||||||
icon: const Icon(FontAwesomeIcons.b),
|
icon: const Icon(FontAwesomeIcons.b),
|
||||||
selectIcon: const Icon(FontAwesomeIcons.b),
|
selectIcon: const Icon(FontAwesomeIcons.b),
|
||||||
onTap: handleState(bangumiIntroController.actionCoinVideo),
|
onTap:
|
||||||
|
handleState(bangumiIntroController.actionCoinVideo),
|
||||||
selectStatus: bangumiIntroController.hasCoin.value,
|
selectStatus: bangumiIntroController.hasCoin.value,
|
||||||
loadingStatus: false,
|
loadingStatus: false,
|
||||||
text: !widget.loadingStatus
|
text: !widget.loadingStatus
|
||||||
|
|||||||
@ -4,11 +4,11 @@ import 'package:easy_debounce/easy_throttle.dart';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/rendering.dart';
|
import 'package:flutter/rendering.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:nil/nil.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import 'package:pilipala/common/constants.dart';
|
||||||
import 'package:pilipala/common/widgets/http_error.dart';
|
import 'package:pilipala/common/widgets/http_error.dart';
|
||||||
import 'package:pilipala/pages/home/index.dart';
|
import 'package:pilipala/pages/home/index.dart';
|
||||||
import 'package:pilipala/pages/main/index.dart';
|
import 'package:pilipala/pages/main/index.dart';
|
||||||
import 'package:pilipala/pages/rcmd/view.dart';
|
|
||||||
|
|
||||||
import 'controller.dart';
|
import 'controller.dart';
|
||||||
import 'widgets/bangumu_card_v.dart';
|
import 'widgets/bangumu_card_v.dart';
|
||||||
@ -74,7 +74,7 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
super.build(context);
|
super.build(context);
|
||||||
return RefreshIndicator(
|
return RefreshIndicator(
|
||||||
onRefresh: () async {
|
onRefresh: () async {
|
||||||
await _bangumidController.queryBangumiListFeed(type: 'init');
|
await _bangumidController.queryBangumiListFeed();
|
||||||
return _bangumidController.queryBangumiFollow();
|
return _bangumidController.queryBangumiFollow();
|
||||||
},
|
},
|
||||||
child: CustomScrollView(
|
child: CustomScrollView(
|
||||||
@ -112,10 +112,11 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
height: 258,
|
height: 268,
|
||||||
child: FutureBuilder(
|
child: FutureBuilder(
|
||||||
future: _futureBuilderFutureFollow,
|
future: _futureBuilderFutureFollow,
|
||||||
builder: (context, snapshot) {
|
builder:
|
||||||
|
(BuildContext context, AsyncSnapshot snapshot) {
|
||||||
if (snapshot.connectionState ==
|
if (snapshot.connectionState ==
|
||||||
ConnectionState.done) {
|
ConnectionState.done) {
|
||||||
if (snapshot.data == null) {
|
if (snapshot.data == null) {
|
||||||
@ -156,10 +157,10 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
return const SizedBox();
|
return nil;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return const SizedBox();
|
return nil;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@ -188,7 +189,7 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
StyleString.safeSpace, 0, StyleString.safeSpace, 0),
|
StyleString.safeSpace, 0, StyleString.safeSpace, 0),
|
||||||
sliver: FutureBuilder(
|
sliver: FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.done) {
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
Map data = snapshot.data as Map;
|
Map data = snapshot.data as Map;
|
||||||
if (data['status']) {
|
if (data['status']) {
|
||||||
@ -197,7 +198,10 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
} else {
|
} else {
|
||||||
return HttpError(
|
return HttpError(
|
||||||
errMsg: data['msg'],
|
errMsg: data['msg'],
|
||||||
fn: () => {},
|
fn: () {
|
||||||
|
_futureBuilderFuture =
|
||||||
|
_bangumidController.queryBangumiListFeed();
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -206,7 +210,6 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
LoadingMore()
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@ -222,13 +225,13 @@ class _BangumiPageState extends State<BangumiPage>
|
|||||||
// 列数
|
// 列数
|
||||||
crossAxisCount: 3,
|
crossAxisCount: 3,
|
||||||
mainAxisExtent: Get.size.width / 3 / 0.65 +
|
mainAxisExtent: Get.size.width / 3 / 0.65 +
|
||||||
32 * MediaQuery.of(context).textScaleFactor,
|
MediaQuery.textScalerOf(context).scale(32.0),
|
||||||
),
|
),
|
||||||
delegate: SliverChildBuilderDelegate(
|
delegate: SliverChildBuilderDelegate(
|
||||||
(BuildContext context, int index) {
|
(BuildContext context, int index) {
|
||||||
return bangumiList!.isNotEmpty
|
return bangumiList!.isNotEmpty
|
||||||
? BangumiCardV(bangumiItem: bangumiList[index])
|
? BangumiCardV(bangumiItem: bangumiList[index])
|
||||||
: const SizedBox();
|
: nil;
|
||||||
},
|
},
|
||||||
childCount: bangumiList!.isNotEmpty ? bangumiList!.length : 10,
|
childCount: bangumiList!.isNotEmpty ? bangumiList!.length : 10,
|
||||||
),
|
),
|
||||||
|
|||||||
@ -8,11 +8,6 @@ import 'package:pilipala/utils/storage.dart';
|
|||||||
import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
|
import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
|
||||||
|
|
||||||
class BangumiPanel extends StatefulWidget {
|
class BangumiPanel extends StatefulWidget {
|
||||||
final List<EpisodeItem> pages;
|
|
||||||
final int? cid;
|
|
||||||
final double? sheetHeight;
|
|
||||||
final Function? changeFuc;
|
|
||||||
|
|
||||||
const BangumiPanel({
|
const BangumiPanel({
|
||||||
super.key,
|
super.key,
|
||||||
required this.pages,
|
required this.pages,
|
||||||
@ -21,6 +16,11 @@ class BangumiPanel extends StatefulWidget {
|
|||||||
this.changeFuc,
|
this.changeFuc,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
final List<EpisodeItem> pages;
|
||||||
|
final int? cid;
|
||||||
|
final double? sheetHeight;
|
||||||
|
final Function? changeFuc;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<BangumiPanel> createState() => _BangumiPanelState();
|
State<BangumiPanel> createState() => _BangumiPanelState();
|
||||||
}
|
}
|
||||||
@ -50,10 +50,10 @@ class _BangumiPanelState extends State<BangumiPanel> {
|
|||||||
}
|
}
|
||||||
videoDetailCtr = Get.find<VideoDetailController>(tag: heroTag);
|
videoDetailCtr = Get.find<VideoDetailController>(tag: heroTag);
|
||||||
|
|
||||||
videoDetailCtr.cid.listen((p0) {
|
videoDetailCtr.cid.listen((int p0) {
|
||||||
cid = p0;
|
cid = p0;
|
||||||
setState(() {});
|
setState(() {});
|
||||||
currentIndex = widget.pages.indexWhere((e) => e.cid == cid);
|
currentIndex = widget.pages.indexWhere((EpisodeItem e) => e.cid == cid);
|
||||||
scrollToIndex();
|
scrollToIndex();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -106,7 +106,8 @@ class _BangumiPanelState extends State<BangumiPanel> {
|
|||||||
child: Material(
|
child: Material(
|
||||||
child: ScrollablePositionedList.builder(
|
child: ScrollablePositionedList.builder(
|
||||||
itemCount: widget.pages.length,
|
itemCount: widget.pages.length,
|
||||||
itemBuilder: (context, index) => ListTile(
|
itemBuilder: (BuildContext context, int index) =>
|
||||||
|
ListTile(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
setState(() {
|
setState(() {
|
||||||
changeFucCall(widget.pages[index], index);
|
changeFucCall(widget.pages[index], index);
|
||||||
@ -150,7 +151,7 @@ class _BangumiPanelState extends State<BangumiPanel> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void changeFucCall(item, i) async {
|
void changeFucCall(item, i) async {
|
||||||
if (item.badge != null && vipStatus != 1) {
|
if (item.badge != null && item.badge == '会员' && vipStatus != 1) {
|
||||||
SmartDialog.showToast('需要大会员');
|
SmartDialog.showToast('需要大会员');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -212,78 +213,87 @@ class _BangumiPanelState extends State<BangumiPanel> {
|
|||||||
SizedBox(
|
SizedBox(
|
||||||
height: 60,
|
height: 60,
|
||||||
child: ListView.builder(
|
child: ListView.builder(
|
||||||
controller: listViewScrollCtr,
|
controller: listViewScrollCtr,
|
||||||
scrollDirection: Axis.horizontal,
|
scrollDirection: Axis.horizontal,
|
||||||
itemCount: widget.pages.length,
|
itemCount: widget.pages.length,
|
||||||
itemExtent: 150,
|
itemExtent: 150,
|
||||||
itemBuilder: ((context, i) {
|
itemBuilder: (BuildContext context, int i) {
|
||||||
return Container(
|
return Container(
|
||||||
width: 150,
|
width: 150,
|
||||||
margin: const EdgeInsets.only(right: 10),
|
margin: const EdgeInsets.only(right: 10),
|
||||||
child: Material(
|
child: Material(
|
||||||
color: Theme.of(context).colorScheme.onInverseSurface,
|
color: Theme.of(context).colorScheme.onInverseSurface,
|
||||||
borderRadius: BorderRadius.circular(6),
|
borderRadius: BorderRadius.circular(6),
|
||||||
clipBehavior: Clip.hardEdge,
|
clipBehavior: Clip.hardEdge,
|
||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: () => changeFucCall(widget.pages[i], i),
|
onTap: () => changeFucCall(widget.pages[i], i),
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.symmetric(
|
padding: const EdgeInsets.symmetric(
|
||||||
vertical: 8, horizontal: 10),
|
vertical: 8, horizontal: 10),
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: <Widget>[
|
||||||
Row(
|
Row(
|
||||||
children: [
|
children: [
|
||||||
if (i == currentIndex) ...[
|
if (i == currentIndex) ...<Widget>[
|
||||||
Image.asset(
|
Image.asset(
|
||||||
'assets/images/live.png',
|
'assets/images/live.png',
|
||||||
color:
|
color: Theme.of(context).colorScheme.primary,
|
||||||
Theme.of(context).colorScheme.primary,
|
height: 12,
|
||||||
height: 12,
|
|
||||||
),
|
|
||||||
const SizedBox(width: 6)
|
|
||||||
],
|
|
||||||
Text(
|
|
||||||
'第${i + 1}话',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 13,
|
|
||||||
color: i == currentIndex
|
|
||||||
? Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.primary
|
|
||||||
: Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.onSurface),
|
|
||||||
),
|
),
|
||||||
const SizedBox(width: 2),
|
const SizedBox(width: 6)
|
||||||
if (widget.pages[i].badge != null) ...[
|
],
|
||||||
|
Text(
|
||||||
|
'第${i + 1}话',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 13,
|
||||||
|
color: i == currentIndex
|
||||||
|
? Theme.of(context).colorScheme.primary
|
||||||
|
: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSurface),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 2),
|
||||||
|
if (widget.pages[i].badge != null) ...[
|
||||||
|
if (widget.pages[i].badge == '会员') ...[
|
||||||
Image.asset(
|
Image.asset(
|
||||||
'assets/images/big-vip.png',
|
'assets/images/big-vip.png',
|
||||||
height: 16,
|
height: 16,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
],
|
if (widget.pages[i].badge != '会员') ...[
|
||||||
),
|
const Spacer(),
|
||||||
const SizedBox(height: 3),
|
Text(
|
||||||
Text(
|
widget.pages[i].badge!,
|
||||||
widget.pages[i].longTitle!,
|
style: TextStyle(
|
||||||
maxLines: 1,
|
fontSize: 11,
|
||||||
style: TextStyle(
|
color:
|
||||||
fontSize: 13,
|
Theme.of(context).colorScheme.primary,
|
||||||
color: i == currentIndex
|
),
|
||||||
? Theme.of(context).colorScheme.primary
|
),
|
||||||
: Theme.of(context)
|
],
|
||||||
.colorScheme
|
]
|
||||||
.onSurface),
|
],
|
||||||
overflow: TextOverflow.ellipsis,
|
),
|
||||||
)
|
const SizedBox(height: 3),
|
||||||
],
|
Text(
|
||||||
),
|
widget.pages[i].longTitle!,
|
||||||
|
maxLines: 1,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 13,
|
||||||
|
color: i == currentIndex
|
||||||
|
? Theme.of(context).colorScheme.primary
|
||||||
|
: Theme.of(context).colorScheme.onSurface),
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
)
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
),
|
||||||
})),
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|||||||
@ -11,17 +11,16 @@ import 'package:pilipala/common/widgets/network_img_layer.dart';
|
|||||||
|
|
||||||
// 视频卡片 - 垂直布局
|
// 视频卡片 - 垂直布局
|
||||||
class BangumiCardV extends StatelessWidget {
|
class BangumiCardV extends StatelessWidget {
|
||||||
// ignore: prefer_typing_uninitialized_variables
|
|
||||||
final bangumiItem;
|
|
||||||
final Function()? longPress;
|
|
||||||
final Function()? longPressEnd;
|
|
||||||
|
|
||||||
const BangumiCardV({
|
const BangumiCardV({
|
||||||
Key? key,
|
super.key,
|
||||||
required this.bangumiItem,
|
required this.bangumiItem,
|
||||||
this.longPress,
|
this.longPress,
|
||||||
this.longPressEnd,
|
this.longPressEnd,
|
||||||
}) : super(key: key);
|
});
|
||||||
|
|
||||||
|
final bangumiItem;
|
||||||
|
final Function()? longPress;
|
||||||
|
final Function()? longPressEnd;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@ -43,9 +42,9 @@ class BangumiCardV extends StatelessWidget {
|
|||||||
// },
|
// },
|
||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
int seasonId = bangumiItem.seasonId;
|
final int seasonId = bangumiItem.seasonId;
|
||||||
SmartDialog.showLoading(msg: '获取中...');
|
SmartDialog.showLoading(msg: '获取中...');
|
||||||
var res = await SearchHttp.bangumiInfo(seasonId: seasonId);
|
final res = await SearchHttp.bangumiInfo(seasonId: seasonId);
|
||||||
SmartDialog.dismiss().then((value) {
|
SmartDialog.dismiss().then((value) {
|
||||||
if (res['status']) {
|
if (res['status']) {
|
||||||
if (res['data'].episodes.isEmpty) {
|
if (res['data'].episodes.isEmpty) {
|
||||||
@ -81,8 +80,8 @@ class BangumiCardV extends StatelessWidget {
|
|||||||
child: AspectRatio(
|
child: AspectRatio(
|
||||||
aspectRatio: 0.65,
|
aspectRatio: 0.65,
|
||||||
child: LayoutBuilder(builder: (context, boxConstraints) {
|
child: LayoutBuilder(builder: (context, boxConstraints) {
|
||||||
double maxWidth = boxConstraints.maxWidth;
|
final double maxWidth = boxConstraints.maxWidth;
|
||||||
double maxHeight = boxConstraints.maxHeight;
|
final double maxHeight = boxConstraints.maxHeight;
|
||||||
return Stack(
|
return Stack(
|
||||||
children: [
|
children: [
|
||||||
Hero(
|
Hero(
|
||||||
@ -124,9 +123,9 @@ class BangumiCardV extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class BangumiContent extends StatelessWidget {
|
class BangumiContent extends StatelessWidget {
|
||||||
|
const BangumiContent({super.key, required this.bangumiItem});
|
||||||
// ignore: prefer_typing_uninitialized_variables
|
// ignore: prefer_typing_uninitialized_variables
|
||||||
final bangumiItem;
|
final bangumiItem;
|
||||||
const BangumiContent({Key? key, required this.bangumiItem}) : super(key: key);
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Expanded(
|
return Expanded(
|
||||||
|
|||||||
@ -70,7 +70,7 @@ class _BlackListPageState extends State<BlackListPage> {
|
|||||||
onRefresh: () async => await _blackListController.queryBlacklist(),
|
onRefresh: () async => await _blackListController.queryBlacklist(),
|
||||||
child: FutureBuilder(
|
child: FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.done) {
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
var data = snapshot.data;
|
var data = snapshot.data;
|
||||||
if (data['status']) {
|
if (data['status']) {
|
||||||
|
|||||||
@ -1,26 +1,23 @@
|
|||||||
import 'package:pilipala/http/danmaku.dart';
|
import 'package:pilipala/http/danmaku.dart';
|
||||||
import 'package:pilipala/models/danmaku/dm.pb.dart';
|
import 'package:pilipala/models/danmaku/dm.pb.dart';
|
||||||
import 'package:pilipala/plugin/pl_player/index.dart';
|
|
||||||
|
|
||||||
class PlDanmakuController {
|
class PlDanmakuController {
|
||||||
PlDanmakuController(this.cid);
|
PlDanmakuController(this.cid);
|
||||||
final int cid;
|
final int cid;
|
||||||
Map<int,List<DanmakuElem>> dmSegMap = {};
|
Map<int, List<DanmakuElem>> dmSegMap = {};
|
||||||
// 已请求的段落标记
|
// 已请求的段落标记
|
||||||
List<bool> requestedSeg = [];
|
List<bool> requestedSeg = [];
|
||||||
|
|
||||||
bool get initiated => requestedSeg.isNotEmpty;
|
bool get initiated => requestedSeg.isNotEmpty;
|
||||||
|
|
||||||
static int SEGMENT_LENGTH = 60 * 6 * 1000;
|
static int segmentLength = 60 * 6 * 1000;
|
||||||
|
|
||||||
void initiate(int videoDuration, int progress) {
|
void initiate(int videoDuration, int progress) {
|
||||||
if (requestedSeg.isEmpty) {
|
if (requestedSeg.isEmpty) {
|
||||||
int segCount = (videoDuration / SEGMENT_LENGTH).ceil();
|
int segCount = (videoDuration / segmentLength).ceil();
|
||||||
requestedSeg = List<bool>.generate(segCount, (index) => false);
|
requestedSeg = List<bool>.generate(segCount, (index) => false);
|
||||||
}
|
}
|
||||||
queryDanmaku(
|
queryDanmaku(calcSegment(progress));
|
||||||
calcSegment(progress)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void dispose() {
|
void dispose() {
|
||||||
@ -29,17 +26,17 @@ class PlDanmakuController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int calcSegment(int progress) {
|
int calcSegment(int progress) {
|
||||||
return progress ~/ SEGMENT_LENGTH;
|
return progress ~/ segmentLength;
|
||||||
}
|
}
|
||||||
|
|
||||||
void queryDanmaku(int segmentIndex) async {
|
void queryDanmaku(int segmentIndex) async {
|
||||||
assert(requestedSeg[segmentIndex] == false);
|
assert(requestedSeg[segmentIndex] == false);
|
||||||
requestedSeg[segmentIndex] = true;
|
requestedSeg[segmentIndex] = true;
|
||||||
DmSegMobileReply result =
|
final DmSegMobileReply result = await DanmakaHttp.queryDanmaku(
|
||||||
await DanmakaHttp.queryDanmaku(cid: cid, segmentIndex: segmentIndex + 1);
|
cid: cid, segmentIndex: segmentIndex + 1);
|
||||||
if (result.elems.isNotEmpty) {
|
if (result.elems.isNotEmpty) {
|
||||||
for (var element in result.elems) {
|
for (var element in result.elems) {
|
||||||
int pos = element.progress ~/ 100;//每0.1秒存储一次
|
int pos = element.progress ~/ 100; //每0.1秒存储一次
|
||||||
if (dmSegMap[pos] == null) {
|
if (dmSegMap[pos] == null) {
|
||||||
dmSegMap[pos] = [];
|
dmSegMap[pos] = [];
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,4 +1,3 @@
|
|||||||
import 'package:easy_debounce/easy_throttle.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
@ -36,6 +35,7 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
late double opacityVal;
|
late double opacityVal;
|
||||||
late double fontSizeVal;
|
late double fontSizeVal;
|
||||||
late double danmakuDurationVal;
|
late double danmakuDurationVal;
|
||||||
|
late double strokeWidth;
|
||||||
int latestAddedPosition = -1;
|
int latestAddedPosition = -1;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -43,15 +43,13 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
super.initState();
|
super.initState();
|
||||||
enableShowDanmaku =
|
enableShowDanmaku =
|
||||||
setting.get(SettingBoxKey.enableShowDanmaku, defaultValue: false);
|
setting.get(SettingBoxKey.enableShowDanmaku, defaultValue: false);
|
||||||
_plDanmakuController =
|
_plDanmakuController = PlDanmakuController(widget.cid);
|
||||||
PlDanmakuController(widget.cid);
|
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
playerController = widget.playerController;
|
playerController = widget.playerController;
|
||||||
if (enableShowDanmaku || playerController.isOpenDanmu.value) {
|
if (enableShowDanmaku || playerController.isOpenDanmu.value) {
|
||||||
_plDanmakuController.initiate(
|
_plDanmakuController.initiate(
|
||||||
playerController.duration.value.inMilliseconds,
|
playerController.duration.value.inMilliseconds,
|
||||||
playerController.position.value.inMilliseconds
|
playerController.position.value.inMilliseconds);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
playerController
|
playerController
|
||||||
..addStatusLister(playerListener)
|
..addStatusLister(playerListener)
|
||||||
@ -61,14 +59,14 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
if (p0 && !_plDanmakuController.initiated) {
|
if (p0 && !_plDanmakuController.initiated) {
|
||||||
_plDanmakuController.initiate(
|
_plDanmakuController.initiate(
|
||||||
playerController.duration.value.inMilliseconds,
|
playerController.duration.value.inMilliseconds,
|
||||||
playerController.position.value.inMilliseconds
|
playerController.position.value.inMilliseconds);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
blockTypes = playerController.blockTypes;
|
blockTypes = playerController.blockTypes;
|
||||||
showArea = playerController.showArea;
|
showArea = playerController.showArea;
|
||||||
opacityVal = playerController.opacityVal;
|
opacityVal = playerController.opacityVal;
|
||||||
fontSizeVal = playerController.fontSizeVal;
|
fontSizeVal = playerController.fontSizeVal;
|
||||||
|
strokeWidth = playerController.strokeWidth;
|
||||||
danmakuDurationVal = playerController.danmakuDurationVal;
|
danmakuDurationVal = playerController.danmakuDurationVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -87,7 +85,7 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
int currentPosition = position.inMilliseconds;
|
int currentPosition = position.inMilliseconds;
|
||||||
currentPosition -= currentPosition % 100;//取整百的毫秒数
|
currentPosition -= currentPosition % 100; //取整百的毫秒数
|
||||||
|
|
||||||
if (currentPosition == latestAddedPosition) {
|
if (currentPosition == latestAddedPosition) {
|
||||||
return;
|
return;
|
||||||
@ -98,17 +96,18 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
_plDanmakuController.getCurrentDanmaku(currentPosition);
|
_plDanmakuController.getCurrentDanmaku(currentPosition);
|
||||||
|
|
||||||
if (currentDanmakuList != null) {
|
if (currentDanmakuList != null) {
|
||||||
Color? defaultColor = playerController.blockTypes.contains(6) ?
|
Color? defaultColor = playerController.blockTypes.contains(6)
|
||||||
DmUtils.decimalToColor(16777215) : null;
|
? DmUtils.decimalToColor(16777215)
|
||||||
|
: null;
|
||||||
|
|
||||||
_controller!.addItems(
|
_controller!.addItems(currentDanmakuList
|
||||||
currentDanmakuList.map((e) => DanmakuItem(
|
.map((e) => DanmakuItem(
|
||||||
e.content,
|
e.content,
|
||||||
color: defaultColor ?? DmUtils.decimalToColor(e.color),
|
color: defaultColor ?? DmUtils.decimalToColor(e.color),
|
||||||
time: e.progress,
|
time: e.progress,
|
||||||
type: DmUtils.getPosition(e.mode),
|
type: DmUtils.getPosition(e.mode),
|
||||||
)).toList()
|
))
|
||||||
);
|
.toList());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -128,7 +127,7 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
duration: const Duration(milliseconds: 100),
|
duration: const Duration(milliseconds: 100),
|
||||||
child: DanmakuView(
|
child: DanmakuView(
|
||||||
createdController: (DanmakuController e) async {
|
createdController: (DanmakuController e) async {
|
||||||
widget.playerController.danmakuController = _controller = e;
|
playerController.danmakuController = _controller = e;
|
||||||
},
|
},
|
||||||
option: DanmakuOption(
|
option: DanmakuOption(
|
||||||
fontSize: 15 * fontSizeVal,
|
fontSize: 15 * fontSizeVal,
|
||||||
@ -137,7 +136,9 @@ class _PlDanmakuState extends State<PlDanmaku> {
|
|||||||
hideTop: blockTypes.contains(5),
|
hideTop: blockTypes.contains(5),
|
||||||
hideScroll: blockTypes.contains(2),
|
hideScroll: blockTypes.contains(2),
|
||||||
hideBottom: blockTypes.contains(4),
|
hideBottom: blockTypes.contains(4),
|
||||||
duration: danmakuDurationVal / widget.playerController.playbackSpeed,
|
duration:
|
||||||
|
danmakuDurationVal / playerController.playbackSpeed,
|
||||||
|
strokeWidth: strokeWidth,
|
||||||
// initDuration /
|
// initDuration /
|
||||||
// (danmakuSpeedVal * widget.playerController.playbackSpeed),
|
// (danmakuSpeedVal * widget.playerController.playbackSpeed),
|
||||||
),
|
),
|
||||||
|
|||||||
@ -37,6 +37,10 @@ class DynamicDetailController extends GetxController {
|
|||||||
}
|
}
|
||||||
int deaultReplySortIndex =
|
int deaultReplySortIndex =
|
||||||
setting.get(SettingBoxKey.replySortType, defaultValue: 0);
|
setting.get(SettingBoxKey.replySortType, defaultValue: 0);
|
||||||
|
if (deaultReplySortIndex == 2) {
|
||||||
|
setting.put(SettingBoxKey.replySortType, 0);
|
||||||
|
deaultReplySortIndex = 0;
|
||||||
|
}
|
||||||
_sortType = ReplySortType.values[deaultReplySortIndex];
|
_sortType = ReplySortType.values[deaultReplySortIndex];
|
||||||
sortTypeTitle.value = _sortType.titles;
|
sortTypeTitle.value = _sortType.titles;
|
||||||
sortTypeLabel.value = _sortType.labels;
|
sortTypeLabel.value = _sortType.labels;
|
||||||
@ -92,9 +96,6 @@ class DynamicDetailController extends GetxController {
|
|||||||
_sortType = ReplySortType.like;
|
_sortType = ReplySortType.like;
|
||||||
break;
|
break;
|
||||||
case ReplySortType.like:
|
case ReplySortType.like:
|
||||||
_sortType = ReplySortType.reply;
|
|
||||||
break;
|
|
||||||
case ReplySortType.reply:
|
|
||||||
_sortType = ReplySortType.time;
|
_sortType = ReplySortType.time;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -8,11 +8,11 @@ import 'package:pilipala/common/skeleton/video_reply.dart';
|
|||||||
import 'package:pilipala/common/widgets/http_error.dart';
|
import 'package:pilipala/common/widgets/http_error.dart';
|
||||||
import 'package:pilipala/models/common/reply_type.dart';
|
import 'package:pilipala/models/common/reply_type.dart';
|
||||||
import 'package:pilipala/models/dynamics/result.dart';
|
import 'package:pilipala/models/dynamics/result.dart';
|
||||||
import 'package:pilipala/pages/dynamics/deatil/index.dart';
|
import 'package:pilipala/pages/dynamics/detail/index.dart';
|
||||||
import 'package:pilipala/pages/dynamics/widgets/author_panel.dart';
|
import 'package:pilipala/pages/dynamics/widgets/author_panel.dart';
|
||||||
import 'package:pilipala/pages/video/detail/reply/widgets/reply_item.dart';
|
import 'package:pilipala/pages/video/detail/reply/widgets/reply_item.dart';
|
||||||
import 'package:pilipala/pages/video/detail/replyNew/index.dart';
|
import 'package:pilipala/pages/video/detail/reply_new/index.dart';
|
||||||
import 'package:pilipala/pages/video/detail/replyReply/index.dart';
|
import 'package:pilipala/pages/video/detail/reply_reply/index.dart';
|
||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
import 'package:pilipala/utils/id_utils.dart';
|
import 'package:pilipala/utils/id_utils.dart';
|
||||||
|
|
||||||
@ -192,22 +192,6 @@ class _DynamicsPageState extends State<DynamicsPage>
|
|||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
// Obx(
|
|
||||||
// () => Visibility(
|
|
||||||
// visible: _dynamicsController.userLogin.value,
|
|
||||||
// child: Positioned(
|
|
||||||
// right: 4,
|
|
||||||
// top: 0,
|
|
||||||
// bottom: 0,
|
|
||||||
// child: IconButton(
|
|
||||||
// padding: EdgeInsets.zero,
|
|
||||||
// onPressed: () =>
|
|
||||||
// {feedBack(), _dynamicsController.resetSearch()},
|
|
||||||
// icon: const Icon(Icons.history, size: 21),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -229,7 +213,8 @@ class _DynamicsPageState extends State<DynamicsPage>
|
|||||||
return Obx(() => UpPanel(_dynamicsController.upData.value));
|
return Obx(() => UpPanel(_dynamicsController.upData.value));
|
||||||
} else {
|
} else {
|
||||||
return const SliverToBoxAdapter(
|
return const SliverToBoxAdapter(
|
||||||
child: SizedBox(height: 80));
|
child: SizedBox(height: 80),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return const SliverToBoxAdapter(
|
return const SliverToBoxAdapter(
|
||||||
@ -240,15 +225,6 @@ class _DynamicsPageState extends State<DynamicsPage>
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
SliverToBoxAdapter(
|
|
||||||
child: Container(
|
|
||||||
height: 6,
|
|
||||||
color: Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.onInverseSurface
|
|
||||||
.withOpacity(0.5),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
FutureBuilder(
|
FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/common/widgets/badge.dart';
|
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/http/user.dart';
|
import 'package:pilipala/http/user.dart';
|
||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
|
|||||||
@ -1,5 +1,7 @@
|
|||||||
// 内容
|
// 内容
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/common/widgets/badge.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/models/dynamics/result.dart';
|
import 'package:pilipala/models/dynamics/result.dart';
|
||||||
import 'package:pilipala/pages/preview/index.dart';
|
import 'package:pilipala/pages/preview/index.dart';
|
||||||
@ -48,6 +50,13 @@ class _ContentState extends State<Content> {
|
|||||||
WidgetSpan(
|
WidgetSpan(
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, BoxConstraints box) {
|
builder: (context, BoxConstraints box) {
|
||||||
|
double maxWidth = box.maxWidth.truncateToDouble();
|
||||||
|
double maxHeight = box.maxWidth * 0.6; // 设置最大高度
|
||||||
|
double height = maxWidth *
|
||||||
|
0.5 *
|
||||||
|
(pictureItem.height != null && pictureItem.width != null
|
||||||
|
? pictureItem.height! / pictureItem.width!
|
||||||
|
: 1);
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
showDialog(
|
showDialog(
|
||||||
@ -58,18 +67,29 @@ class _ContentState extends State<Content> {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
child: Padding(
|
child: Container(
|
||||||
padding: const EdgeInsets.only(top: 4),
|
padding: const EdgeInsets.only(top: 4),
|
||||||
child: NetworkImgLayer(
|
constraints: BoxConstraints(maxHeight: maxHeight),
|
||||||
src: pictureItem.url,
|
|
||||||
width: box.maxWidth / 2,
|
width: box.maxWidth / 2,
|
||||||
height: box.maxWidth *
|
height: height,
|
||||||
0.5 *
|
child: Stack(
|
||||||
(pictureItem.height != null && pictureItem.width != null
|
children: [
|
||||||
? pictureItem.height! / pictureItem.width!
|
Positioned.fill(
|
||||||
: 1),
|
child: NetworkImgLayer(
|
||||||
),
|
src: pictureItem.url,
|
||||||
),
|
width: maxWidth / 2,
|
||||||
|
height: height,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
height > Get.size.height * 0.9
|
||||||
|
? const PBadge(
|
||||||
|
text: '长图',
|
||||||
|
right: 8,
|
||||||
|
bottom: 8,
|
||||||
|
)
|
||||||
|
: const SizedBox(),
|
||||||
|
],
|
||||||
|
)),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@ -83,6 +103,7 @@ class _ContentState extends State<Content> {
|
|||||||
list.add(
|
list.add(
|
||||||
LayoutBuilder(
|
LayoutBuilder(
|
||||||
builder: (context, BoxConstraints box) {
|
builder: (context, BoxConstraints box) {
|
||||||
|
double maxWidth = box.maxWidth.truncateToDouble();
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
showDialog(
|
showDialog(
|
||||||
@ -95,8 +116,10 @@ class _ContentState extends State<Content> {
|
|||||||
},
|
},
|
||||||
child: NetworkImgLayer(
|
child: NetworkImgLayer(
|
||||||
src: pics[i].url,
|
src: pics[i].url,
|
||||||
width: box.maxWidth,
|
width: maxWidth,
|
||||||
height: box.maxWidth,
|
height: maxWidth,
|
||||||
|
origAspectRatio:
|
||||||
|
pics[i].width!.toInt() / pics[i].height!.toInt(),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -107,7 +130,7 @@ class _ContentState extends State<Content> {
|
|||||||
WidgetSpan(
|
WidgetSpan(
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, BoxConstraints box) {
|
builder: (context, BoxConstraints box) {
|
||||||
double maxWidth = box.maxWidth;
|
double maxWidth = box.maxWidth.truncateToDouble();
|
||||||
double crossCount = len < 3 ? 2 : 3;
|
double crossCount = len < 3 ? 2 : 3;
|
||||||
double height = maxWidth /
|
double height = maxWidth /
|
||||||
crossCount *
|
crossCount *
|
||||||
|
|||||||
@ -1,4 +1,5 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/common/constants.dart';
|
import 'package:pilipala/common/constants.dart';
|
||||||
import 'package:pilipala/common/widgets/badge.dart';
|
import 'package:pilipala/common/widgets/badge.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
@ -87,7 +88,7 @@ Widget picWidget(item, context) {
|
|||||||
childAspectRatio: aspectRatio,
|
childAspectRatio: aspectRatio,
|
||||||
children: list,
|
children: list,
|
||||||
),
|
),
|
||||||
if (len == 1 && origAspectRatio < 0.4)
|
if (len == 1 && height > Get.size.height * 0.9)
|
||||||
const PBadge(
|
const PBadge(
|
||||||
text: '长图',
|
text: '长图',
|
||||||
top: null,
|
top: null,
|
||||||
|
|||||||
@ -36,8 +36,7 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
}
|
}
|
||||||
upList.insert(
|
upList.insert(
|
||||||
0,
|
0,
|
||||||
UpItem(
|
UpItem(face: '', uname: '全部动态', mid: -1),
|
||||||
face: 'https://files.catbox.moe/8uc48f.png', uname: '全部动态', mid: -1),
|
|
||||||
);
|
);
|
||||||
userInfo = userInfoCache.get('userInfoCache');
|
userInfo = userInfoCache.get('userInfoCache');
|
||||||
upList.insert(
|
upList.insert(
|
||||||
@ -56,7 +55,7 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
floating: true,
|
floating: true,
|
||||||
pinned: false,
|
pinned: false,
|
||||||
delegate: _SliverHeaderDelegate(
|
delegate: _SliverHeaderDelegate(
|
||||||
height: 124,
|
height: 126,
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
@ -121,6 +120,13 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
Container(
|
||||||
|
height: 6,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onInverseSurface
|
||||||
|
.withOpacity(0.5),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)),
|
)),
|
||||||
);
|
);
|
||||||
@ -139,7 +145,7 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
int liveLen = liveList.length;
|
int liveLen = liveList.length;
|
||||||
int upLen = upList.length;
|
int upLen = upList.length;
|
||||||
double itemWidth = contentWidth + itemPadding.horizontal;
|
double itemWidth = contentWidth + itemPadding.horizontal;
|
||||||
double screenWidth = MediaQuery.of(context).size.width;
|
double screenWidth = MediaQuery.sizeOf(context).width;
|
||||||
double moveDistance = 0.0;
|
double moveDistance = 0.0;
|
||||||
if (itemWidth * (upList.length + liveList.length) <= screenWidth) {
|
if (itemWidth * (upList.length + liveList.length) <= screenWidth) {
|
||||||
} else if ((upLen - i - 0.5) * itemWidth > screenWidth / 2) {
|
} else if ((upLen - i - 0.5) * itemWidth > screenWidth / 2) {
|
||||||
@ -171,6 +177,9 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
},
|
},
|
||||||
onLongPress: () {
|
onLongPress: () {
|
||||||
feedBack();
|
feedBack();
|
||||||
|
if (data.mid == -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
String heroTag = Utils.makeHeroTag(data.mid);
|
String heroTag = Utils.makeHeroTag(data.mid);
|
||||||
Get.toNamed('/member?mid=${data.mid}',
|
Get.toNamed('/member?mid=${data.mid}',
|
||||||
arguments: {'face': data.face, 'heroTag': heroTag});
|
arguments: {'face': data.face, 'heroTag': heroTag});
|
||||||
@ -198,12 +207,19 @@ class _UpPanelState extends State<UpPanel> {
|
|||||||
backgroundColor: data.type == 'live'
|
backgroundColor: data.type == 'live'
|
||||||
? Theme.of(context).colorScheme.secondaryContainer
|
? Theme.of(context).colorScheme.secondaryContainer
|
||||||
: Theme.of(context).colorScheme.primary,
|
: Theme.of(context).colorScheme.primary,
|
||||||
child: NetworkImgLayer(
|
child: data.face != ''
|
||||||
width: 49,
|
? NetworkImgLayer(
|
||||||
height: 49,
|
width: 50,
|
||||||
src: data.face,
|
height: 50,
|
||||||
type: 'avatar',
|
src: data.face,
|
||||||
),
|
type: 'avatar',
|
||||||
|
)
|
||||||
|
: const CircleAvatar(
|
||||||
|
radius: 25,
|
||||||
|
backgroundImage: AssetImage(
|
||||||
|
'assets/images/noface.jpeg',
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(top: 4),
|
padding: const EdgeInsets.only(top: 4),
|
||||||
@ -271,13 +287,11 @@ class UpPanelSkeleton extends StatelessWidget {
|
|||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
Container(
|
Container(
|
||||||
width: 49,
|
width: 50,
|
||||||
height: 49,
|
height: 50,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Theme.of(context).colorScheme.onInverseSurface,
|
color: Theme.of(context).colorScheme.onInverseSurface,
|
||||||
borderRadius: const BorderRadius.all(
|
borderRadius: BorderRadius.circular(50),
|
||||||
Radius.circular(24),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Container(
|
Container(
|
||||||
|
|||||||
@ -1,216 +0,0 @@
|
|||||||
import 'package:get/get.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:pilipala/common/constants.dart';
|
|
||||||
import 'package:pilipala/common/widgets/stat/danmu.dart';
|
|
||||||
import 'package:pilipala/common/widgets/stat/view.dart';
|
|
||||||
import 'package:pilipala/http/search.dart';
|
|
||||||
import 'package:pilipala/http/video.dart';
|
|
||||||
import 'package:pilipala/models/common/search_type.dart';
|
|
||||||
import 'package:pilipala/utils/id_utils.dart';
|
|
||||||
import 'package:pilipala/utils/utils.dart';
|
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
|
||||||
|
|
||||||
// 收藏视频卡片 - 水平布局
|
|
||||||
class FavVideoCardH extends StatelessWidget {
|
|
||||||
final dynamic videoItem;
|
|
||||||
final Function? callFn;
|
|
||||||
|
|
||||||
const FavVideoCardH({Key? key, required this.videoItem, this.callFn})
|
|
||||||
: super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
int id = videoItem.id;
|
|
||||||
String bvid = videoItem.bvid ?? IdUtils.av2bv(id);
|
|
||||||
String heroTag = Utils.makeHeroTag(id);
|
|
||||||
return InkWell(
|
|
||||||
onTap: () async {
|
|
||||||
// int? seasonId;
|
|
||||||
String? epId;
|
|
||||||
if (videoItem.ogv != null && videoItem.ogv['type_name'] == '番剧') {
|
|
||||||
videoItem.cid = await SearchHttp.ab2c(bvid: bvid);
|
|
||||||
// seasonId = videoItem.ogv['season_id'];
|
|
||||||
epId = videoItem.epId;
|
|
||||||
} else if (videoItem.page == 0 || videoItem.page > 1) {
|
|
||||||
var result = await VideoHttp.videoIntro(bvid: bvid);
|
|
||||||
if (result['status']) {
|
|
||||||
epId = result['data'].epId;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Map<String, String> parameters = {
|
|
||||||
'bvid': bvid,
|
|
||||||
'cid': videoItem.cid.toString(),
|
|
||||||
'epId': epId ?? '',
|
|
||||||
};
|
|
||||||
// if (seasonId != null) {
|
|
||||||
// parameters['seasonId'] = seasonId.toString();
|
|
||||||
// }
|
|
||||||
Get.toNamed('/video', parameters: parameters, arguments: {
|
|
||||||
'videoItem': videoItem,
|
|
||||||
'heroTag': heroTag,
|
|
||||||
'videoType':
|
|
||||||
epId != null ? SearchType.media_bangumi : SearchType.video,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.fromLTRB(
|
|
||||||
StyleString.safeSpace, 5, StyleString.safeSpace, 5),
|
|
||||||
child: LayoutBuilder(
|
|
||||||
builder: (context, boxConstraints) {
|
|
||||||
double width =
|
|
||||||
(boxConstraints.maxWidth - StyleString.cardSpace * 6) / 2;
|
|
||||||
return SizedBox(
|
|
||||||
height: width / StyleString.aspectRatio,
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
AspectRatio(
|
|
||||||
aspectRatio: StyleString.aspectRatio,
|
|
||||||
child: LayoutBuilder(
|
|
||||||
builder: (context, boxConstraints) {
|
|
||||||
double maxWidth = boxConstraints.maxWidth;
|
|
||||||
double maxHeight = boxConstraints.maxHeight;
|
|
||||||
return Stack(
|
|
||||||
children: [
|
|
||||||
Hero(
|
|
||||||
tag: heroTag,
|
|
||||||
child: NetworkImgLayer(
|
|
||||||
src: videoItem.pic,
|
|
||||||
width: maxWidth,
|
|
||||||
height: maxHeight,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
right: 4,
|
|
||||||
bottom: 4,
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.symmetric(
|
|
||||||
vertical: 1, horizontal: 6),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(4),
|
|
||||||
color: Colors.black54.withOpacity(0.4)),
|
|
||||||
child: Text(
|
|
||||||
Utils.timeFormat(videoItem.duration!),
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 11, color: Colors.white),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
VideoContent(videoItem: videoItem, callFn: callFn)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class VideoContent extends StatelessWidget {
|
|
||||||
final dynamic videoItem;
|
|
||||||
final Function? callFn;
|
|
||||||
const VideoContent({super.key, required this.videoItem, this.callFn});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Expanded(
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.fromLTRB(10, 2, 6, 0),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
videoItem.title,
|
|
||||||
textAlign: TextAlign.start,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
letterSpacing: 0.3,
|
|
||||||
),
|
|
||||||
maxLines: 2,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
const Spacer(),
|
|
||||||
Text(
|
|
||||||
Utils.dateFormat(videoItem.ctime!),
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 11, color: Theme.of(context).colorScheme.outline),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
videoItem.owner.name,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: Theme.of(context).textTheme.labelMedium!.fontSize,
|
|
||||||
color: Theme.of(context).colorScheme.outline,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
StatView(
|
|
||||||
theme: 'gray',
|
|
||||||
view: videoItem.cntInfo['play'],
|
|
||||||
),
|
|
||||||
const SizedBox(width: 8),
|
|
||||||
StatDanMu(theme: 'gray', danmu: videoItem.cntInfo['danmaku']),
|
|
||||||
const Spacer(),
|
|
||||||
SizedBox(
|
|
||||||
width: 26,
|
|
||||||
height: 26,
|
|
||||||
child: IconButton(
|
|
||||||
style: ButtonStyle(
|
|
||||||
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
|
||||||
),
|
|
||||||
onPressed: () {
|
|
||||||
showDialog(
|
|
||||||
context: Get.context!,
|
|
||||||
builder: (context) {
|
|
||||||
return AlertDialog(
|
|
||||||
title: const Text('提示'),
|
|
||||||
content: const Text('要取消收藏吗?'),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Get.back(),
|
|
||||||
child: Text(
|
|
||||||
'取消',
|
|
||||||
style: TextStyle(
|
|
||||||
color: Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.outline),
|
|
||||||
)),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () async {
|
|
||||||
await callFn!();
|
|
||||||
Get.back();
|
|
||||||
},
|
|
||||||
child: const Text('确定取消'),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.clear_outlined,
|
|
||||||
color: Theme.of(context).colorScheme.outline,
|
|
||||||
size: 18,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -7,7 +7,7 @@ import 'package:pilipala/common/skeleton/video_card_h.dart';
|
|||||||
import 'package:pilipala/common/widgets/http_error.dart';
|
import 'package:pilipala/common/widgets/http_error.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/common/widgets/no_data.dart';
|
import 'package:pilipala/common/widgets/no_data.dart';
|
||||||
import 'package:pilipala/pages/favDetail/index.dart';
|
import 'package:pilipala/pages/fav_detail/index.dart';
|
||||||
|
|
||||||
import 'widget/fav_video_card.dart';
|
import 'widget/fav_video_card.dart';
|
||||||
|
|
||||||
@ -24,11 +24,13 @@ class _FavDetailPageState extends State<FavDetailPage> {
|
|||||||
Get.put(FavDetailController());
|
Get.put(FavDetailController());
|
||||||
late StreamController<bool> titleStreamC; // a
|
late StreamController<bool> titleStreamC; // a
|
||||||
Future? _futureBuilderFuture;
|
Future? _futureBuilderFuture;
|
||||||
|
late String mediaId;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_futureBuilderFuture = _favDetailController.queryUserFavFolderDetail();
|
_futureBuilderFuture = _favDetailController.queryUserFavFolderDetail();
|
||||||
|
mediaId = Get.parameters['mediaId']!;
|
||||||
titleStreamC = StreamController<bool>();
|
titleStreamC = StreamController<bool>();
|
||||||
_controller.addListener(
|
_controller.addListener(
|
||||||
() {
|
() {
|
||||||
@ -94,8 +96,8 @@ class _FavDetailPageState extends State<FavDetailPage> {
|
|||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () => Get.toNamed(
|
onPressed: () =>
|
||||||
'/favSearch?searchType=0&mediaId=${Get.parameters['mediaId']!}'),
|
Get.toNamed('/favSearch?searchType=0&mediaId=$mediaId'),
|
||||||
icon: const Icon(Icons.search_outlined),
|
icon: const Icon(Icons.search_outlined),
|
||||||
),
|
),
|
||||||
// IconButton(
|
// IconButton(
|
||||||
257
lib/pages/fav_detail/widget/fav_video_card.dart
Normal file
257
lib/pages/fav_detail/widget/fav_video_card.dart
Normal file
@ -0,0 +1,257 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:pilipala/common/constants.dart';
|
||||||
|
import 'package:pilipala/common/widgets/stat/danmu.dart';
|
||||||
|
import 'package:pilipala/common/widgets/stat/view.dart';
|
||||||
|
import 'package:pilipala/http/search.dart';
|
||||||
|
import 'package:pilipala/http/video.dart';
|
||||||
|
import 'package:pilipala/models/common/search_type.dart';
|
||||||
|
import 'package:pilipala/utils/id_utils.dart';
|
||||||
|
import 'package:pilipala/utils/utils.dart';
|
||||||
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
|
import '../../../common/widgets/badge.dart';
|
||||||
|
|
||||||
|
// 收藏视频卡片 - 水平布局
|
||||||
|
class FavVideoCardH extends StatelessWidget {
|
||||||
|
final dynamic videoItem;
|
||||||
|
final Function? callFn;
|
||||||
|
final int? searchType;
|
||||||
|
|
||||||
|
const FavVideoCardH({
|
||||||
|
Key? key,
|
||||||
|
required this.videoItem,
|
||||||
|
this.callFn,
|
||||||
|
this.searchType,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
int id = videoItem.id;
|
||||||
|
String bvid = videoItem.bvid ?? IdUtils.av2bv(id);
|
||||||
|
String heroTag = Utils.makeHeroTag(id);
|
||||||
|
return InkWell(
|
||||||
|
onTap: () async {
|
||||||
|
// int? seasonId;
|
||||||
|
String? epId;
|
||||||
|
if (videoItem.ogv != null &&
|
||||||
|
(videoItem.ogv['type_name'] == '番剧' ||
|
||||||
|
videoItem.ogv['type_name'] == '国创')) {
|
||||||
|
videoItem.cid = await SearchHttp.ab2c(bvid: bvid);
|
||||||
|
// seasonId = videoItem.ogv['season_id'];
|
||||||
|
epId = videoItem.epId;
|
||||||
|
} else if (videoItem.page == 0 || videoItem.page > 1) {
|
||||||
|
var result = await VideoHttp.videoIntro(bvid: bvid);
|
||||||
|
if (result['status']) {
|
||||||
|
epId = result['data'].epId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, String> parameters = {
|
||||||
|
'bvid': bvid,
|
||||||
|
'cid': videoItem.cid.toString(),
|
||||||
|
'epId': epId ?? '',
|
||||||
|
};
|
||||||
|
// if (seasonId != null) {
|
||||||
|
// parameters['seasonId'] = seasonId.toString();
|
||||||
|
// }
|
||||||
|
Get.toNamed('/video', parameters: parameters, arguments: {
|
||||||
|
'videoItem': videoItem,
|
||||||
|
'heroTag': heroTag,
|
||||||
|
'videoType':
|
||||||
|
epId != null ? SearchType.media_bangumi : SearchType.video,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(
|
||||||
|
StyleString.safeSpace, 5, StyleString.safeSpace, 5),
|
||||||
|
child: LayoutBuilder(
|
||||||
|
builder: (context, boxConstraints) {
|
||||||
|
double width =
|
||||||
|
(boxConstraints.maxWidth - StyleString.cardSpace * 6) / 2;
|
||||||
|
return SizedBox(
|
||||||
|
height: width / StyleString.aspectRatio,
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
AspectRatio(
|
||||||
|
aspectRatio: StyleString.aspectRatio,
|
||||||
|
child: LayoutBuilder(
|
||||||
|
builder: (context, boxConstraints) {
|
||||||
|
double maxWidth = boxConstraints.maxWidth;
|
||||||
|
double maxHeight = boxConstraints.maxHeight;
|
||||||
|
return Stack(
|
||||||
|
children: [
|
||||||
|
Hero(
|
||||||
|
tag: heroTag,
|
||||||
|
child: NetworkImgLayer(
|
||||||
|
src: videoItem.pic,
|
||||||
|
width: maxWidth,
|
||||||
|
height: maxHeight,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
PBadge(
|
||||||
|
text: Utils.timeFormat(videoItem.duration!),
|
||||||
|
right: 6.0,
|
||||||
|
bottom: 6.0,
|
||||||
|
type: 'gray',
|
||||||
|
),
|
||||||
|
if (videoItem.ogv != null) ...[
|
||||||
|
PBadge(
|
||||||
|
text: videoItem.ogv['type_name'],
|
||||||
|
top: 6.0,
|
||||||
|
right: 6.0,
|
||||||
|
bottom: null,
|
||||||
|
left: null,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
VideoContent(
|
||||||
|
videoItem: videoItem,
|
||||||
|
callFn: callFn,
|
||||||
|
searchType: searchType,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class VideoContent extends StatelessWidget {
|
||||||
|
final dynamic videoItem;
|
||||||
|
final Function? callFn;
|
||||||
|
final int? searchType;
|
||||||
|
const VideoContent({
|
||||||
|
super.key,
|
||||||
|
required this.videoItem,
|
||||||
|
this.callFn,
|
||||||
|
this.searchType,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Expanded(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(10, 2, 6, 0),
|
||||||
|
child: Stack(
|
||||||
|
children: [
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
videoItem.title,
|
||||||
|
textAlign: TextAlign.start,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontWeight: FontWeight.w500,
|
||||||
|
letterSpacing: 0.3,
|
||||||
|
),
|
||||||
|
maxLines: 2,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
if (videoItem.ogv != null) ...[
|
||||||
|
Text(
|
||||||
|
videoItem.intro,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize:
|
||||||
|
Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
const Spacer(),
|
||||||
|
Text(
|
||||||
|
Utils.dateFormat(videoItem.favTime),
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 11,
|
||||||
|
color: Theme.of(context).colorScheme.outline),
|
||||||
|
),
|
||||||
|
if (videoItem.owner.name != '') ...[
|
||||||
|
Text(
|
||||||
|
videoItem.owner.name,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize:
|
||||||
|
Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 2),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
StatView(
|
||||||
|
theme: 'gray',
|
||||||
|
view: videoItem.cntInfo['play'],
|
||||||
|
),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
StatDanMu(
|
||||||
|
theme: 'gray', danmu: videoItem.cntInfo['danmaku']),
|
||||||
|
const Spacer(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
searchType != 1
|
||||||
|
? Positioned(
|
||||||
|
right: 0,
|
||||||
|
bottom: -4,
|
||||||
|
child: IconButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
showDialog(
|
||||||
|
context: Get.context!,
|
||||||
|
builder: (context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('提示'),
|
||||||
|
content: const Text('要取消收藏吗?'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Get.back(),
|
||||||
|
child: Text(
|
||||||
|
'取消',
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.outline),
|
||||||
|
)),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
await callFn!();
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
child: const Text('确定取消'),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
icon: Icon(
|
||||||
|
Icons.clear_outlined,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
size: 18,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: const SizedBox(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,8 +1,11 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/http/user.dart';
|
import 'package:pilipala/http/user.dart';
|
||||||
import 'package:pilipala/models/user/fav_detail.dart';
|
import 'package:pilipala/models/user/fav_detail.dart';
|
||||||
|
|
||||||
|
import '../../http/video.dart';
|
||||||
|
|
||||||
class FavSearchController extends GetxController {
|
class FavSearchController extends GetxController {
|
||||||
final ScrollController scrollController = ScrollController();
|
final ScrollController scrollController = ScrollController();
|
||||||
Rx<TextEditingController> controller = TextEditingController().obs;
|
Rx<TextEditingController> controller = TextEditingController().obs;
|
||||||
@ -72,4 +75,21 @@ class FavSearchController extends GetxController {
|
|||||||
if (!hasMore) return;
|
if (!hasMore) return;
|
||||||
searchFav(type: 'onLoad');
|
searchFav(type: 'onLoad');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onCancelFav(int id) async {
|
||||||
|
var result = await VideoHttp.favVideo(
|
||||||
|
aid: id, addIds: '', delIds: mediaId.toString());
|
||||||
|
if (result['status']) {
|
||||||
|
if (result['data']['prompt']) {
|
||||||
|
List dataList = favList;
|
||||||
|
for (var i in dataList) {
|
||||||
|
if (i.id == id) {
|
||||||
|
dataList.remove(i);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SmartDialog.showToast('取消收藏');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -3,14 +3,12 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/common/skeleton/video_card_h.dart';
|
import 'package:pilipala/common/skeleton/video_card_h.dart';
|
||||||
import 'package:pilipala/common/widgets/no_data.dart';
|
import 'package:pilipala/common/widgets/no_data.dart';
|
||||||
import 'package:pilipala/pages/favDetail/widget/fav_video_card.dart';
|
import 'package:pilipala/pages/fav_detail/widget/fav_video_card.dart';
|
||||||
|
|
||||||
import 'controller.dart';
|
import 'controller.dart';
|
||||||
|
|
||||||
class FavSearchPage extends StatefulWidget {
|
class FavSearchPage extends StatefulWidget {
|
||||||
final int? sourceType;
|
const FavSearchPage({super.key});
|
||||||
final int? mediaId;
|
|
||||||
const FavSearchPage({super.key, this.sourceType, this.mediaId});
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<FavSearchPage> createState() => _FavSearchPageState();
|
State<FavSearchPage> createState() => _FavSearchPageState();
|
||||||
@ -19,11 +17,12 @@ class FavSearchPage extends StatefulWidget {
|
|||||||
class _FavSearchPageState extends State<FavSearchPage> {
|
class _FavSearchPageState extends State<FavSearchPage> {
|
||||||
final FavSearchController _favSearchCtr = Get.put(FavSearchController());
|
final FavSearchController _favSearchCtr = Get.put(FavSearchController());
|
||||||
late ScrollController scrollController;
|
late ScrollController scrollController;
|
||||||
|
late int searchType;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
searchType = int.parse(Get.parameters['searchType']!);
|
||||||
scrollController = _favSearchCtr.scrollController;
|
scrollController = _favSearchCtr.scrollController;
|
||||||
scrollController.addListener(
|
scrollController.addListener(
|
||||||
() {
|
() {
|
||||||
@ -100,7 +99,11 @@ class _FavSearchPageState extends State<FavSearchPage> {
|
|||||||
} else {
|
} else {
|
||||||
return FavVideoCardH(
|
return FavVideoCardH(
|
||||||
videoItem: _favSearchCtr.favList[index],
|
videoItem: _favSearchCtr.favList[index],
|
||||||
callFn: () => null,
|
searchType: searchType,
|
||||||
|
callFn: () => searchType != 1
|
||||||
|
? _favSearchCtr
|
||||||
|
.onCancelFav(_favSearchCtr.favList[index].id!)
|
||||||
|
: {},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@ -37,6 +37,29 @@ class _FollowPageState extends State<FollowPage> {
|
|||||||
: '${_followController.name}的关注',
|
: '${_followController.name}的关注',
|
||||||
style: Theme.of(context).textTheme.titleMedium,
|
style: Theme.of(context).textTheme.titleMedium,
|
||||||
),
|
),
|
||||||
|
actions: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () => Get.toNamed('/followSearch?mid=$mid'),
|
||||||
|
icon: const Icon(Icons.search_outlined),
|
||||||
|
),
|
||||||
|
PopupMenuButton(
|
||||||
|
icon: const Icon(Icons.more_vert),
|
||||||
|
itemBuilder: (BuildContext context) => <PopupMenuEntry>[
|
||||||
|
PopupMenuItem(
|
||||||
|
onTap: () => Get.toNamed('/blackListPage'),
|
||||||
|
child: const Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
Icon(Icons.block, size: 19),
|
||||||
|
SizedBox(width: 10),
|
||||||
|
Text('黑名单管理'),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
body: Obx(
|
body: Obx(
|
||||||
() => !_followController.isOwner.value
|
() => !_followController.isOwner.value
|
||||||
@ -87,3 +110,22 @@ class _FollowPageState extends State<FollowPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _FakeAPI {
|
||||||
|
static const List<String> _kOptions = <String>[
|
||||||
|
'aardvark',
|
||||||
|
'bobcat',
|
||||||
|
'chameleon',
|
||||||
|
];
|
||||||
|
// Searches the options, but injects a fake "network" delay.
|
||||||
|
static Future<Iterable<String>> search(String query) async {
|
||||||
|
await Future<void>.delayed(
|
||||||
|
const Duration(seconds: 1)); // Fake 1 second delay.
|
||||||
|
if (query == '') {
|
||||||
|
return const Iterable<String>.empty();
|
||||||
|
}
|
||||||
|
return _kOptions.where((String option) {
|
||||||
|
return option.contains(query.toLowerCase());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -42,7 +42,7 @@ class FollowItem extends StatelessWidget {
|
|||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
dense: true,
|
dense: true,
|
||||||
trailing: ctr!.isOwner.value
|
trailing: ctr != null && ctr!.isOwner.value
|
||||||
? SizedBox(
|
? SizedBox(
|
||||||
height: 34,
|
height: 34,
|
||||||
child: TextButton(
|
child: TextButton(
|
||||||
|
|||||||
73
lib/pages/follow_search/controller.dart
Normal file
73
lib/pages/follow_search/controller.dart
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/http/member.dart';
|
||||||
|
|
||||||
|
import '../../models/follow/result.dart';
|
||||||
|
|
||||||
|
class FollowSearchController extends GetxController {
|
||||||
|
Rx<TextEditingController> controller = TextEditingController().obs;
|
||||||
|
final FocusNode searchFocusNode = FocusNode();
|
||||||
|
RxString searchKeyWord = ''.obs;
|
||||||
|
String hintText = '搜索';
|
||||||
|
RxString loadingStatus = 'init'.obs;
|
||||||
|
late int mid = 1;
|
||||||
|
RxString uname = ''.obs;
|
||||||
|
int ps = 20;
|
||||||
|
int pn = 1;
|
||||||
|
RxList<FollowItemModel> followList = <FollowItemModel>[].obs;
|
||||||
|
RxInt total = 0.obs;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onInit() {
|
||||||
|
super.onInit();
|
||||||
|
mid = int.parse(Get.parameters['mid']!);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 清空搜索
|
||||||
|
void onClear() {
|
||||||
|
if (searchKeyWord.value.isNotEmpty && controller.value.text != '') {
|
||||||
|
controller.value.clear();
|
||||||
|
searchKeyWord.value = '';
|
||||||
|
} else {
|
||||||
|
Get.back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void onChange(value) {
|
||||||
|
searchKeyWord.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 提交搜索内容
|
||||||
|
void submit() {
|
||||||
|
loadingStatus.value = 'loading';
|
||||||
|
searchFollow();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future searchFollow({type = 'init'}) async {
|
||||||
|
if (controller.value.text == '') {
|
||||||
|
return {'status': true, 'data': <FollowItemModel>[].obs};
|
||||||
|
}
|
||||||
|
if (type == 'init') {
|
||||||
|
ps = 1;
|
||||||
|
}
|
||||||
|
var res = await MemberHttp.getfollowSearch(
|
||||||
|
mid: mid,
|
||||||
|
ps: ps,
|
||||||
|
pn: pn,
|
||||||
|
name: controller.value.text,
|
||||||
|
);
|
||||||
|
if (res['status']) {
|
||||||
|
if (type == 'init') {
|
||||||
|
followList.value = res['data'].list;
|
||||||
|
} else {
|
||||||
|
followList.addAll(res['data'].list);
|
||||||
|
}
|
||||||
|
total.value = res['data'].total;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
void onLoad() {
|
||||||
|
searchFollow(type: 'onLoad');
|
||||||
|
}
|
||||||
|
}
|
||||||
4
lib/pages/follow_search/index.dart
Normal file
4
lib/pages/follow_search/index.dart
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
library follow_search;
|
||||||
|
|
||||||
|
export './controller.dart';
|
||||||
|
export './view.dart';
|
||||||
121
lib/pages/follow_search/view.dart
Normal file
121
lib/pages/follow_search/view.dart
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
import 'package:easy_debounce/easy_throttle.dart';
|
||||||
|
import 'package:flutter/cupertino.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/common/widgets/http_error.dart';
|
||||||
|
import 'package:pilipala/pages/follow_search/index.dart';
|
||||||
|
|
||||||
|
import '../follow/widgets/follow_item.dart';
|
||||||
|
|
||||||
|
class FollowSearchPage extends StatefulWidget {
|
||||||
|
const FollowSearchPage({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<FollowSearchPage> createState() => _FollowSearchPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _FollowSearchPageState extends State<FollowSearchPage> {
|
||||||
|
final FollowSearchController _followSearchController =
|
||||||
|
Get.put(FollowSearchController());
|
||||||
|
late Future? _futureBuilder;
|
||||||
|
final ScrollController scrollController = ScrollController();
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_futureBuilder = _followSearchController.searchFollow();
|
||||||
|
scrollController.addListener(
|
||||||
|
() {
|
||||||
|
if (scrollController.position.pixels >=
|
||||||
|
scrollController.position.maxScrollExtent - 200) {
|
||||||
|
EasyThrottle.throttle(
|
||||||
|
'my-throttler', const Duration(milliseconds: 500), () {
|
||||||
|
_followSearchController.onLoad();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void reRequest() {
|
||||||
|
setState(() {
|
||||||
|
_futureBuilder = _followSearchController.searchFollow();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
titleSpacing: 0,
|
||||||
|
actions: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: reRequest,
|
||||||
|
icon: const Icon(CupertinoIcons.search, size: 22),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
],
|
||||||
|
title: TextField(
|
||||||
|
autofocus: true,
|
||||||
|
focusNode: _followSearchController.searchFocusNode,
|
||||||
|
controller: _followSearchController.controller.value,
|
||||||
|
textInputAction: TextInputAction.search,
|
||||||
|
onChanged: (value) => _followSearchController.onChange(value),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
hintText: _followSearchController.hintText,
|
||||||
|
border: InputBorder.none,
|
||||||
|
suffixIcon: IconButton(
|
||||||
|
icon: Icon(
|
||||||
|
Icons.clear,
|
||||||
|
size: 22,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
onPressed: () => _followSearchController.onClear(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSubmitted: (String value) => reRequest(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: FutureBuilder(
|
||||||
|
future: _futureBuilder,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
|
var data = snapshot.data;
|
||||||
|
if (data == null) {
|
||||||
|
return CustomScrollView(
|
||||||
|
slivers: [
|
||||||
|
HttpError(errMsg: snapshot.data['msg'], fn: reRequest)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (data['status']) {
|
||||||
|
RxList followList = _followSearchController.followList;
|
||||||
|
return Obx(
|
||||||
|
() => followList.isNotEmpty
|
||||||
|
? ListView.builder(
|
||||||
|
controller: scrollController,
|
||||||
|
itemCount: followList.length,
|
||||||
|
itemBuilder: ((context, index) {
|
||||||
|
return FollowItem(
|
||||||
|
item: followList[index],
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
: CustomScrollView(
|
||||||
|
slivers: [HttpError(errMsg: '未搜索到结果', fn: reRequest)],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return CustomScrollView(
|
||||||
|
slivers: [
|
||||||
|
HttpError(errMsg: snapshot.data['msg'], fn: reRequest)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -5,15 +5,17 @@ import 'package:get/get.dart';
|
|||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/models/common/tab_type.dart';
|
import 'package:pilipala/models/common/tab_type.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
|
import '../../http/index.dart';
|
||||||
|
|
||||||
class HomeController extends GetxController with GetTickerProviderStateMixin {
|
class HomeController extends GetxController with GetTickerProviderStateMixin {
|
||||||
bool flag = false;
|
bool flag = false;
|
||||||
late List tabs;
|
late RxList tabs = [].obs;
|
||||||
int initialIndex = 1;
|
RxInt initialIndex = 1.obs;
|
||||||
late TabController tabController;
|
late TabController tabController;
|
||||||
late List tabsCtrList;
|
late List tabsCtrList;
|
||||||
late List<Widget> tabsPageList;
|
late List<Widget> tabsPageList;
|
||||||
Box userInfoCache = GStrorage.userInfo;
|
Box userInfoCache = GStrorage.userInfo;
|
||||||
|
Box settingStorage = GStrorage.setting;
|
||||||
RxBool userLogin = false.obs;
|
RxBool userLogin = false.obs;
|
||||||
RxString userFace = ''.obs;
|
RxString userFace = ''.obs;
|
||||||
var userInfo;
|
var userInfo;
|
||||||
@ -21,6 +23,9 @@ class HomeController extends GetxController with GetTickerProviderStateMixin {
|
|||||||
late final StreamController<bool> searchBarStream =
|
late final StreamController<bool> searchBarStream =
|
||||||
StreamController<bool>.broadcast();
|
StreamController<bool>.broadcast();
|
||||||
late bool hideSearchBar;
|
late bool hideSearchBar;
|
||||||
|
late List defaultTabs;
|
||||||
|
late List<String> tabbarSort;
|
||||||
|
RxString defaultSearch = ''.obs;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -28,19 +33,13 @@ class HomeController extends GetxController with GetTickerProviderStateMixin {
|
|||||||
userInfo = userInfoCache.get('userInfoCache');
|
userInfo = userInfoCache.get('userInfoCache');
|
||||||
userLogin.value = userInfo != null;
|
userLogin.value = userInfo != null;
|
||||||
userFace.value = userInfo != null ? userInfo.face : '';
|
userFace.value = userInfo != null ? userInfo.face : '';
|
||||||
|
|
||||||
// 进行tabs配置
|
// 进行tabs配置
|
||||||
tabs = tabsConfig;
|
setTabConfig();
|
||||||
tabsCtrList = tabsConfig.map((e) => e['ctr']).toList();
|
|
||||||
tabsPageList = tabsConfig.map<Widget>((e) => e['page']).toList();
|
|
||||||
|
|
||||||
tabController = TabController(
|
|
||||||
initialIndex: initialIndex,
|
|
||||||
length: tabs.length,
|
|
||||||
vsync: this,
|
|
||||||
);
|
|
||||||
hideSearchBar =
|
hideSearchBar =
|
||||||
setting.get(SettingBoxKey.hideSearchBar, defaultValue: true);
|
setting.get(SettingBoxKey.hideSearchBar, defaultValue: true);
|
||||||
|
if (setting.get(SettingBoxKey.enableSearchWord, defaultValue: true)) {
|
||||||
|
searchDefault();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onRefresh() {
|
void onRefresh() {
|
||||||
@ -62,4 +61,52 @@ class HomeController extends GetxController with GetTickerProviderStateMixin {
|
|||||||
if (val) return;
|
if (val) return;
|
||||||
userFace.value = userInfo != null ? userInfo.face : '';
|
userFace.value = userInfo != null ? userInfo.face : '';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setTabConfig() async {
|
||||||
|
defaultTabs = [...tabsConfig];
|
||||||
|
tabbarSort = settingStorage.get(SettingBoxKey.tabbarSort,
|
||||||
|
defaultValue: ['live', 'rcmd', 'hot', 'bangumi']);
|
||||||
|
defaultTabs.retainWhere(
|
||||||
|
(item) => tabbarSort.contains((item['type'] as TabType).id));
|
||||||
|
defaultTabs.sort((a, b) => tabbarSort
|
||||||
|
.indexOf((a['type'] as TabType).id)
|
||||||
|
.compareTo(tabbarSort.indexOf((b['type'] as TabType).id)));
|
||||||
|
|
||||||
|
tabs.value = defaultTabs;
|
||||||
|
|
||||||
|
if (tabbarSort.contains(TabType.rcmd.id)) {
|
||||||
|
initialIndex.value = tabbarSort.indexOf(TabType.rcmd.id);
|
||||||
|
} else {
|
||||||
|
initialIndex.value = 0;
|
||||||
|
}
|
||||||
|
tabsCtrList = tabs.map((e) => e['ctr']).toList();
|
||||||
|
tabsPageList = tabs.map<Widget>((e) => e['page']).toList();
|
||||||
|
|
||||||
|
tabController = TabController(
|
||||||
|
initialIndex: initialIndex.value,
|
||||||
|
length: tabs.length,
|
||||||
|
vsync: this,
|
||||||
|
);
|
||||||
|
// 监听 tabController 切换
|
||||||
|
tabController.animation!.addListener(() {
|
||||||
|
if (tabController.indexIsChanging) {
|
||||||
|
if (initialIndex.value != tabController.index) {
|
||||||
|
initialIndex.value = tabController.index;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final int temp = tabController.animation!.value.round();
|
||||||
|
if (initialIndex.value != temp) {
|
||||||
|
initialIndex.value = temp;
|
||||||
|
tabController.index = initialIndex.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void searchDefault() async {
|
||||||
|
var res = await Request().get(Api.searchDefault);
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
defaultSearch.value = res.data['data']['name'];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,10 +1,10 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/pages/mine/index.dart';
|
import 'package:pilipala/pages/mine/index.dart';
|
||||||
import 'package:pilipala/pages/search/index.dart';
|
|
||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
import './controller.dart';
|
import './controller.dart';
|
||||||
|
|
||||||
@ -30,7 +30,7 @@ class _HomePageState extends State<HomePage>
|
|||||||
stream = _homeController.searchBarStream.stream;
|
stream = _homeController.searchBarStream.stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
showUserBottonSheet() {
|
showUserBottomSheet() {
|
||||||
feedBack();
|
feedBack();
|
||||||
showModalBottomSheet(
|
showModalBottomSheet(
|
||||||
context: context,
|
context: context,
|
||||||
@ -46,50 +46,61 @@ class _HomePageState extends State<HomePage>
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
super.build(context);
|
||||||
|
Brightness currentBrightness = MediaQuery.of(context).platformBrightness;
|
||||||
|
// 设置状态栏图标的亮度
|
||||||
|
SystemChrome.setSystemUIOverlayStyle(SystemUiOverlayStyle(
|
||||||
|
statusBarIconBrightness: currentBrightness == Brightness.light
|
||||||
|
? Brightness.dark
|
||||||
|
: Brightness.light,
|
||||||
|
));
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
extendBody: true,
|
extendBody: true,
|
||||||
extendBodyBehindAppBar: true,
|
extendBodyBehindAppBar: true,
|
||||||
appBar: AppBar(toolbarHeight: 0, elevation: 0),
|
body: Stack(
|
||||||
body: Column(
|
|
||||||
children: [
|
children: [
|
||||||
CustomAppBar(
|
// gradient background
|
||||||
stream: _homeController.hideSearchBar
|
Align(
|
||||||
? stream
|
alignment: Alignment.topLeft,
|
||||||
: StreamController<bool>.broadcast().stream,
|
child: Opacity(
|
||||||
ctr: _homeController,
|
opacity: 0.6,
|
||||||
callback: showUserBottonSheet,
|
child: Container(
|
||||||
),
|
width: MediaQuery.of(context).size.width,
|
||||||
const SizedBox(height: 8),
|
height: MediaQuery.of(context).size.height,
|
||||||
SizedBox(
|
decoration: BoxDecoration(
|
||||||
width: double.infinity,
|
gradient: LinearGradient(
|
||||||
height: 42,
|
colors: [
|
||||||
child: Align(
|
Theme.of(context).colorScheme.primary.withOpacity(0.9),
|
||||||
alignment: Alignment.center,
|
Theme.of(context).colorScheme.primary.withOpacity(0.5),
|
||||||
child: TabBar(
|
Theme.of(context).colorScheme.surface
|
||||||
controller: _homeController.tabController,
|
],
|
||||||
tabs: [
|
begin: Alignment.topLeft,
|
||||||
for (var i in _homeController.tabs) Tab(text: i['label'])
|
end: Alignment.bottomRight,
|
||||||
],
|
stops: const [0, 0.0034, 0.34]),
|
||||||
isScrollable: true,
|
),
|
||||||
dividerColor: Colors.transparent,
|
|
||||||
enableFeedback: true,
|
|
||||||
splashBorderRadius: BorderRadius.circular(10),
|
|
||||||
tabAlignment: TabAlignment.center,
|
|
||||||
onTap: (value) {
|
|
||||||
feedBack();
|
|
||||||
if (_homeController.initialIndex == value) {
|
|
||||||
_homeController.tabsCtrList[value]().animateToTop();
|
|
||||||
}
|
|
||||||
_homeController.initialIndex = value;
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Expanded(
|
Column(
|
||||||
child: TabBarView(
|
children: [
|
||||||
controller: _homeController.tabController,
|
CustomAppBar(
|
||||||
children: _homeController.tabsPageList,
|
stream: _homeController.hideSearchBar
|
||||||
),
|
? stream
|
||||||
|
: StreamController<bool>.broadcast().stream,
|
||||||
|
ctr: _homeController,
|
||||||
|
callback: showUserBottomSheet,
|
||||||
|
),
|
||||||
|
if (_homeController.tabs.length > 1) ...[
|
||||||
|
const CustomTabs(),
|
||||||
|
] else ...[
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
],
|
||||||
|
Expanded(
|
||||||
|
child: TabBarView(
|
||||||
|
controller: _homeController.tabController,
|
||||||
|
children: _homeController.tabsPageList,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@ -119,87 +130,23 @@ class CustomAppBar extends StatelessWidget implements PreferredSizeWidget {
|
|||||||
return StreamBuilder(
|
return StreamBuilder(
|
||||||
stream: stream,
|
stream: stream,
|
||||||
initialData: true,
|
initialData: true,
|
||||||
builder: (context, AsyncSnapshot snapshot) {
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
final RxBool isUserLoggedIn = ctr!.userLogin;
|
||||||
|
final double top = MediaQuery.of(context).padding.top;
|
||||||
return AnimatedOpacity(
|
return AnimatedOpacity(
|
||||||
opacity: snapshot.data ? 1 : 0,
|
opacity: snapshot.data ? 1 : 0,
|
||||||
duration: const Duration(milliseconds: 300),
|
duration: const Duration(milliseconds: 300),
|
||||||
child: AnimatedContainer(
|
child: AnimatedContainer(
|
||||||
curve: Curves.easeInOutCubicEmphasized,
|
curve: Curves.easeInOutCubicEmphasized,
|
||||||
duration: const Duration(milliseconds: 500),
|
duration: const Duration(milliseconds: 500),
|
||||||
height: snapshot.data
|
height: snapshot.data ? top + 52 : top,
|
||||||
? MediaQuery.of(context).padding.top + 52
|
padding: EdgeInsets.fromLTRB(14, top + 6, 14, 0),
|
||||||
: MediaQuery.of(context).padding.top - 10,
|
child: UserInfoWidget(
|
||||||
child: Container(
|
top: top,
|
||||||
padding: EdgeInsets.only(
|
ctr: ctr,
|
||||||
left: 20,
|
userLogin: isUserLoggedIn,
|
||||||
right: 20,
|
userFace: ctr?.userFace.value,
|
||||||
bottom: 0,
|
callback: () => callback!(),
|
||||||
top: MediaQuery.of(context).padding.top + 4,
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
const Expanded(child: SearchPage()),
|
|
||||||
if (ctr!.userLogin.value) ...[
|
|
||||||
const SizedBox(width: 6),
|
|
||||||
IconButton(
|
|
||||||
onPressed: () => Get.toNamed('/whisper'),
|
|
||||||
icon: const Icon(Icons.notifications_none))
|
|
||||||
],
|
|
||||||
const SizedBox(width: 6),
|
|
||||||
Obx(
|
|
||||||
() => ctr!.userLogin.value
|
|
||||||
? Stack(
|
|
||||||
children: [
|
|
||||||
Obx(
|
|
||||||
() => NetworkImgLayer(
|
|
||||||
type: 'avatar',
|
|
||||||
width: 34,
|
|
||||||
height: 34,
|
|
||||||
src: ctr!.userFace.value,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned.fill(
|
|
||||||
child: Material(
|
|
||||||
color: Colors.transparent,
|
|
||||||
child: InkWell(
|
|
||||||
onTap: () => callback!(),
|
|
||||||
splashColor: Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.primaryContainer
|
|
||||||
.withOpacity(0.3),
|
|
||||||
borderRadius: const BorderRadius.all(
|
|
||||||
Radius.circular(50),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
)
|
|
||||||
: SizedBox(
|
|
||||||
width: 38,
|
|
||||||
height: 38,
|
|
||||||
child: IconButton(
|
|
||||||
style: ButtonStyle(
|
|
||||||
padding:
|
|
||||||
MaterialStateProperty.all(EdgeInsets.zero),
|
|
||||||
backgroundColor:
|
|
||||||
MaterialStateProperty.resolveWith((states) {
|
|
||||||
return Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.onInverseSurface;
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
onPressed: () => callback!(),
|
|
||||||
icon: Icon(
|
|
||||||
Icons.person_rounded,
|
|
||||||
size: 22,
|
|
||||||
color: Theme.of(context).colorScheme.primary,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@ -207,3 +154,236 @@ class CustomAppBar extends StatelessWidget implements PreferredSizeWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class UserInfoWidget extends StatelessWidget {
|
||||||
|
const UserInfoWidget({
|
||||||
|
Key? key,
|
||||||
|
required this.top,
|
||||||
|
required this.userLogin,
|
||||||
|
required this.userFace,
|
||||||
|
required this.callback,
|
||||||
|
required this.ctr,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
final double top;
|
||||||
|
final RxBool userLogin;
|
||||||
|
final String? userFace;
|
||||||
|
final VoidCallback? callback;
|
||||||
|
final HomeController? ctr;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Row(
|
||||||
|
children: [
|
||||||
|
SearchBar(ctr: ctr),
|
||||||
|
if (userLogin.value) ...[
|
||||||
|
const SizedBox(width: 4),
|
||||||
|
ClipRect(
|
||||||
|
child: IconButton(
|
||||||
|
onPressed: () => Get.toNamed('/whisper'),
|
||||||
|
icon: const Icon(Icons.notifications_none),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Obx(
|
||||||
|
() => userLogin.value
|
||||||
|
? Stack(
|
||||||
|
children: [
|
||||||
|
NetworkImgLayer(
|
||||||
|
type: 'avatar',
|
||||||
|
width: 34,
|
||||||
|
height: 34,
|
||||||
|
src: userFace,
|
||||||
|
),
|
||||||
|
Positioned.fill(
|
||||||
|
child: Material(
|
||||||
|
color: Colors.transparent,
|
||||||
|
child: InkWell(
|
||||||
|
onTap: () => callback?.call(),
|
||||||
|
splashColor: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.primaryContainer
|
||||||
|
.withOpacity(0.3),
|
||||||
|
borderRadius: const BorderRadius.all(
|
||||||
|
Radius.circular(50),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
: DefaultUser(callback: () => callback!()),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class DefaultUser extends StatelessWidget {
|
||||||
|
const DefaultUser({super.key, this.callback});
|
||||||
|
final Function? callback;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return SizedBox(
|
||||||
|
width: 38,
|
||||||
|
height: 38,
|
||||||
|
child: IconButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
|
backgroundColor: MaterialStateProperty.resolveWith((states) {
|
||||||
|
return Theme.of(context).colorScheme.onInverseSurface;
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
onPressed: () => callback?.call(),
|
||||||
|
icon: Icon(
|
||||||
|
Icons.person_rounded,
|
||||||
|
size: 22,
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CustomTabs extends StatefulWidget {
|
||||||
|
const CustomTabs({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CustomTabs> createState() => _CustomTabsState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CustomTabsState extends State<CustomTabs> {
|
||||||
|
final HomeController _homeController = Get.put(HomeController());
|
||||||
|
|
||||||
|
void onTap(int index) {
|
||||||
|
feedBack();
|
||||||
|
if (_homeController.initialIndex.value == index) {
|
||||||
|
_homeController.tabsCtrList[index]().animateToTop();
|
||||||
|
}
|
||||||
|
_homeController.initialIndex.value = index;
|
||||||
|
_homeController.tabController.index = index;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Container(
|
||||||
|
height: 44,
|
||||||
|
margin: const EdgeInsets.only(top: 4),
|
||||||
|
child: Obx(
|
||||||
|
() => ListView.separated(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 14.0),
|
||||||
|
scrollDirection: Axis.horizontal,
|
||||||
|
itemCount: _homeController.tabs.length,
|
||||||
|
separatorBuilder: (BuildContext context, int index) {
|
||||||
|
return const SizedBox(width: 10);
|
||||||
|
},
|
||||||
|
itemBuilder: (BuildContext context, int index) {
|
||||||
|
String label = _homeController.tabs[index]['label'];
|
||||||
|
return Obx(
|
||||||
|
() => CustomChip(
|
||||||
|
onTap: () => onTap(index),
|
||||||
|
label: label,
|
||||||
|
selected: index == _homeController.initialIndex.value,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CustomChip extends StatelessWidget {
|
||||||
|
final Function onTap;
|
||||||
|
final String label;
|
||||||
|
final bool selected;
|
||||||
|
const CustomChip({
|
||||||
|
super.key,
|
||||||
|
required this.onTap,
|
||||||
|
required this.label,
|
||||||
|
required this.selected,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final ColorScheme colorTheme = Theme.of(context).colorScheme;
|
||||||
|
final Color secondaryContainer = colorTheme.secondaryContainer;
|
||||||
|
final TextStyle chipTextStyle = selected
|
||||||
|
? const TextStyle(fontWeight: FontWeight.bold, fontSize: 13)
|
||||||
|
: const TextStyle(fontSize: 13);
|
||||||
|
final ColorScheme colorScheme = Theme.of(context).colorScheme;
|
||||||
|
const VisualDensity visualDensity =
|
||||||
|
VisualDensity(horizontal: -4.0, vertical: -2.0);
|
||||||
|
return InputChip(
|
||||||
|
side: BorderSide(
|
||||||
|
color: selected
|
||||||
|
? colorScheme.onSecondaryContainer.withOpacity(0.2)
|
||||||
|
: Colors.transparent,
|
||||||
|
),
|
||||||
|
backgroundColor: secondaryContainer,
|
||||||
|
selectedColor: secondaryContainer,
|
||||||
|
color: MaterialStateProperty.resolveWith<Color>(
|
||||||
|
(Set<MaterialState> states) => secondaryContainer.withAlpha(200)),
|
||||||
|
padding: const EdgeInsets.fromLTRB(7, 1, 7, 1),
|
||||||
|
label: Text(label, style: chipTextStyle),
|
||||||
|
onPressed: () => onTap(),
|
||||||
|
selected: selected,
|
||||||
|
showCheckmark: false,
|
||||||
|
visualDensity: visualDensity,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class SearchBar extends StatelessWidget {
|
||||||
|
const SearchBar({
|
||||||
|
Key? key,
|
||||||
|
required this.ctr,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
final HomeController? ctr;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final ColorScheme colorScheme = Theme.of(context).colorScheme;
|
||||||
|
return Expanded(
|
||||||
|
child: Container(
|
||||||
|
width: 250,
|
||||||
|
height: 44,
|
||||||
|
clipBehavior: Clip.hardEdge,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
borderRadius: BorderRadius.circular(25),
|
||||||
|
),
|
||||||
|
child: Material(
|
||||||
|
color: colorScheme.onSecondaryContainer.withOpacity(0.05),
|
||||||
|
child: InkWell(
|
||||||
|
splashColor: colorScheme.primaryContainer.withOpacity(0.3),
|
||||||
|
onTap: () => Get.toNamed(
|
||||||
|
'/search',
|
||||||
|
parameters: {'hintText': ctr!.defaultSearch.value},
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const SizedBox(width: 14),
|
||||||
|
Icon(
|
||||||
|
Icons.search_outlined,
|
||||||
|
color: colorScheme.onSecondaryContainer,
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Obx(
|
||||||
|
() => Text(
|
||||||
|
ctr!.defaultSearch.value,
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
style: TextStyle(color: colorScheme.outline),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -70,68 +70,70 @@ class _HotPageState extends State<HotPage> with AutomaticKeepAliveClientMixin {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
super.build(context);
|
||||||
return Scaffold(
|
return RefreshIndicator(
|
||||||
body: RefreshIndicator(
|
onRefresh: () async {
|
||||||
onRefresh: () async {
|
return await _hotController.onRefresh();
|
||||||
return await _hotController.onRefresh();
|
},
|
||||||
},
|
child: CustomScrollView(
|
||||||
child: CustomScrollView(
|
controller: _hotController.scrollController,
|
||||||
controller: _hotController.scrollController,
|
slivers: [
|
||||||
slivers: [
|
SliverPadding(
|
||||||
SliverPadding(
|
// 单列布局 EdgeInsets.zero
|
||||||
// 单列布局 EdgeInsets.zero
|
padding:
|
||||||
padding:
|
const EdgeInsets.fromLTRB(0, StyleString.safeSpace - 5, 0, 0),
|
||||||
const EdgeInsets.fromLTRB(0, StyleString.safeSpace - 5, 0, 0),
|
sliver: FutureBuilder(
|
||||||
sliver: FutureBuilder(
|
future: _futureBuilderFuture,
|
||||||
future: _futureBuilderFuture,
|
builder: (context, snapshot) {
|
||||||
builder: (context, snapshot) {
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
if (snapshot.connectionState == ConnectionState.done) {
|
Map data = snapshot.data as Map;
|
||||||
Map data = snapshot.data as Map;
|
if (data['status']) {
|
||||||
if (data['status']) {
|
return Obx(
|
||||||
return Obx(
|
() => SliverList(
|
||||||
() => SliverList(
|
delegate: SliverChildBuilderDelegate((context, index) {
|
||||||
delegate:
|
return VideoCardH(
|
||||||
SliverChildBuilderDelegate((context, index) {
|
videoItem: _hotController.videoList[index],
|
||||||
return VideoCardH(
|
showPubdate: true,
|
||||||
videoItem: _hotController.videoList[index],
|
longPress: () {
|
||||||
showPubdate: true,
|
_hotController.popupDialog = _createPopupDialog(
|
||||||
longPress: () {
|
_hotController.videoList[index]);
|
||||||
_hotController.popupDialog = _createPopupDialog(
|
Overlay.of(context)
|
||||||
_hotController.videoList[index]);
|
.insert(_hotController.popupDialog!);
|
||||||
Overlay.of(context)
|
},
|
||||||
.insert(_hotController.popupDialog!);
|
longPressEnd: () {
|
||||||
},
|
_hotController.popupDialog?.remove();
|
||||||
longPressEnd: () {
|
},
|
||||||
_hotController.popupDialog?.remove();
|
);
|
||||||
},
|
}, childCount: _hotController.videoList.length),
|
||||||
);
|
),
|
||||||
}, childCount: _hotController.videoList.length),
|
);
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return HttpError(
|
|
||||||
errMsg: data['msg'],
|
|
||||||
fn: () => setState(() {}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// 骨架屏
|
return HttpError(
|
||||||
return SliverList(
|
errMsg: data['msg'],
|
||||||
delegate: SliverChildBuilderDelegate((context, index) {
|
fn: () {
|
||||||
return const VideoCardHSkeleton();
|
setState(() {
|
||||||
}, childCount: 10),
|
_futureBuilderFuture =
|
||||||
|
_hotController.queryHotFeed('init');
|
||||||
|
});
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
},
|
} else {
|
||||||
),
|
// 骨架屏
|
||||||
|
return SliverList(
|
||||||
|
delegate: SliverChildBuilderDelegate((context, index) {
|
||||||
|
return const VideoCardHSkeleton();
|
||||||
|
}, childCount: 10),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
),
|
),
|
||||||
SliverToBoxAdapter(
|
),
|
||||||
child: SizedBox(
|
SliverToBoxAdapter(
|
||||||
height: MediaQuery.of(context).padding.bottom + 10,
|
child: SizedBox(
|
||||||
),
|
height: MediaQuery.of(context).padding.bottom + 10,
|
||||||
)
|
),
|
||||||
],
|
)
|
||||||
),
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -96,9 +96,6 @@ class HtmlRenderController extends GetxController {
|
|||||||
_sortType = ReplySortType.like;
|
_sortType = ReplySortType.like;
|
||||||
break;
|
break;
|
||||||
case ReplySortType.like:
|
case ReplySortType.like:
|
||||||
_sortType = ReplySortType.reply;
|
|
||||||
break;
|
|
||||||
case ReplySortType.reply:
|
|
||||||
_sortType = ReplySortType.time;
|
_sortType = ReplySortType.time;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|||||||
@ -10,8 +10,8 @@ import 'package:pilipala/common/widgets/http_error.dart';
|
|||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/models/common/reply_type.dart';
|
import 'package:pilipala/models/common/reply_type.dart';
|
||||||
import 'package:pilipala/pages/video/detail/reply/widgets/reply_item.dart';
|
import 'package:pilipala/pages/video/detail/reply/widgets/reply_item.dart';
|
||||||
import 'package:pilipala/pages/video/detail/replyNew/index.dart';
|
import 'package:pilipala/pages/video/detail/reply_new/index.dart';
|
||||||
import 'package:pilipala/pages/video/detail/replyReply/index.dart';
|
import 'package:pilipala/pages/video/detail/reply_reply/index.dart';
|
||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
|
|
||||||
import 'controller.dart';
|
import 'controller.dart';
|
||||||
|
|||||||
@ -10,8 +10,7 @@ class LiveController extends GetxController {
|
|||||||
int count = 12;
|
int count = 12;
|
||||||
int _currentPage = 1;
|
int _currentPage = 1;
|
||||||
RxInt crossAxisCount = 2.obs;
|
RxInt crossAxisCount = 2.obs;
|
||||||
RxList<LiveItemModel> liveList = [LiveItemModel()].obs;
|
RxList<LiveItemModel> liveList = <LiveItemModel>[].obs;
|
||||||
bool isLoadingMore = false;
|
|
||||||
bool flag = false;
|
bool flag = false;
|
||||||
OverlayEntry? popupDialog;
|
OverlayEntry? popupDialog;
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
@ -39,7 +38,6 @@ class LiveController extends GetxController {
|
|||||||
}
|
}
|
||||||
_currentPage += 1;
|
_currentPage += 1;
|
||||||
}
|
}
|
||||||
isLoadingMore = false;
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -11,7 +11,6 @@ import 'package:pilipala/common/widgets/http_error.dart';
|
|||||||
import 'package:pilipala/common/widgets/overlay_pop.dart';
|
import 'package:pilipala/common/widgets/overlay_pop.dart';
|
||||||
import 'package:pilipala/pages/home/index.dart';
|
import 'package:pilipala/pages/home/index.dart';
|
||||||
import 'package:pilipala/pages/main/index.dart';
|
import 'package:pilipala/pages/main/index.dart';
|
||||||
import 'package:pilipala/pages/rcmd/index.dart';
|
|
||||||
|
|
||||||
import 'controller.dart';
|
import 'controller.dart';
|
||||||
import 'widgets/live_item.dart';
|
import 'widgets/live_item.dart';
|
||||||
@ -45,8 +44,8 @@ class _LivePageState extends State<LivePage>
|
|||||||
() {
|
() {
|
||||||
if (scrollController.position.pixels >=
|
if (scrollController.position.pixels >=
|
||||||
scrollController.position.maxScrollExtent - 200) {
|
scrollController.position.maxScrollExtent - 200) {
|
||||||
EasyThrottle.throttle('liveList', const Duration(seconds: 1), () {
|
EasyThrottle.throttle('liveList', const Duration(milliseconds: 200),
|
||||||
_liveController.isLoadingMore = true;
|
() {
|
||||||
_liveController.onLoad();
|
_liveController.onLoad();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -108,24 +107,20 @@ class _LivePageState extends State<LivePage>
|
|||||||
} else {
|
} else {
|
||||||
return HttpError(
|
return HttpError(
|
||||||
errMsg: data['msg'],
|
errMsg: data['msg'],
|
||||||
fn: () => {},
|
fn: () {
|
||||||
|
setState(() {
|
||||||
|
_futureBuilderFuture =
|
||||||
|
_liveController.queryLiveList('init');
|
||||||
|
});
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// 缓存数据
|
return contentGrid(_liveController, []);
|
||||||
if (_liveController.liveList.length > 1) {
|
|
||||||
return contentGrid(
|
|
||||||
_liveController, _liveController.liveList);
|
|
||||||
}
|
|
||||||
// 骨架屏
|
|
||||||
else {
|
|
||||||
return contentGrid(_liveController, []);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
LoadingMore(ctr: _liveController)
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -162,8 +157,9 @@ class _LivePageState extends State<LivePage>
|
|||||||
crossAxisCount: crossAxisCount,
|
crossAxisCount: crossAxisCount,
|
||||||
mainAxisExtent:
|
mainAxisExtent:
|
||||||
Get.size.width / crossAxisCount / StyleString.aspectRatio +
|
Get.size.width / crossAxisCount / StyleString.aspectRatio +
|
||||||
(crossAxisCount == 1 ? 48 : 68) *
|
MediaQuery.textScalerOf(context).scale(
|
||||||
MediaQuery.of(context).textScaleFactor,
|
(crossAxisCount == 1 ? 48 : 68),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
delegate: SliverChildBuilderDelegate(
|
delegate: SliverChildBuilderDelegate(
|
||||||
(BuildContext context, int index) {
|
(BuildContext context, int index) {
|
||||||
|
|||||||
@ -184,18 +184,32 @@ class VideoStat extends StatelessWidget {
|
|||||||
tileMode: TileMode.mirror,
|
tileMode: TileMode.mirror,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
child: RichText(
|
child: Row(
|
||||||
maxLines: 1,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
textAlign: TextAlign.justify,
|
children: [
|
||||||
softWrap: false,
|
Text(
|
||||||
text: TextSpan(
|
liveItem!.areaName!,
|
||||||
style: const TextStyle(fontSize: 11, color: Colors.white),
|
style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
children: [
|
),
|
||||||
TextSpan(text: liveItem!.areaName!),
|
Text(
|
||||||
TextSpan(text: liveItem!.watchedShow!['text_small']),
|
liveItem!.watchedShow!['text_small'],
|
||||||
],
|
style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
),
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// child: RichText(
|
||||||
|
// maxLines: 1,
|
||||||
|
// textAlign: TextAlign.justify,
|
||||||
|
// softWrap: false,
|
||||||
|
// text: TextSpan(
|
||||||
|
// style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
|
// children: [
|
||||||
|
// TextSpan(text: liveItem!.areaName!),
|
||||||
|
// TextSpan(text: liveItem!.watchedShow!['text_small']),
|
||||||
|
// ],
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,177 +0,0 @@
|
|||||||
import 'dart:io';
|
|
||||||
|
|
||||||
import 'package:floating/floating.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:get/get.dart';
|
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
|
||||||
import 'package:pilipala/plugin/pl_player/index.dart';
|
|
||||||
|
|
||||||
import 'controller.dart';
|
|
||||||
import 'widgets/bottom_control.dart';
|
|
||||||
|
|
||||||
class LiveRoomPage extends StatefulWidget {
|
|
||||||
const LiveRoomPage({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<LiveRoomPage> createState() => _LiveRoomPageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _LiveRoomPageState extends State<LiveRoomPage> {
|
|
||||||
final LiveRoomController _liveRoomController = Get.put(LiveRoomController());
|
|
||||||
PlPlayerController? plPlayerController;
|
|
||||||
|
|
||||||
bool isShowCover = true;
|
|
||||||
bool isPlay = true;
|
|
||||||
Floating? floating;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
plPlayerController = _liveRoomController.plPlayerController;
|
|
||||||
plPlayerController!.onPlayerStatusChanged.listen(
|
|
||||||
(PlayerStatus status) {
|
|
||||||
if (status == PlayerStatus.playing) {
|
|
||||||
isShowCover = false;
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
if (Platform.isAndroid) {
|
|
||||||
floating = Floating();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
plPlayerController!.dispose();
|
|
||||||
if (floating != null) {
|
|
||||||
floating!.dispose();
|
|
||||||
}
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
Widget childWhenDisabled = Scaffold(
|
|
||||||
primary: true,
|
|
||||||
appBar: PreferredSize(
|
|
||||||
preferredSize: Size.fromHeight(
|
|
||||||
MediaQuery.of(context).orientation == Orientation.portrait ? 56 : 0,
|
|
||||||
),
|
|
||||||
child: AppBar(
|
|
||||||
centerTitle: false,
|
|
||||||
titleSpacing: 0,
|
|
||||||
title: _liveRoomController.liveItem != null
|
|
||||||
? Row(
|
|
||||||
children: [
|
|
||||||
NetworkImgLayer(
|
|
||||||
width: 34,
|
|
||||||
height: 34,
|
|
||||||
type: 'avatar',
|
|
||||||
src: _liveRoomController.liveItem.face,
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
_liveRoomController.liveItem.uname,
|
|
||||||
style: const TextStyle(fontSize: 14),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 1),
|
|
||||||
if (_liveRoomController.liveItem.watchedShow != null)
|
|
||||||
Text(
|
|
||||||
_liveRoomController
|
|
||||||
.liveItem.watchedShow['text_large'] ??
|
|
||||||
'',
|
|
||||||
style: const TextStyle(fontSize: 12)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
// actions: [
|
|
||||||
// SizedBox(
|
|
||||||
// height: 34,
|
|
||||||
// child: ElevatedButton(onPressed: () {}, child: const Text('关注')),
|
|
||||||
// ),
|
|
||||||
// const SizedBox(width: 12),
|
|
||||||
// ],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
body: Column(
|
|
||||||
children: [
|
|
||||||
Stack(
|
|
||||||
children: [
|
|
||||||
PopScope(
|
|
||||||
canPop: plPlayerController?.isFullScreen.value != true,
|
|
||||||
onPopInvoked: (bool didPop) {
|
|
||||||
if (plPlayerController?.isFullScreen.value == true) {
|
|
||||||
plPlayerController!.triggerFullScreen(status: false);
|
|
||||||
}
|
|
||||||
if (MediaQuery.of(context).orientation ==
|
|
||||||
Orientation.landscape) {
|
|
||||||
verticalScreen();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: SizedBox(
|
|
||||||
width: Get.size.width,
|
|
||||||
height: MediaQuery.of(context).orientation ==
|
|
||||||
Orientation.landscape
|
|
||||||
? Get.size.height
|
|
||||||
: Get.size.width * 9 / 16,
|
|
||||||
child: plPlayerController!.videoPlayerController != null
|
|
||||||
? PLVideoPlayer(
|
|
||||||
controller: plPlayerController!,
|
|
||||||
bottomControl: BottomControl(
|
|
||||||
controller: plPlayerController,
|
|
||||||
liveRoomCtr: _liveRoomController,
|
|
||||||
floating: floating,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// if (_liveRoomController.liveItem != null &&
|
|
||||||
// _liveRoomController.liveItem.cover != null)
|
|
||||||
// Visibility(
|
|
||||||
// visible: isShowCover,
|
|
||||||
// child: Positioned(
|
|
||||||
// top: 0,
|
|
||||||
// left: 0,
|
|
||||||
// right: 0,
|
|
||||||
// child: NetworkImgLayer(
|
|
||||||
// type: 'emote',
|
|
||||||
// src: _liveRoomController.liveItem.cover,
|
|
||||||
// width: Get.size.width,
|
|
||||||
// height: videoHeight,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
Widget childWhenEnabled = AspectRatio(
|
|
||||||
aspectRatio: 16 / 9,
|
|
||||||
child: plPlayerController!.videoPlayerController != null
|
|
||||||
? PLVideoPlayer(
|
|
||||||
controller: plPlayerController!,
|
|
||||||
bottomControl: BottomControl(
|
|
||||||
controller: plPlayerController,
|
|
||||||
liveRoomCtr: _liveRoomController,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
);
|
|
||||||
if (Platform.isAndroid) {
|
|
||||||
return PiPSwitcher(
|
|
||||||
childWhenDisabled: childWhenDisabled,
|
|
||||||
childWhenEnabled: childWhenEnabled,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return childWhenDisabled;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -3,6 +3,7 @@ import 'package:pilipala/http/constants.dart';
|
|||||||
import 'package:pilipala/http/live.dart';
|
import 'package:pilipala/http/live.dart';
|
||||||
import 'package:pilipala/models/live/room_info.dart';
|
import 'package:pilipala/models/live/room_info.dart';
|
||||||
import 'package:pilipala/plugin/pl_player/index.dart';
|
import 'package:pilipala/plugin/pl_player/index.dart';
|
||||||
|
import '../../models/live/room_info_h5.dart';
|
||||||
|
|
||||||
class LiveRoomController extends GetxController {
|
class LiveRoomController extends GetxController {
|
||||||
String cover = '';
|
String cover = '';
|
||||||
@ -14,13 +15,7 @@ class LiveRoomController extends GetxController {
|
|||||||
RxBool volumeOff = false.obs;
|
RxBool volumeOff = false.obs;
|
||||||
PlPlayerController plPlayerController =
|
PlPlayerController plPlayerController =
|
||||||
PlPlayerController.getInstance(videoType: 'live');
|
PlPlayerController.getInstance(videoType: 'live');
|
||||||
|
Rx<RoomInfoH5Model> roomInfoH5 = RoomInfoH5Model().obs;
|
||||||
// MeeduPlayerController meeduPlayerController = MeeduPlayerController(
|
|
||||||
// colorTheme: Theme.of(Get.context!).colorScheme.primary,
|
|
||||||
// pipEnabled: true,
|
|
||||||
// controlsStyle: ControlsStyle.live,
|
|
||||||
// enabledButtons: const EnabledButtons(pip: true),
|
|
||||||
// );
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -36,11 +31,10 @@ class LiveRoomController extends GetxController {
|
|||||||
cover = liveItem.cover;
|
cover = liveItem.cover;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
queryLiveInfo();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
playerInit(source) {
|
playerInit(source) async {
|
||||||
plPlayerController.setDataSource(
|
await plPlayerController.setDataSource(
|
||||||
DataSource(
|
DataSource(
|
||||||
videoSource: source,
|
videoSource: source,
|
||||||
audioSource: null,
|
audioSource: null,
|
||||||
@ -66,7 +60,8 @@ class LiveRoomController extends GetxController {
|
|||||||
String videoUrl = (item.urlInfo?.first.host)! +
|
String videoUrl = (item.urlInfo?.first.host)! +
|
||||||
item.baseUrl! +
|
item.baseUrl! +
|
||||||
item.urlInfo!.first.extra!;
|
item.urlInfo!.first.extra!;
|
||||||
playerInit(videoUrl);
|
await playerInit(videoUrl);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -80,4 +75,12 @@ class LiveRoomController extends GetxController {
|
|||||||
volumeOff.value = true;
|
volumeOff.value = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future queryLiveInfoH5() async {
|
||||||
|
var res = await LiveHttp.liveRoomInfoH5(roomId: roomId);
|
||||||
|
if (res['status']) {
|
||||||
|
roomInfoH5.value = res['data'];
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
207
lib/pages/live_room/view.dart
Normal file
207
lib/pages/live_room/view.dart
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:floating/floating.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
|
import 'package:pilipala/plugin/pl_player/index.dart';
|
||||||
|
|
||||||
|
import 'controller.dart';
|
||||||
|
import 'widgets/bottom_control.dart';
|
||||||
|
|
||||||
|
class LiveRoomPage extends StatefulWidget {
|
||||||
|
const LiveRoomPage({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<LiveRoomPage> createState() => _LiveRoomPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _LiveRoomPageState extends State<LiveRoomPage> {
|
||||||
|
final LiveRoomController _liveRoomController = Get.put(LiveRoomController());
|
||||||
|
PlPlayerController? plPlayerController;
|
||||||
|
late Future? _futureBuilder;
|
||||||
|
late Future? _futureBuilderFuture;
|
||||||
|
|
||||||
|
bool isShowCover = true;
|
||||||
|
bool isPlay = true;
|
||||||
|
Floating? floating;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
floating = Floating();
|
||||||
|
}
|
||||||
|
videoSourceInit();
|
||||||
|
_futureBuilderFuture = _liveRoomController.queryLiveInfo();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> videoSourceInit() async {
|
||||||
|
_futureBuilder = _liveRoomController.queryLiveInfoH5();
|
||||||
|
plPlayerController = _liveRoomController.plPlayerController;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
plPlayerController!.dispose();
|
||||||
|
if (floating != null) {
|
||||||
|
floating!.dispose();
|
||||||
|
}
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
Widget videoPlayerPanel = FutureBuilder(
|
||||||
|
future: _futureBuilderFuture,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.hasData && snapshot.data['status']) {
|
||||||
|
return PLVideoPlayer(
|
||||||
|
controller: plPlayerController!,
|
||||||
|
bottomControl: BottomControl(
|
||||||
|
controller: plPlayerController,
|
||||||
|
liveRoomCtr: _liveRoomController,
|
||||||
|
floating: floating,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
Widget childWhenDisabled = Scaffold(
|
||||||
|
primary: true,
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
// Obx(
|
||||||
|
// () => Positioned.fill(
|
||||||
|
// child: Opacity(
|
||||||
|
// opacity: 0.8,
|
||||||
|
// child: _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground !=
|
||||||
|
// '' &&
|
||||||
|
// _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground !=
|
||||||
|
// null
|
||||||
|
// ? NetworkImgLayer(
|
||||||
|
// width: Get.width,
|
||||||
|
// height: Get.height,
|
||||||
|
// src: _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground ??
|
||||||
|
// '',
|
||||||
|
// )
|
||||||
|
// : Image.asset(
|
||||||
|
// 'assets/images/live/default_bg.webp',
|
||||||
|
// width: Get.width,
|
||||||
|
// height: Get.height,
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
Positioned.fill(
|
||||||
|
child: Opacity(
|
||||||
|
opacity: 0.8,
|
||||||
|
child: Image.asset(
|
||||||
|
'assets/images/live/default_bg.webp',
|
||||||
|
width: Get.width,
|
||||||
|
height: Get.height,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
AppBar(
|
||||||
|
centerTitle: false,
|
||||||
|
titleSpacing: 0,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
toolbarHeight:
|
||||||
|
MediaQuery.of(context).orientation == Orientation.portrait
|
||||||
|
? 56
|
||||||
|
: 0,
|
||||||
|
title: FutureBuilder(
|
||||||
|
future: _futureBuilder,
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
if (snapshot.data == null) {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
Map data = snapshot.data as Map;
|
||||||
|
if (data['status']) {
|
||||||
|
return Obx(
|
||||||
|
() => Row(
|
||||||
|
children: [
|
||||||
|
NetworkImgLayer(
|
||||||
|
width: 34,
|
||||||
|
height: 34,
|
||||||
|
type: 'avatar',
|
||||||
|
src: _liveRoomController
|
||||||
|
.roomInfoH5.value.anchorInfo!.baseInfo!.face,
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
_liveRoomController.roomInfoH5.value
|
||||||
|
.anchorInfo!.baseInfo!.uname!,
|
||||||
|
style: const TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 1),
|
||||||
|
if (_liveRoomController
|
||||||
|
.roomInfoH5.value.watchedShow !=
|
||||||
|
null)
|
||||||
|
Text(
|
||||||
|
_liveRoomController.roomInfoH5.value
|
||||||
|
.watchedShow!['text_large'] ??
|
||||||
|
'',
|
||||||
|
style: const TextStyle(fontSize: 12),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
PopScope(
|
||||||
|
canPop: plPlayerController?.isFullScreen.value != true,
|
||||||
|
onPopInvoked: (bool didPop) {
|
||||||
|
if (plPlayerController?.isFullScreen.value == true) {
|
||||||
|
plPlayerController!.triggerFullScreen(status: false);
|
||||||
|
}
|
||||||
|
if (MediaQuery.of(context).orientation ==
|
||||||
|
Orientation.landscape) {
|
||||||
|
verticalScreen();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: SizedBox(
|
||||||
|
width: Get.size.width,
|
||||||
|
height: MediaQuery.of(context).orientation ==
|
||||||
|
Orientation.landscape
|
||||||
|
? Get.size.height
|
||||||
|
: Get.size.width * 9 / 16,
|
||||||
|
child: videoPlayerPanel,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
return PiPSwitcher(
|
||||||
|
childWhenDisabled: childWhenDisabled,
|
||||||
|
childWhenEnabled: videoPlayerPanel,
|
||||||
|
floating: floating,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return childWhenDisabled;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -5,7 +5,7 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/models/video/play/url.dart';
|
import 'package:pilipala/models/video/play/url.dart';
|
||||||
import 'package:pilipala/pages/liveRoom/index.dart';
|
import 'package:pilipala/pages/live_room/index.dart';
|
||||||
import 'package:pilipala/plugin/pl_player/index.dart';
|
import 'package:pilipala/plugin/pl_player/index.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
|
|
||||||
@ -1,14 +1,17 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
|
import 'package:pilipala/http/common.dart';
|
||||||
import 'package:pilipala/pages/dynamics/index.dart';
|
import 'package:pilipala/pages/dynamics/index.dart';
|
||||||
import 'package:pilipala/pages/home/view.dart';
|
import 'package:pilipala/pages/home/view.dart';
|
||||||
import 'package:pilipala/pages/media/index.dart';
|
import 'package:pilipala/pages/media/index.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
|
import '../../models/common/dynamic_badge_mode.dart';
|
||||||
|
|
||||||
class MainController extends GetxController {
|
class MainController extends GetxController {
|
||||||
List<Widget> pages = <Widget>[
|
List<Widget> pages = <Widget>[
|
||||||
@ -19,14 +22,15 @@ class MainController extends GetxController {
|
|||||||
RxList navigationBars = [
|
RxList navigationBars = [
|
||||||
{
|
{
|
||||||
'icon': const Icon(
|
'icon': const Icon(
|
||||||
Icons.favorite_outline,
|
Icons.home_outlined,
|
||||||
size: 21,
|
size: 21,
|
||||||
),
|
),
|
||||||
'selectIcon': const Icon(
|
'selectIcon': const Icon(
|
||||||
Icons.favorite,
|
Icons.home,
|
||||||
size: 21,
|
size: 21,
|
||||||
),
|
),
|
||||||
'label': "首页",
|
'label': "首页",
|
||||||
|
'count': 0,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
'icon': const Icon(
|
'icon': const Icon(
|
||||||
@ -38,17 +42,19 @@ class MainController extends GetxController {
|
|||||||
size: 21,
|
size: 21,
|
||||||
),
|
),
|
||||||
'label': "动态",
|
'label': "动态",
|
||||||
|
'count': 0,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
'icon': const Icon(
|
'icon': const Icon(
|
||||||
Icons.folder_outlined,
|
Icons.video_collection_outlined,
|
||||||
size: 20,
|
size: 20,
|
||||||
),
|
),
|
||||||
'selectIcon': const Icon(
|
'selectIcon': const Icon(
|
||||||
Icons.folder,
|
Icons.video_collection,
|
||||||
size: 21,
|
size: 21,
|
||||||
),
|
),
|
||||||
'label': "媒体库",
|
'label': "媒体库",
|
||||||
|
'count': 0,
|
||||||
}
|
}
|
||||||
].obs;
|
].obs;
|
||||||
final StreamController<bool> bottomBarStream =
|
final StreamController<bool> bottomBarStream =
|
||||||
@ -56,6 +62,11 @@ class MainController extends GetxController {
|
|||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
DateTime? _lastPressedAt;
|
DateTime? _lastPressedAt;
|
||||||
late bool hideTabBar;
|
late bool hideTabBar;
|
||||||
|
late PageController pageController;
|
||||||
|
int selectedIndex = 0;
|
||||||
|
Box userInfoCache = GStrorage.userInfo;
|
||||||
|
RxBool userLogin = false.obs;
|
||||||
|
late Rx<DynamicBadgeMode> dynamicBadgeType = DynamicBadgeMode.number.obs;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -64,17 +75,52 @@ class MainController extends GetxController {
|
|||||||
Utils.checkUpdata();
|
Utils.checkUpdata();
|
||||||
}
|
}
|
||||||
hideTabBar = setting.get(SettingBoxKey.hideTabBar, defaultValue: true);
|
hideTabBar = setting.get(SettingBoxKey.hideTabBar, defaultValue: true);
|
||||||
|
var userInfo = userInfoCache.get('userInfoCache');
|
||||||
|
userLogin.value = userInfo != null;
|
||||||
|
dynamicBadgeType.value = DynamicBadgeMode.values[setting.get(
|
||||||
|
SettingBoxKey.dynamicBadgeMode,
|
||||||
|
defaultValue: DynamicBadgeMode.number.code)];
|
||||||
|
if (dynamicBadgeType.value != DynamicBadgeMode.hidden) {
|
||||||
|
getUnreadDynamic();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> onBackPressed(BuildContext context) {
|
void onBackPressed(BuildContext context) {
|
||||||
if (_lastPressedAt == null ||
|
if (_lastPressedAt == null ||
|
||||||
DateTime.now().difference(_lastPressedAt!) >
|
DateTime.now().difference(_lastPressedAt!) >
|
||||||
const Duration(seconds: 2)) {
|
const Duration(seconds: 2)) {
|
||||||
// 两次点击时间间隔超过2秒,重新记录时间戳
|
// 两次点击时间间隔超过2秒,重新记录时间戳
|
||||||
_lastPressedAt = DateTime.now();
|
_lastPressedAt = DateTime.now();
|
||||||
|
if (selectedIndex != 0) {
|
||||||
|
pageController.jumpTo(0);
|
||||||
|
}
|
||||||
SmartDialog.showToast("再按一次退出Pili");
|
SmartDialog.showToast("再按一次退出Pili");
|
||||||
return Future.value(false); // 不退出应用
|
return; // 不退出应用
|
||||||
}
|
}
|
||||||
return Future.value(true); // 退出应用
|
SystemNavigator.pop(); // 退出应用
|
||||||
|
}
|
||||||
|
|
||||||
|
void getUnreadDynamic() async {
|
||||||
|
if (!userLogin.value) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
int dynamicItemIndex =
|
||||||
|
navigationBars.indexWhere((item) => item['label'] == "动态");
|
||||||
|
var res = await CommonHttp.unReadDynamic();
|
||||||
|
var data = res['data'];
|
||||||
|
if (dynamicItemIndex != -1) {
|
||||||
|
navigationBars[dynamicItemIndex]['count'] =
|
||||||
|
data == null ? 0 : data.length; // 修改 count 属性为新的值
|
||||||
|
}
|
||||||
|
navigationBars.refresh();
|
||||||
|
}
|
||||||
|
|
||||||
|
void clearUnread() async {
|
||||||
|
int dynamicItemIndex =
|
||||||
|
navigationBars.indexWhere((item) => item['label'] == "动态");
|
||||||
|
if (dynamicItemIndex != -1) {
|
||||||
|
navigationBars[dynamicItemIndex]['count'] = 0; // 修改 count 属性为新的值
|
||||||
|
}
|
||||||
|
navigationBars.refresh();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -3,6 +3,7 @@ import 'dart:async';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
|
import 'package:pilipala/models/common/dynamic_badge_mode.dart';
|
||||||
import 'package:pilipala/pages/dynamics/index.dart';
|
import 'package:pilipala/pages/dynamics/index.dart';
|
||||||
import 'package:pilipala/pages/home/index.dart';
|
import 'package:pilipala/pages/home/index.dart';
|
||||||
import 'package:pilipala/pages/media/index.dart';
|
import 'package:pilipala/pages/media/index.dart';
|
||||||
@ -24,8 +25,6 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
final DynamicsController _dynamicController = Get.put(DynamicsController());
|
final DynamicsController _dynamicController = Get.put(DynamicsController());
|
||||||
final MediaController _mediaController = Get.put(MediaController());
|
final MediaController _mediaController = Get.put(MediaController());
|
||||||
|
|
||||||
PageController? _pageController;
|
|
||||||
int selectedIndex = 0;
|
|
||||||
int? _lastSelectTime; //上次点击时间
|
int? _lastSelectTime; //上次点击时间
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
late bool enableMYBar;
|
late bool enableMYBar;
|
||||||
@ -34,13 +33,14 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_lastSelectTime = DateTime.now().millisecondsSinceEpoch;
|
_lastSelectTime = DateTime.now().millisecondsSinceEpoch;
|
||||||
_pageController = PageController(initialPage: selectedIndex);
|
_mainController.pageController =
|
||||||
|
PageController(initialPage: _mainController.selectedIndex);
|
||||||
enableMYBar = setting.get(SettingBoxKey.enableMYBar, defaultValue: true);
|
enableMYBar = setting.get(SettingBoxKey.enableMYBar, defaultValue: true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void setIndex(int value) async {
|
void setIndex(int value) async {
|
||||||
feedBack();
|
feedBack();
|
||||||
_pageController!.jumpToPage(value);
|
_mainController.pageController.jumpToPage(value);
|
||||||
var currentPage = _mainController.pages[value];
|
var currentPage = _mainController.pages[value];
|
||||||
if (currentPage is HomePage) {
|
if (currentPage is HomePage) {
|
||||||
if (_homeController.flag) {
|
if (_homeController.flag) {
|
||||||
@ -68,6 +68,7 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
_lastSelectTime = DateTime.now().millisecondsSinceEpoch;
|
_lastSelectTime = DateTime.now().millisecondsSinceEpoch;
|
||||||
}
|
}
|
||||||
_dynamicController.flag = true;
|
_dynamicController.flag = true;
|
||||||
|
_mainController.clearUnread();
|
||||||
} else {
|
} else {
|
||||||
_dynamicController.flag = false;
|
_dynamicController.flag = false;
|
||||||
}
|
}
|
||||||
@ -88,20 +89,23 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
Box localCache = GStrorage.localCache;
|
Box localCache = GStrorage.localCache;
|
||||||
double statusBarHeight = MediaQuery.of(context).padding.top;
|
double statusBarHeight = MediaQuery.of(context).padding.top;
|
||||||
double sheetHeight = MediaQuery.of(context).size.height -
|
double sheetHeight = MediaQuery.sizeOf(context).height -
|
||||||
MediaQuery.of(context).padding.top -
|
MediaQuery.of(context).padding.top -
|
||||||
MediaQuery.of(context).size.width * 9 / 16;
|
MediaQuery.sizeOf(context).width * 9 / 16;
|
||||||
localCache.put('sheetHeight', sheetHeight);
|
localCache.put('sheetHeight', sheetHeight);
|
||||||
localCache.put('statusBarHeight', statusBarHeight);
|
localCache.put('statusBarHeight', statusBarHeight);
|
||||||
return PopScope(
|
return PopScope(
|
||||||
onPopInvoked: (bool status) => _mainController.onBackPressed(context),
|
canPop: false,
|
||||||
|
onPopInvoked: (bool didPop) async {
|
||||||
|
_mainController.onBackPressed(context);
|
||||||
|
},
|
||||||
child: Scaffold(
|
child: Scaffold(
|
||||||
extendBody: true,
|
extendBody: true,
|
||||||
body: PageView(
|
body: PageView(
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
physics: const NeverScrollableScrollPhysics(),
|
||||||
controller: _pageController,
|
controller: _mainController.pageController,
|
||||||
onPageChanged: (index) {
|
onPageChanged: (index) {
|
||||||
selectedIndex = index;
|
_mainController.selectedIndex = index;
|
||||||
setState(() {});
|
setState(() {});
|
||||||
},
|
},
|
||||||
children: _mainController.pages,
|
children: _mainController.pages,
|
||||||
@ -116,36 +120,68 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
curve: Curves.easeInOutCubicEmphasized,
|
curve: Curves.easeInOutCubicEmphasized,
|
||||||
duration: const Duration(milliseconds: 500),
|
duration: const Duration(milliseconds: 500),
|
||||||
offset: Offset(0, snapshot.data ? 0 : 1),
|
offset: Offset(0, snapshot.data ? 0 : 1),
|
||||||
child: enableMYBar
|
child: Obx(
|
||||||
? NavigationBar(
|
() => enableMYBar
|
||||||
onDestinationSelected: (value) => setIndex(value),
|
? NavigationBar(
|
||||||
selectedIndex: selectedIndex,
|
onDestinationSelected: (value) => setIndex(value),
|
||||||
destinations: <Widget>[
|
selectedIndex: _mainController.selectedIndex,
|
||||||
..._mainController.navigationBars.map((e) {
|
destinations: <Widget>[
|
||||||
return NavigationDestination(
|
..._mainController.navigationBars.map((e) {
|
||||||
icon: e['icon'],
|
return NavigationDestination(
|
||||||
selectedIcon: e['selectIcon'],
|
icon: Obx(
|
||||||
label: e['label'],
|
() => Badge(
|
||||||
);
|
label:
|
||||||
}).toList(),
|
_mainController.dynamicBadgeType.value ==
|
||||||
],
|
DynamicBadgeMode.number
|
||||||
)
|
? Text(e['count'].toString())
|
||||||
: BottomNavigationBar(
|
: null,
|
||||||
currentIndex: selectedIndex,
|
padding:
|
||||||
onTap: (value) => setIndex(value),
|
const EdgeInsets.fromLTRB(6, 0, 6, 0),
|
||||||
iconSize: 16,
|
isLabelVisible:
|
||||||
selectedFontSize: 12,
|
_mainController.dynamicBadgeType.value !=
|
||||||
unselectedFontSize: 12,
|
DynamicBadgeMode.hidden &&
|
||||||
items: [
|
e['count'] > 0,
|
||||||
..._mainController.navigationBars.map((e) {
|
child: e['icon'],
|
||||||
return BottomNavigationBarItem(
|
),
|
||||||
icon: e['icon'],
|
),
|
||||||
activeIcon: e['selectIcon'],
|
selectedIcon: e['selectIcon'],
|
||||||
label: e['label'],
|
label: e['label'],
|
||||||
);
|
);
|
||||||
}).toList(),
|
}).toList(),
|
||||||
],
|
],
|
||||||
),
|
)
|
||||||
|
: BottomNavigationBar(
|
||||||
|
currentIndex: _mainController.selectedIndex,
|
||||||
|
onTap: (value) => setIndex(value),
|
||||||
|
iconSize: 16,
|
||||||
|
selectedFontSize: 12,
|
||||||
|
unselectedFontSize: 12,
|
||||||
|
items: [
|
||||||
|
..._mainController.navigationBars.map((e) {
|
||||||
|
return BottomNavigationBarItem(
|
||||||
|
icon: Obx(
|
||||||
|
() => Badge(
|
||||||
|
label:
|
||||||
|
_mainController.dynamicBadgeType.value ==
|
||||||
|
DynamicBadgeMode.number
|
||||||
|
? Text(e['count'].toString())
|
||||||
|
: null,
|
||||||
|
padding:
|
||||||
|
const EdgeInsets.fromLTRB(6, 0, 6, 0),
|
||||||
|
isLabelVisible:
|
||||||
|
_mainController.dynamicBadgeType.value !=
|
||||||
|
DynamicBadgeMode.hidden &&
|
||||||
|
e['count'] > 0,
|
||||||
|
child: e['icon'],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
activeIcon: e['selectIcon'],
|
||||||
|
label: e['label'],
|
||||||
|
);
|
||||||
|
}).toList(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@ -163,7 +163,7 @@ class _MediaPageState extends State<MediaPage>
|
|||||||
// const SizedBox(height: 10),
|
// const SizedBox(height: 10),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
height: 200 * MediaQuery.of(context).textScaleFactor,
|
height: MediaQuery.textScalerOf(context).scale(200),
|
||||||
child: FutureBuilder(
|
child: FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
|
|||||||
@ -41,7 +41,7 @@ class _MemberPageState extends State<MemberPage>
|
|||||||
_memberCoinsFuture = _memberController.getRecentCoinVideo();
|
_memberCoinsFuture = _memberController.getRecentCoinVideo();
|
||||||
_extendNestCtr.addListener(
|
_extendNestCtr.addListener(
|
||||||
() {
|
() {
|
||||||
double offset = _extendNestCtr.position.pixels;
|
final double offset = _extendNestCtr.position.pixels;
|
||||||
if (offset > 100) {
|
if (offset > 100) {
|
||||||
appbarStream.add(true);
|
appbarStream.add(true);
|
||||||
} else {
|
} else {
|
||||||
@ -67,7 +67,7 @@ class _MemberPageState extends State<MemberPage>
|
|||||||
title: StreamBuilder(
|
title: StreamBuilder(
|
||||||
stream: appbarStream.stream,
|
stream: appbarStream.stream,
|
||||||
initialData: false,
|
initialData: false,
|
||||||
builder: (context, AsyncSnapshot snapshot) {
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
return AnimatedOpacity(
|
return AnimatedOpacity(
|
||||||
opacity: snapshot.data ? 1 : 0,
|
opacity: snapshot.data ? 1 : 0,
|
||||||
curve: Curves.easeOut,
|
curve: Curves.easeOut,
|
||||||
@ -105,7 +105,7 @@ class _MemberPageState extends State<MemberPage>
|
|||||||
actions: [
|
actions: [
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () => Get.toNamed(
|
onPressed: () => Get.toNamed(
|
||||||
'/memberSearch?mid=${Get.parameters['mid']}&uname=${_memberController.memberInfo.value.name!}'),
|
'/memberSearch?mid=$mid&uname=${_memberController.memberInfo.value.name!}'),
|
||||||
icon: const Icon(Icons.search_outlined),
|
icon: const Icon(Icons.search_outlined),
|
||||||
),
|
),
|
||||||
PopupMenuButton(
|
PopupMenuButton(
|
||||||
|
|||||||
@ -25,7 +25,7 @@ class MemberArchiveController extends GetxController {
|
|||||||
|
|
||||||
// 获取用户投稿
|
// 获取用户投稿
|
||||||
Future getMemberArchive(type) async {
|
Future getMemberArchive(type) async {
|
||||||
if (type == 'onRefresh') {
|
if (type == 'init') {
|
||||||
pn = 1;
|
pn = 1;
|
||||||
}
|
}
|
||||||
var res = await MemberHttp.memberArchive(
|
var res = await MemberHttp.memberArchive(
|
||||||
@ -34,7 +34,12 @@ class MemberArchiveController extends GetxController {
|
|||||||
order: currentOrder['type']!,
|
order: currentOrder['type']!,
|
||||||
);
|
);
|
||||||
if (res['status']) {
|
if (res['status']) {
|
||||||
archivesList.addAll(res['data'].list.vlist);
|
if (type == 'init') {
|
||||||
|
archivesList.value = res['data'].list.vlist;
|
||||||
|
}
|
||||||
|
if (type == 'onLoad') {
|
||||||
|
archivesList.addAll(res['data'].list.vlist);
|
||||||
|
}
|
||||||
count = res['data'].page['count'];
|
count = res['data'].page['count'];
|
||||||
pn += 1;
|
pn += 1;
|
||||||
}
|
}
|
||||||
@ -42,13 +47,14 @@ class MemberArchiveController extends GetxController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
toggleSort() async {
|
toggleSort() async {
|
||||||
pn = 1;
|
List<String> typeList = orderList.map((e) => e['type']!).toList();
|
||||||
int index = orderList.indexOf(currentOrder);
|
int index = typeList.indexOf(currentOrder['type']!);
|
||||||
if (index == orderList.length - 1) {
|
if (index == orderList.length - 1) {
|
||||||
currentOrder.value = orderList.first;
|
currentOrder.value = orderList.first;
|
||||||
} else {
|
} else {
|
||||||
currentOrder.value = orderList[index + 1];
|
currentOrder.value = orderList[index + 1];
|
||||||
}
|
}
|
||||||
|
getMemberArchive('init');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 上拉加载
|
// 上拉加载
|
||||||
|
|||||||
@ -25,8 +25,7 @@ class _MemberArchivePageState extends State<MemberArchivePage> {
|
|||||||
final String heroTag = Utils.makeHeroTag(mid);
|
final String heroTag = Utils.makeHeroTag(mid);
|
||||||
_memberArchivesController =
|
_memberArchivesController =
|
||||||
Get.put(MemberArchiveController(), tag: heroTag);
|
Get.put(MemberArchiveController(), tag: heroTag);
|
||||||
_futureBuilderFuture =
|
_futureBuilderFuture = _memberArchivesController.getMemberArchive('init');
|
||||||
_memberArchivesController.getMemberArchive('onRefresh');
|
|
||||||
scrollController = _memberArchivesController.scrollController;
|
scrollController = _memberArchivesController.scrollController;
|
||||||
scrollController.addListener(
|
scrollController.addListener(
|
||||||
() {
|
() {
|
||||||
@ -48,46 +47,23 @@ class _MemberArchivePageState extends State<MemberArchivePage> {
|
|||||||
titleSpacing: 0,
|
titleSpacing: 0,
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
title: Text('他的投稿', style: Theme.of(context).textTheme.titleMedium),
|
title: Text('他的投稿', style: Theme.of(context).textTheme.titleMedium),
|
||||||
// actions: [
|
actions: [
|
||||||
// Obx(
|
Obx(
|
||||||
// () => PopupMenuButton<String>(
|
() => TextButton.icon(
|
||||||
// padding: EdgeInsets.zero,
|
icon: const Icon(Icons.sort, size: 20),
|
||||||
// tooltip: '投稿排序',
|
onPressed: _memberArchivesController.toggleSort,
|
||||||
// icon: Icon(
|
label: Text(_memberArchivesController.currentOrder['label']!),
|
||||||
// Icons.more_vert_outlined,
|
),
|
||||||
// color: Theme.of(context).colorScheme.outline,
|
),
|
||||||
// ),
|
const SizedBox(width: 6),
|
||||||
// position: PopupMenuPosition.under,
|
],
|
||||||
// onSelected: (String type) {},
|
|
||||||
// itemBuilder: (BuildContext context) => <PopupMenuEntry<String>>[
|
|
||||||
// for (var i in _memberArchivesController.orderList) ...[
|
|
||||||
// PopupMenuItem<String>(
|
|
||||||
// onTap: () {},
|
|
||||||
// value: _memberArchivesController.currentOrder['label'],
|
|
||||||
// child: Row(
|
|
||||||
// mainAxisSize: MainAxisSize.min,
|
|
||||||
// children: [
|
|
||||||
// Text(i['label']!),
|
|
||||||
// if (_memberArchivesController.currentOrder['label'] ==
|
|
||||||
// i['label']) ...[
|
|
||||||
// const SizedBox(width: 10),
|
|
||||||
// const Icon(Icons.done, size: 20),
|
|
||||||
// ],
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ]
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ],
|
|
||||||
),
|
),
|
||||||
body: CustomScrollView(
|
body: CustomScrollView(
|
||||||
controller: _memberArchivesController.scrollController,
|
controller: _memberArchivesController.scrollController,
|
||||||
slivers: [
|
slivers: [
|
||||||
FutureBuilder(
|
FutureBuilder(
|
||||||
future: _futureBuilderFuture,
|
future: _futureBuilderFuture,
|
||||||
builder: (context, snapshot) {
|
builder: (BuildContext context, snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.done) {
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
if (snapshot.data != null) {
|
if (snapshot.data != null) {
|
||||||
Map data = snapshot.data as Map;
|
Map data = snapshot.data as Map;
|
||||||
@ -97,7 +73,7 @@ class _MemberArchivePageState extends State<MemberArchivePage> {
|
|||||||
() => list.isNotEmpty
|
() => list.isNotEmpty
|
||||||
? SliverList(
|
? SliverList(
|
||||||
delegate: SliverChildBuilderDelegate(
|
delegate: SliverChildBuilderDelegate(
|
||||||
(context, index) {
|
(BuildContext context, index) {
|
||||||
return VideoCardH(
|
return VideoCardH(
|
||||||
videoItem: list[index],
|
videoItem: list[index],
|
||||||
showOwner: false,
|
showOwner: false,
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user