mirror of
https://github.com/bggRGjQaUbCoE/PiliPlus.git
synced 2026-05-31 08:08:19 +08:00
feat: navigation Bar编辑
Co-authored-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
This commit is contained in:
@@ -2,25 +2,21 @@ import 'dart:async';
|
|||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
|
|
||||||
import 'package:PiliPalaX/grpc/grpc_repo.dart';
|
import 'package:PiliPalaX/grpc/grpc_repo.dart';
|
||||||
|
import 'package:PiliPalaX/pages/dynamics/view.dart';
|
||||||
|
import 'package:PiliPalaX/pages/home/view.dart';
|
||||||
|
import 'package:PiliPalaX/pages/media/view.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.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:PiliPalaX/pages/dynamics/index.dart';
|
|
||||||
import 'package:PiliPalaX/pages/home/view.dart';
|
|
||||||
import 'package:PiliPalaX/pages/media/index.dart';
|
|
||||||
import 'package:PiliPalaX/utils/storage.dart';
|
import 'package:PiliPalaX/utils/storage.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>[];
|
||||||
const HomePage(),
|
RxList navigationBars = [].obs;
|
||||||
// const RankPage(),
|
|
||||||
const DynamicsPage(),
|
|
||||||
const MediaPage(),
|
|
||||||
];
|
|
||||||
RxList navigationBars = defaultNavigationBars.obs;
|
|
||||||
final StreamController<bool> bottomBarStream =
|
final StreamController<bool> bottomBarStream =
|
||||||
StreamController<bool>.broadcast();
|
StreamController<bool>.broadcast();
|
||||||
Box setting = GStorage.setting;
|
Box setting = GStorage.setting;
|
||||||
@@ -33,30 +29,31 @@ class MainController extends GetxController {
|
|||||||
late bool checkDynamic;
|
late bool checkDynamic;
|
||||||
late int dynamicPeriod;
|
late int dynamicPeriod;
|
||||||
int? _lastCheckAt;
|
int? _lastCheckAt;
|
||||||
|
int? dynIndex;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
super.onInit();
|
super.onInit();
|
||||||
checkDynamic = GStorage.checkDynamic;
|
checkDynamic = GStorage.checkDynamic;
|
||||||
dynamicPeriod = GStorage.dynamicPeriod;
|
dynamicPeriod = GStorage.dynamicPeriod;
|
||||||
// if (setting.get(SettingBoxKey.autoUpdate, defaultValue: false)) {
|
|
||||||
// Utils.checkUpdate();
|
|
||||||
// }
|
|
||||||
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);
|
|
||||||
dynamic userInfo = userInfoCache.get('userInfoCache');
|
dynamic userInfo = userInfoCache.get('userInfoCache');
|
||||||
userLogin.value = userInfo != null;
|
userLogin.value = userInfo != null;
|
||||||
dynamicBadgeType = DynamicBadgeMode.values[setting.get(
|
dynamicBadgeType = DynamicBadgeMode.values[setting.get(
|
||||||
SettingBoxKey.dynamicBadgeMode,
|
SettingBoxKey.dynamicBadgeMode,
|
||||||
defaultValue: DynamicBadgeMode.number.code)];
|
defaultValue: DynamicBadgeMode.number.code)];
|
||||||
|
|
||||||
|
setNavBarConfig();
|
||||||
if (dynamicBadgeType != DynamicBadgeMode.hidden) {
|
if (dynamicBadgeType != DynamicBadgeMode.hidden) {
|
||||||
|
dynIndex = navigationBars.indexWhere((e) => e['id'] == 1);
|
||||||
|
if (dynIndex != -1) {
|
||||||
if (checkDynamic) {
|
if (checkDynamic) {
|
||||||
_lastCheckAt = DateTime.now().millisecondsSinceEpoch;
|
_lastCheckAt = DateTime.now().millisecondsSinceEpoch;
|
||||||
}
|
}
|
||||||
getUnreadDynamic();
|
getUnreadDynamic();
|
||||||
|
} else {
|
||||||
|
checkDynamic = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -82,36 +79,28 @@ class MainController extends GetxController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void getUnreadDynamic() async {
|
void getUnreadDynamic() async {
|
||||||
if (!userLogin.value) {
|
if (!userLogin.value || dynIndex == null || dynIndex == -1) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// not needed yet
|
|
||||||
// int dynamicItemIndex =
|
|
||||||
// navigationBars.indexWhere((item) => item['label'] == "动态");
|
|
||||||
// if (dynamicItemIndex == -1) return;
|
|
||||||
// var res = await CommonHttp.unReadDynamic();
|
|
||||||
// var data = res['data'];
|
|
||||||
// navigationBars[1]['count'] =
|
|
||||||
// data == null ? 0 : data.length; // 修改 count 属性为新的值
|
|
||||||
await GrpcRepo.dynRed().then((res) {
|
await GrpcRepo.dynRed().then((res) {
|
||||||
if (res['status']) {
|
if (res['status']) {
|
||||||
navigationBars[1]['count'] = res['data'];
|
navigationBars[dynIndex!]['count'] = res['data'];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
navigationBars.refresh();
|
navigationBars.refresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
void clearUnread() async {
|
void clearUnread() async {
|
||||||
// not needed yet
|
if (dynamicBadgeType != DynamicBadgeMode.hidden) {
|
||||||
// int dynamicItemIndex =
|
navigationBars[dynIndex!]['count'] = 0; // 修改 count 属性为新的值
|
||||||
// navigationBars.indexWhere((item) => item['label'] == "动态");
|
|
||||||
// if (dynamicItemIndex == -1) return;
|
|
||||||
navigationBars[1]['count'] = 0; // 修改 count 属性为新的值
|
|
||||||
navigationBars.refresh();
|
navigationBars.refresh();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void checkUnreadDynamic() {
|
void checkUnreadDynamic() {
|
||||||
if (!userLogin.value ||
|
if (dynIndex == null ||
|
||||||
|
dynIndex == -1 ||
|
||||||
|
!userLogin.value ||
|
||||||
dynamicBadgeType == DynamicBadgeMode.hidden ||
|
dynamicBadgeType == DynamicBadgeMode.hidden ||
|
||||||
!checkDynamic) return;
|
!checkDynamic) return;
|
||||||
int now = DateTime.now().millisecondsSinceEpoch;
|
int now = DateTime.now().millisecondsSinceEpoch;
|
||||||
@@ -120,4 +109,28 @@ class MainController extends GetxController {
|
|||||||
getUnreadDynamic();
|
getUnreadDynamic();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setNavBarConfig() async {
|
||||||
|
List defaultNavTabs = [...defaultNavigationBars];
|
||||||
|
List navBarSort =
|
||||||
|
setting.get(SettingBoxKey.navBarSort, defaultValue: [0, 1, 2]);
|
||||||
|
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) => switch (e['id']) {
|
||||||
|
0 => HomePage(),
|
||||||
|
1 => DynamicsPage(),
|
||||||
|
2 => MediaPage(),
|
||||||
|
_ => throw UnimplementedError(),
|
||||||
|
})
|
||||||
|
.toList();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -152,14 +152,16 @@ class _MainAppState extends State<MainApp>
|
|||||||
36.801 +
|
36.801 +
|
||||||
MediaQuery.of(context).padding.left,
|
MediaQuery.of(context).padding.left,
|
||||||
child: Obx(
|
child: Obx(
|
||||||
() => NavigationRail(
|
() => _mainController.navigationBars.length > 1
|
||||||
|
? NavigationRail(
|
||||||
groupAlignment: 1,
|
groupAlignment: 1,
|
||||||
minWidth: context.width * 0.0286 + 28.56,
|
minWidth: context.width * 0.0286 + 28.56,
|
||||||
backgroundColor: Colors.transparent,
|
backgroundColor: Colors.transparent,
|
||||||
selectedIndex: _mainController.selectedIndex,
|
selectedIndex: _mainController.selectedIndex,
|
||||||
onDestinationSelected: (value) => setIndex(value),
|
onDestinationSelected: setIndex,
|
||||||
labelType: NavigationRailLabelType.none,
|
labelType: NavigationRailLabelType.none,
|
||||||
leading: UserAndSearchVertical(ctr: _homeController),
|
leading:
|
||||||
|
UserAndSearchVertical(ctr: _homeController),
|
||||||
destinations: _mainController.navigationBars
|
destinations: _mainController.navigationBars
|
||||||
.map((e) => NavigationRailDestination(
|
.map((e) => NavigationRailDestination(
|
||||||
icon: _buildIcon(
|
icon: _buildIcon(
|
||||||
@@ -178,13 +180,23 @@ class _MainAppState extends State<MainApp>
|
|||||||
))
|
))
|
||||||
.toList(),
|
.toList(),
|
||||||
trailing: SizedBox(height: 0.1 * context.height),
|
trailing: SizedBox(height: 0.1 * context.height),
|
||||||
|
)
|
||||||
|
: Container(
|
||||||
|
padding: EdgeInsets.only(
|
||||||
|
top: MediaQuery.paddingOf(context).top + 10),
|
||||||
|
constraints: BoxConstraints(
|
||||||
|
minWidth: context.width * 0.0286 + 28.56,
|
||||||
|
),
|
||||||
|
child:
|
||||||
|
UserAndSearchVertical(ctr: _homeController),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
] else if (!isPortait)
|
] else if (!isPortait)
|
||||||
Obx(
|
Obx(
|
||||||
() => NavigationRail(
|
() => _mainController.navigationBars.length > 1
|
||||||
onDestinationSelected: (value) => setIndex(value),
|
? NavigationRail(
|
||||||
|
onDestinationSelected: setIndex,
|
||||||
selectedIndex: _mainController.selectedIndex,
|
selectedIndex: _mainController.selectedIndex,
|
||||||
destinations: _mainController.navigationBars
|
destinations: _mainController.navigationBars
|
||||||
.map(
|
.map(
|
||||||
@@ -203,7 +215,8 @@ class _MainAppState extends State<MainApp>
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
.toList(),
|
.toList(),
|
||||||
),
|
)
|
||||||
|
: const SizedBox.shrink(),
|
||||||
),
|
),
|
||||||
VerticalDivider(
|
VerticalDivider(
|
||||||
width: 1,
|
width: 1,
|
||||||
@@ -240,10 +253,11 @@ class _MainAppState extends State<MainApp>
|
|||||||
offset: Offset(0, snapshot.data ? 0 : 1),
|
offset: Offset(0, snapshot.data ? 0 : 1),
|
||||||
child: enableMYBar
|
child: enableMYBar
|
||||||
? Obx(
|
? Obx(
|
||||||
() => NavigationBar(
|
() => _mainController.navigationBars.length > 1
|
||||||
onDestinationSelected: (value) =>
|
? NavigationBar(
|
||||||
setIndex(value),
|
onDestinationSelected: setIndex,
|
||||||
selectedIndex: _mainController.selectedIndex,
|
selectedIndex:
|
||||||
|
_mainController.selectedIndex,
|
||||||
destinations:
|
destinations:
|
||||||
_mainController.navigationBars.map(
|
_mainController.navigationBars.map(
|
||||||
(e) {
|
(e) {
|
||||||
@@ -262,12 +276,15 @@ class _MainAppState extends State<MainApp>
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
).toList(),
|
).toList(),
|
||||||
),
|
)
|
||||||
|
: const SizedBox.shrink(),
|
||||||
)
|
)
|
||||||
: Obx(
|
: Obx(
|
||||||
() => BottomNavigationBar(
|
() => _mainController.navigationBars.length > 1
|
||||||
currentIndex: _mainController.selectedIndex,
|
? BottomNavigationBar(
|
||||||
onTap: (value) => setIndex(value),
|
currentIndex:
|
||||||
|
_mainController.selectedIndex,
|
||||||
|
onTap: setIndex,
|
||||||
iconSize: 16,
|
iconSize: 16,
|
||||||
selectedFontSize: 12,
|
selectedFontSize: 12,
|
||||||
unselectedFontSize: 12,
|
unselectedFontSize: 12,
|
||||||
@@ -293,7 +310,8 @@ class _MainAppState extends State<MainApp>
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
.toList(),
|
.toList(),
|
||||||
),
|
)
|
||||||
|
: const SizedBox.shrink(),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import 'package:PiliPalaX/http/loading_state.dart';
|
import 'package:PiliPalaX/http/loading_state.dart';
|
||||||
import 'package:PiliPalaX/models/home/rcmd/result.dart';
|
|
||||||
import 'package:PiliPalaX/pages/common/popup_controller.dart';
|
import 'package:PiliPalaX/pages/common/popup_controller.dart';
|
||||||
import 'package:PiliPalaX/http/video.dart';
|
import 'package:PiliPalaX/http/video.dart';
|
||||||
import 'package:PiliPalaX/utils/storage.dart';
|
import 'package:PiliPalaX/utils/storage.dart';
|
||||||
|
|||||||
100
lib/pages/setting/navigation_bar_set.dart
Normal file
100
lib/pages/setting/navigation_bar_set.dart
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
import 'package:PiliPalaX/utils/storage.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.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> {
|
||||||
|
late List defaultNavTabs;
|
||||||
|
late List<int> navBarSort;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
defaultNavTabs = defaultNavigationBars;
|
||||||
|
navBarSort =
|
||||||
|
GStorage.setting.get(SettingBoxKey.navBarSort, defaultValue: [0, 1, 2]);
|
||||||
|
// 对 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();
|
||||||
|
if (sortedTabbar.isEmpty) {
|
||||||
|
sortedTabbar = [0, 1, 2];
|
||||||
|
}
|
||||||
|
GStorage.setting.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) {
|
||||||
|
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: defaultNavTabs
|
||||||
|
.map(
|
||||||
|
(item) => CheckboxListTile(
|
||||||
|
key: Key(item['label']),
|
||||||
|
value: navBarSort.contains(item['id']),
|
||||||
|
onChanged: (bool? newValue) {
|
||||||
|
int tabTypeId = item['id'];
|
||||||
|
if (!newValue!) {
|
||||||
|
navBarSort.remove(tabTypeId);
|
||||||
|
} else {
|
||||||
|
navBarSort.add(tabTypeId);
|
||||||
|
}
|
||||||
|
setState(() {});
|
||||||
|
},
|
||||||
|
title: Text(item['label']),
|
||||||
|
secondary: const Icon(Icons.drag_indicator_rounded),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -359,13 +359,19 @@ class _StyleSettingState extends State<StyleSetting> {
|
|||||||
subtitle: Text('删除或调换首页标签页', style: subTitleStyle),
|
subtitle: Text('删除或调换首页标签页', style: subTitleStyle),
|
||||||
leading: const Icon(Icons.toc_outlined),
|
leading: const Icon(Icons.toc_outlined),
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
dense: false,
|
||||||
|
onTap: () => Get.toNamed('/navbarSetting'),
|
||||||
|
title: Text('Navbar编辑', style: titleStyle),
|
||||||
|
leading: const Icon(Icons.toc_outlined),
|
||||||
|
),
|
||||||
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),
|
||||||
leading: const Icon(Icons.autofps_select_outlined),
|
leading: const Icon(Icons.autofps_select_outlined),
|
||||||
)
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
import 'package:PiliPalaX/pages/member/new/member_page.dart';
|
import 'package:PiliPalaX/pages/member/new/member_page.dart';
|
||||||
import 'package:PiliPalaX/pages/member/new/widget/edit_profile_page.dart';
|
import 'package:PiliPalaX/pages/member/new/widget/edit_profile_page.dart';
|
||||||
|
import 'package:PiliPalaX/pages/setting/navigation_bar_set.dart';
|
||||||
import 'package:PiliPalaX/pages/setting/sponsor_block_page.dart';
|
import 'package:PiliPalaX/pages/setting/sponsor_block_page.dart';
|
||||||
import 'package:PiliPalaX/pages/video/detail/introduction/widgets/create_fav_page.dart';
|
import 'package:PiliPalaX/pages/video/detail/introduction/widgets/create_fav_page.dart';
|
||||||
import 'package:PiliPalaX/pages/webview/webview_page.dart';
|
import 'package:PiliPalaX/pages/webview/webview_page.dart';
|
||||||
@@ -186,6 +187,9 @@ class Routes {
|
|||||||
CustomGetPage(name: '/sponsorBlock', page: () => const SponsorBlockPage()),
|
CustomGetPage(name: '/sponsorBlock', page: () => const SponsorBlockPage()),
|
||||||
CustomGetPage(name: '/createFav', page: () => const CreateFavPage()),
|
CustomGetPage(name: '/createFav', page: () => const CreateFavPage()),
|
||||||
CustomGetPage(name: '/editProfile', page: () => const EditProfilePage()),
|
CustomGetPage(name: '/editProfile', page: () => const EditProfilePage()),
|
||||||
|
// navigation bar
|
||||||
|
CustomGetPage(
|
||||||
|
name: '/navbarSetting', page: () => const NavigationBarSetPage()),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -304,7 +304,8 @@ class SettingBoxKey {
|
|||||||
tabbarSort = 'tabbarSort', // 首页tabbar
|
tabbarSort = 'tabbarSort', // 首页tabbar
|
||||||
dynamicBadgeMode = 'dynamicBadgeMode',
|
dynamicBadgeMode = 'dynamicBadgeMode',
|
||||||
hiddenSettingUnlocked = 'hiddenSettingUnlocked',
|
hiddenSettingUnlocked = 'hiddenSettingUnlocked',
|
||||||
enableGradientBg = 'enableGradientBg';
|
enableGradientBg = 'enableGradientBg',
|
||||||
|
navBarSort = 'navBarSort';
|
||||||
}
|
}
|
||||||
|
|
||||||
class LocalCacheKey {
|
class LocalCacheKey {
|
||||||
|
|||||||
Reference in New Issue
Block a user