Compare commits
9 Commits
feature-ed
...
feature-hi
| Author | SHA1 | Date | |
|---|---|---|---|
| af1163f6e0 | |||
| fb3be848b4 | |||
| 7d7df17317 | |||
| aae08d0688 | |||
| 9fe5b78cfa | |||
| 6b028c36af | |||
| 92c385ff58 | |||
| 463ee1d5b5 | |||
| 0a416c95bc |
@ -1,10 +1,5 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
import '../../pages/dynamics/index.dart';
|
|
||||||
import '../../pages/home/index.dart';
|
|
||||||
import '../../pages/media/index.dart';
|
|
||||||
import '../../pages/rank/index.dart';
|
|
||||||
|
|
||||||
List defaultNavigationBars = [
|
List defaultNavigationBars = [
|
||||||
{
|
{
|
||||||
'id': 0,
|
'id': 0,
|
||||||
@ -18,7 +13,6 @@ List defaultNavigationBars = [
|
|||||||
),
|
),
|
||||||
'label': "首页",
|
'label': "首页",
|
||||||
'count': 0,
|
'count': 0,
|
||||||
'page': const HomePage(),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
'id': 1,
|
'id': 1,
|
||||||
@ -32,7 +26,6 @@ List defaultNavigationBars = [
|
|||||||
),
|
),
|
||||||
'label': "排行榜",
|
'label': "排行榜",
|
||||||
'count': 0,
|
'count': 0,
|
||||||
'page': const RankPage(),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
'id': 2,
|
'id': 2,
|
||||||
@ -46,7 +39,6 @@ List defaultNavigationBars = [
|
|||||||
),
|
),
|
||||||
'label': "动态",
|
'label': "动态",
|
||||||
'count': 0,
|
'count': 0,
|
||||||
'page': const DynamicsPage(),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
'id': 3,
|
'id': 3,
|
||||||
@ -60,6 +52,5 @@ List defaultNavigationBars = [
|
|||||||
),
|
),
|
||||||
'label': "媒体库",
|
'label': "媒体库",
|
||||||
'count': 0,
|
'count': 0,
|
||||||
'page': const MediaPage(),
|
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|||||||
@ -437,7 +437,8 @@ class SearchArticleItemModel {
|
|||||||
pubTime = json['pub_time'];
|
pubTime = json['pub_time'];
|
||||||
like = json['like'];
|
like = json['like'];
|
||||||
title = Em.regTitle(json['title']);
|
title = Em.regTitle(json['title']);
|
||||||
subTitle = json['title'].replaceAll(RegExp(r'<[^>]*>'), '');
|
subTitle =
|
||||||
|
Em.decodeHtmlEntities(json['title'].replaceAll(RegExp(r'<[^>]*>'), ''));
|
||||||
rankOffset = json['rank_offset'];
|
rankOffset = json['rank_offset'];
|
||||||
mid = json['mid'];
|
mid = json['mid'];
|
||||||
imageUrls = json['image_urls'];
|
imageUrls = json['image_urls'];
|
||||||
|
|||||||
@ -185,7 +185,7 @@ class HistoryItem extends StatelessWidget {
|
|||||||
? '已看完'
|
? '已看完'
|
||||||
: '${Utils.timeFormat(videoItem.progress!)}/${Utils.timeFormat(videoItem.duration!)}',
|
: '${Utils.timeFormat(videoItem.progress!)}/${Utils.timeFormat(videoItem.duration!)}',
|
||||||
right: 6.0,
|
right: 6.0,
|
||||||
bottom: 6.0,
|
bottom: 8.0,
|
||||||
type: 'gray',
|
type: 'gray',
|
||||||
),
|
),
|
||||||
// 右上角
|
// 右上角
|
||||||
@ -258,6 +258,27 @@ class HistoryItem extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
videoItem.progress != 0
|
||||||
|
? Positioned(
|
||||||
|
left: 3,
|
||||||
|
right: 3,
|
||||||
|
bottom: 0,
|
||||||
|
child: ClipRRect(
|
||||||
|
borderRadius: BorderRadius.only(
|
||||||
|
bottomLeft: Radius.circular(
|
||||||
|
StyleString.imgRadius.x),
|
||||||
|
bottomRight: Radius.circular(
|
||||||
|
StyleString.imgRadius.x),
|
||||||
|
),
|
||||||
|
child: LinearProgressIndicator(
|
||||||
|
value: videoItem.progress == -1
|
||||||
|
? 100
|
||||||
|
: videoItem.progress /
|
||||||
|
videoItem.duration,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: const SizedBox()
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
VideoContent(videoItem: videoItem, ctr: ctr)
|
VideoContent(videoItem: videoItem, ctr: ctr)
|
||||||
|
|||||||
@ -6,16 +6,23 @@ 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/http/common.dart';
|
||||||
|
import 'package:pilipala/pages/dynamics/index.dart';
|
||||||
|
import 'package:pilipala/pages/home/view.dart';
|
||||||
|
import 'package:pilipala/pages/media/index.dart';
|
||||||
|
import 'package:pilipala/pages/rank/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';
|
import '../../models/common/dynamic_badge_mode.dart';
|
||||||
import '../../models/common/nav_bar_config.dart';
|
import '../../models/common/nav_bar_config.dart';
|
||||||
|
|
||||||
class MainController extends GetxController {
|
class MainController extends GetxController {
|
||||||
List<Widget> pages = <Widget>[];
|
List<Widget> pages = <Widget>[
|
||||||
RxList navigationBars = [].obs;
|
const HomePage(),
|
||||||
late List defaultNavTabs;
|
const RankPage(),
|
||||||
late List<int> navBarSort;
|
const DynamicsPage(),
|
||||||
|
const MediaPage(),
|
||||||
|
];
|
||||||
|
RxList navigationBars = defaultNavigationBars.obs;
|
||||||
final StreamController<bool> bottomBarStream =
|
final StreamController<bool> bottomBarStream =
|
||||||
StreamController<bool>.broadcast();
|
StreamController<bool>.broadcast();
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
@ -34,7 +41,10 @@ class MainController extends GetxController {
|
|||||||
Utils.checkUpdata();
|
Utils.checkUpdata();
|
||||||
}
|
}
|
||||||
hideTabBar = setting.get(SettingBoxKey.hideTabBar, defaultValue: true);
|
hideTabBar = setting.get(SettingBoxKey.hideTabBar, defaultValue: true);
|
||||||
|
int defaultHomePage =
|
||||||
|
setting.get(SettingBoxKey.defaultHomePage, defaultValue: 0) as int;
|
||||||
|
selectedIndex = defaultNavigationBars
|
||||||
|
.indexWhere((item) => item['id'] == defaultHomePage);
|
||||||
var userInfo = userInfoCache.get('userInfoCache');
|
var userInfo = userInfoCache.get('userInfoCache');
|
||||||
userLogin.value = userInfo != null;
|
userLogin.value = userInfo != null;
|
||||||
dynamicBadgeType.value = DynamicBadgeMode.values[setting.get(
|
dynamicBadgeType.value = DynamicBadgeMode.values[setting.get(
|
||||||
@ -43,7 +53,6 @@ class MainController extends GetxController {
|
|||||||
if (dynamicBadgeType.value != DynamicBadgeMode.hidden) {
|
if (dynamicBadgeType.value != DynamicBadgeMode.hidden) {
|
||||||
getUnreadDynamic();
|
getUnreadDynamic();
|
||||||
}
|
}
|
||||||
setNavBarConfig();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void onBackPressed(BuildContext context) {
|
void onBackPressed(BuildContext context) {
|
||||||
@ -84,21 +93,4 @@ class MainController extends GetxController {
|
|||||||
}
|
}
|
||||||
navigationBars.refresh();
|
navigationBars.refresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
void setNavBarConfig() async {
|
|
||||||
defaultNavTabs = [...defaultNavigationBars];
|
|
||||||
navBarSort =
|
|
||||||
setting.get(SettingBoxKey.navBarSort, defaultValue: [0, 1, 2, 3]);
|
|
||||||
defaultNavTabs.retainWhere((item) => navBarSort.contains(item['id']));
|
|
||||||
defaultNavTabs.sort((a, b) =>
|
|
||||||
navBarSort.indexOf(a['id']).compareTo(navBarSort.indexOf(b['id'])));
|
|
||||||
navigationBars.value = defaultNavTabs;
|
|
||||||
int defaultHomePage =
|
|
||||||
setting.get(SettingBoxKey.defaultHomePage, defaultValue: 0) as int;
|
|
||||||
int defaultIndex =
|
|
||||||
navigationBars.indexWhere((item) => item['id'] == defaultHomePage);
|
|
||||||
// 如果找不到匹配项,默认索引设置为0或其他合适的值
|
|
||||||
selectedIndex = defaultIndex != -1 ? defaultIndex : 0;
|
|
||||||
pages = navigationBars.map<Widget>((e) => e['page']).toList();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,100 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
|
||||||
import 'package:hive/hive.dart';
|
|
||||||
import 'package:pilipala/models/common/tab_type.dart';
|
|
||||||
import 'package:pilipala/utils/storage.dart';
|
|
||||||
|
|
||||||
import '../../../models/common/nav_bar_config.dart';
|
|
||||||
|
|
||||||
class NavigationBarSetPage extends StatefulWidget {
|
|
||||||
const NavigationBarSetPage({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<NavigationBarSetPage> createState() => _NavigationbarSetPageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _NavigationbarSetPageState extends State<NavigationBarSetPage> {
|
|
||||||
Box settingStorage = GStrorage.setting;
|
|
||||||
late List defaultNavTabs;
|
|
||||||
late List<int> navBarSort;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
defaultNavTabs = defaultNavigationBars;
|
|
||||||
navBarSort = settingStorage
|
|
||||||
.get(SettingBoxKey.navBarSort, defaultValue: [0, 1, 2, 3]);
|
|
||||||
// 对 tabData 进行排序
|
|
||||||
defaultNavTabs.sort((a, b) {
|
|
||||||
int indexA = navBarSort.indexOf(a['id']);
|
|
||||||
int indexB = navBarSort.indexOf(b['id']);
|
|
||||||
|
|
||||||
// 如果类型在 sortOrder 中不存在,则放在末尾
|
|
||||||
if (indexA == -1) indexA = navBarSort.length;
|
|
||||||
if (indexB == -1) indexB = navBarSort.length;
|
|
||||||
|
|
||||||
return indexA.compareTo(indexB);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void saveEdit() {
|
|
||||||
List<int> sortedTabbar = defaultNavTabs
|
|
||||||
.where((i) => navBarSort.contains(i['id']))
|
|
||||||
.map<int>((i) => i['id'])
|
|
||||||
.toList();
|
|
||||||
settingStorage.put(SettingBoxKey.navBarSort, sortedTabbar);
|
|
||||||
SmartDialog.showToast('保存成功,下次启动时生效');
|
|
||||||
}
|
|
||||||
|
|
||||||
void onReorder(int oldIndex, int newIndex) {
|
|
||||||
setState(() {
|
|
||||||
if (newIndex > oldIndex) {
|
|
||||||
newIndex -= 1;
|
|
||||||
}
|
|
||||||
final tabsItem = defaultNavTabs.removeAt(oldIndex);
|
|
||||||
defaultNavTabs.insert(newIndex, tabsItem);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final listTiles = [
|
|
||||||
for (int i = 0; i < defaultNavTabs.length; i++) ...[
|
|
||||||
CheckboxListTile(
|
|
||||||
key: Key(defaultNavTabs[i]['label']),
|
|
||||||
value: navBarSort.contains(defaultNavTabs[i]['id']),
|
|
||||||
onChanged: (bool? newValue) {
|
|
||||||
int tabTypeId = defaultNavTabs[i]['id'];
|
|
||||||
if (!newValue!) {
|
|
||||||
navBarSort.remove(tabTypeId);
|
|
||||||
} else {
|
|
||||||
navBarSort.add(tabTypeId);
|
|
||||||
}
|
|
||||||
setState(() {});
|
|
||||||
},
|
|
||||||
title: Text(defaultNavTabs[i]['label']),
|
|
||||||
secondary: const Icon(Icons.drag_indicator_rounded),
|
|
||||||
enabled: defaultNavTabs[i]['id'] != 3,
|
|
||||||
)
|
|
||||||
]
|
|
||||||
];
|
|
||||||
|
|
||||||
return Scaffold(
|
|
||||||
appBar: AppBar(
|
|
||||||
title: const Text('Navbar编辑'),
|
|
||||||
actions: [
|
|
||||||
TextButton(onPressed: () => saveEdit(), child: const Text('保存')),
|
|
||||||
const SizedBox(width: 12)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
body: ReorderableListView(
|
|
||||||
onReorder: onReorder,
|
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
footer: SizedBox(
|
|
||||||
height: MediaQuery.of(context).padding.bottom + 30,
|
|
||||||
),
|
|
||||||
children: listTiles,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -284,17 +284,12 @@ class _StyleSettingState extends State<StyleSetting> {
|
|||||||
onTap: () => Get.toNamed('/tabbarSetting'),
|
onTap: () => Get.toNamed('/tabbarSetting'),
|
||||||
title: Text('首页tabbar', style: titleStyle),
|
title: Text('首页tabbar', style: titleStyle),
|
||||||
),
|
),
|
||||||
ListTile(
|
|
||||||
dense: false,
|
|
||||||
onTap: () => Get.toNamed('/navbarSetting'),
|
|
||||||
title: Text('navbar设置', style: titleStyle),
|
|
||||||
),
|
|
||||||
if (Platform.isAndroid)
|
if (Platform.isAndroid)
|
||||||
ListTile(
|
ListTile(
|
||||||
dense: false,
|
dense: false,
|
||||||
onTap: () => Get.toNamed('/displayModeSetting'),
|
onTap: () => Get.toNamed('/displayModeSetting'),
|
||||||
title: Text('屏幕帧率', style: titleStyle),
|
title: Text('屏幕帧率', style: titleStyle),
|
||||||
),
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@ -149,13 +149,16 @@ class _VideoReplyPanelState extends State<VideoReplyPanel>
|
|||||||
delegate: _MySliverPersistentHeaderDelegate(
|
delegate: _MySliverPersistentHeaderDelegate(
|
||||||
child: Container(
|
child: Container(
|
||||||
height: 40,
|
height: 40,
|
||||||
padding: const EdgeInsets.fromLTRB(12, 6, 6, 0),
|
padding: const EdgeInsets.fromLTRB(12, 0, 6, 0),
|
||||||
|
color: Theme.of(context).colorScheme.surface,
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Obx(
|
||||||
'${_videoReplyController.sortTypeLabel.value}评论',
|
() => Text(
|
||||||
style: const TextStyle(fontSize: 13),
|
'${_videoReplyController.sortTypeLabel.value}评论',
|
||||||
|
style: const TextStyle(fontSize: 13),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
height: 35,
|
height: 35,
|
||||||
|
|||||||
@ -39,7 +39,6 @@ import '../pages/setting/pages/color_select.dart';
|
|||||||
import '../pages/setting/pages/display_mode.dart';
|
import '../pages/setting/pages/display_mode.dart';
|
||||||
import '../pages/setting/pages/font_size_select.dart';
|
import '../pages/setting/pages/font_size_select.dart';
|
||||||
import '../pages/setting/pages/home_tabbar_set.dart';
|
import '../pages/setting/pages/home_tabbar_set.dart';
|
||||||
import '../pages/setting/pages/navigation_bar_set.dart';
|
|
||||||
import '../pages/setting/pages/play_gesture_set.dart';
|
import '../pages/setting/pages/play_gesture_set.dart';
|
||||||
import '../pages/setting/pages/play_speed_set.dart';
|
import '../pages/setting/pages/play_speed_set.dart';
|
||||||
import '../pages/setting/recommend_setting.dart';
|
import '../pages/setting/recommend_setting.dart';
|
||||||
@ -171,9 +170,6 @@ class Routes {
|
|||||||
// 播放器手势
|
// 播放器手势
|
||||||
CustomGetPage(
|
CustomGetPage(
|
||||||
name: '/playerGestureSet', page: () => const PlayGesturePage()),
|
name: '/playerGestureSet', page: () => const PlayGesturePage()),
|
||||||
// navigation bar
|
|
||||||
CustomGetPage(
|
|
||||||
name: '/navbarSetting', page: () => const NavigationBarSetPage()),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -19,15 +19,7 @@ class Em {
|
|||||||
return regCate(matchStr);
|
return regCate(matchStr);
|
||||||
}, onNonMatch: (String str) {
|
}, onNonMatch: (String str) {
|
||||||
if (str != '') {
|
if (str != '') {
|
||||||
str = str
|
str = decodeHtmlEntities(str);
|
||||||
.replaceAll('<', '<')
|
|
||||||
.replaceAll('>', '>')
|
|
||||||
.replaceAll('"', '"')
|
|
||||||
.replaceAll(''', "'")
|
|
||||||
.replaceAll('"', '"')
|
|
||||||
.replaceAll(''', "'")
|
|
||||||
.replaceAll(' ', " ")
|
|
||||||
.replaceAll('&', "&");
|
|
||||||
Map map = {'type': 'text', 'text': str};
|
Map map = {'type': 'text', 'text': str};
|
||||||
res.add(map);
|
res.add(map);
|
||||||
}
|
}
|
||||||
@ -35,4 +27,17 @@ class Em {
|
|||||||
});
|
});
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static String decodeHtmlEntities(String title) {
|
||||||
|
return title
|
||||||
|
.replaceAll('<', '<')
|
||||||
|
.replaceAll('>', '>')
|
||||||
|
.replaceAll('"', '"')
|
||||||
|
.replaceAll(''', "'")
|
||||||
|
.replaceAll('"', '"')
|
||||||
|
.replaceAll(''', "'")
|
||||||
|
.replaceAll(' ', " ")
|
||||||
|
.replaceAll('&', "&")
|
||||||
|
.replaceAll(''', "'");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -148,8 +148,7 @@ class SettingBoxKey {
|
|||||||
hideTabBar = 'hideTabBar', // 收起底栏
|
hideTabBar = 'hideTabBar', // 收起底栏
|
||||||
tabbarSort = 'tabbarSort', // 首页tabbar
|
tabbarSort = 'tabbarSort', // 首页tabbar
|
||||||
dynamicBadgeMode = 'dynamicBadgeMode',
|
dynamicBadgeMode = 'dynamicBadgeMode',
|
||||||
enableGradientBg = 'enableGradientBg',
|
enableGradientBg = 'enableGradientBg';
|
||||||
navBarSort = 'navBarSort';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class LocalCacheKey {
|
class LocalCacheKey {
|
||||||
|
|||||||
Reference in New Issue
Block a user