app/lib/ui/home/home_ui.dart

841 lines
32 KiB
Dart
Raw Normal View History

2024-03-09 21:53:37 +08:00
import 'package:card_swiper/card_swiper.dart';
import 'package:fluent_ui/fluent_ui.dart';
import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart';
import 'package:flutter_svg/flutter_svg.dart';
import 'package:flutter_tilt/flutter_tilt.dart';
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
2024-03-10 14:18:30 +08:00
import 'package:go_router/go_router.dart';
2024-03-09 21:53:37 +08:00
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:starcitizen_doctor/api/analytics.dart';
import 'package:starcitizen_doctor/widgets/widgets.dart';
import 'package:url_launcher/url_launcher_string.dart';
import 'dialogs/home_countdown_dialog_ui.dart';
import 'dialogs/home_md_content_dialog_ui.dart';
2024-03-09 21:53:37 +08:00
import 'home_ui_model.dart';
import 'localization/localization_dialog_ui.dart';
2024-03-10 21:25:03 +08:00
import 'localization/localization_ui_model.dart';
2024-03-09 21:53:37 +08:00
class HomeUI extends HookConsumerWidget {
const HomeUI({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
final homeState = ref.watch(homeUIModelProvider);
final model = ref.watch(homeUIModelProvider.notifier);
2024-03-10 21:25:03 +08:00
ref.watch(localizationUIModelProvider);
2024-03-09 21:53:37 +08:00
return Stack(
children: [
Center(
child: SingleChildScrollView(
padding: EdgeInsets.zero,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
if (homeState.appPlacardData != null) ...[
InfoBar(
title: Text("${homeState.appPlacardData?.title}"),
content: Text("${homeState.appPlacardData?.content}"),
severity: InfoBarSeverity.info,
action: homeState.appPlacardData?.link == null
? null
: Button(
2024-03-15 00:01:06 +08:00
child: Text(S.current.doctor_action_view_details),
2024-03-10 12:17:43 +08:00
onPressed: () => _showPlacard(context, homeState),
2024-03-09 21:53:37 +08:00
),
onClose: homeState.appPlacardData?.alwaysShow == true
? null
: () => model.closePlacard(),
),
const SizedBox(height: 6),
],
2024-03-10 21:25:03 +08:00
...makeIndex(context, model, homeState, ref)
2024-03-09 21:53:37 +08:00
],
),
),
),
if (homeState.isFixing)
Container(
decoration: BoxDecoration(
color: Colors.black.withAlpha(150),
),
child: Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
const ProgressRing(),
const SizedBox(height: 12),
Text(homeState.isFixingString.isNotEmpty
? homeState.isFixingString
2024-03-15 00:01:06 +08:00
: S.current.doctor_info_processing),
2024-03-09 21:53:37 +08:00
],
),
),
)
],
);
}
2024-03-10 21:25:03 +08:00
List<Widget> makeIndex(BuildContext context, HomeUIModel model,
HomeUIModelState homeState, WidgetRef ref) {
2024-03-15 00:01:06 +08:00
double width = 280;
2024-03-09 21:53:37 +08:00
return [
Stack(
children: [
Padding(
padding: const EdgeInsets.only(top: 64, bottom: 0),
child: SizedBox(
width: MediaQuery.of(context).size.width,
child: Center(
child: Column(
children: [
Padding(
padding: const EdgeInsets.only(bottom: 30),
child: Image.asset(
"assets/sc_logo.png",
fit: BoxFit.fitHeight,
height: 260,
),
),
makeGameStatusCard(context, model, 340, homeState)
],
),
),
),
),
Positioned(
top: 0,
left: 24,
child: makeLeftColumn(context, model, width, homeState),
),
Positioned(
right: 24,
top: 0,
child: makeNewsCard(context, model, homeState),
),
],
),
2024-09-04 19:02:31 +08:00
const SizedBox(height: 24),
Padding(
padding: const EdgeInsets.only(left: 24, right: 24),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Text(S.current.home_install_location),
const SizedBox(width: 6),
Expanded(
child: ComboBox<String>(
value: homeState.scInstalledPath,
isExpanded: true,
items: [
ComboBoxItem(
value: "not_install",
child: Text(S.current.home_not_installed_or_failed),
),
for (final path in homeState.scInstallPaths)
ComboBoxItem(
value: path,
child: Row(
children: [Text(path)],
),
)
],
onChanged: model.onChangeInstallPath,
),
),
const SizedBox(width: 12),
Button(
onPressed: homeState.webLocalizationVersionsData == null
? null
: () => model.launchRSI(context),
style: homeState.isCurGameRunning
? null
: ButtonStyle(
backgroundColor:
WidgetStateProperty.resolveWith(_getRunButtonColor),
),
child: Padding(
padding: const EdgeInsets.all(6),
child: Icon(
homeState.isCurGameRunning
? FluentIcons.stop_solid
: FluentIcons.play_solid,
color: homeState.isCurGameRunning
? Colors.red.withOpacity(.8)
: Colors.white,
),
)),
const SizedBox(width: 12),
Button(
onPressed: () {},
child: const Padding(
padding: EdgeInsets.all(6),
child: Icon(FluentIcons.folder_open),
),
),
const SizedBox(width: 12),
Button(
onPressed: model.reScanPath,
child: const Padding(
padding: EdgeInsets.all(6),
child: Icon(FluentIcons.refresh),
),
),
],
),
),
2024-03-09 21:53:37 +08:00
const SizedBox(height: 8),
Text(homeState.lastScreenInfo, maxLines: 1),
2024-03-10 21:25:03 +08:00
makeIndexActionLists(context, model, homeState, ref),
2024-03-09 21:53:37 +08:00
];
}
Widget makeLeftColumn(BuildContext context, HomeUIModel model, double width,
HomeUIModelState homeState) {
return Stack(
children: [
Column(
children: [
Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(12),
color: FluentTheme.of(context).cardColor.withOpacity(.03),
),
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
makeWebViewButton(context, model,
icon: SvgPicture.asset(
"assets/rsi.svg",
colorFilter: makeSvgColor(Colors.white),
height: 18,
),
2024-03-16 19:13:49 +08:00
name: S.current
.home_action_star_citizen_website_localization,
webTitle: S.current
.home_action_star_citizen_website_localization,
2024-03-09 21:53:37 +08:00
webURL: "https://robertsspaceindustries.com",
2024-03-16 19:13:49 +08:00
info: S.current
.home_action_info_roberts_space_industries_origin,
2024-03-09 21:53:37 +08:00
useLocalization: true,
width: width,
touchKey: "webLocalization_rsi"),
const SizedBox(height: 12),
makeWebViewButton(context, model,
icon: Row(
children: [
SvgPicture.asset(
"assets/uex.svg",
height: 18,
),
const SizedBox(width: 12),
],
),
2024-03-15 00:01:06 +08:00
name: S.current.home_action_uex_localization,
webTitle: S.current.home_action_uex_localization,
2024-03-09 21:53:37 +08:00
webURL: "https://uexcorp.space/",
2024-03-16 19:13:49 +08:00
info: S.current
.home_action_info_mining_refining_trade_calculator,
2024-03-09 21:53:37 +08:00
useLocalization: true,
width: width,
touchKey: "webLocalization_uex"),
const SizedBox(height: 12),
makeWebViewButton(context, model,
icon: Row(
children: [
Image.asset(
"assets/dps.png",
height: 20,
),
const SizedBox(width: 12),
],
),
2024-03-15 00:01:06 +08:00
name: S.current.home_action_dps_calculator_localization,
2024-03-16 19:13:49 +08:00
webTitle:
S.current.home_action_dps_calculator_localization,
2024-03-09 21:53:37 +08:00
webURL: "https://www.erkul.games/live/calculator",
2024-03-16 19:13:49 +08:00
info: S.current
.home_action_info_ship_upgrade_damage_value_query,
2024-03-09 21:53:37 +08:00
useLocalization: true,
width: width,
touchKey: "webLocalization_dps"),
const SizedBox(height: 12),
2024-03-15 00:01:06 +08:00
Text(S.current.home_action_external_browser_extension),
2024-03-09 21:53:37 +08:00
const SizedBox(height: 12),
Row(
children: [
Button(
child:
const FaIcon(FontAwesomeIcons.chrome, size: 18),
onPressed: () {
launchUrlString(
"https://chrome.google.com/webstore/detail/gocnjckojmledijgmadmacoikibcggja?authuser=0&hl=zh-CN");
},
),
const SizedBox(width: 12),
Button(
child: const FaIcon(FontAwesomeIcons.edge, size: 18),
onPressed: () {
launchUrlString(
"https://microsoftedge.microsoft.com/addons/detail/lipbbcckldklpdcpfagicipecaacikgi");
},
),
const SizedBox(width: 12),
Button(
child: const FaIcon(FontAwesomeIcons.firefoxBrowser,
size: 18),
onPressed: () {
launchUrlString(
"https://addons.mozilla.org/zh-CN/firefox/"
"addon/%E6%98%9F%E9%99%85%E5%85%AC%E6%B0%91%E7%9B%92%E5%AD%90%E6%B5%8F%E8%A7%88%E5%99%A8%E6%8B%93%E5%B1%95/");
},
),
const SizedBox(width: 12),
Button(
child:
const FaIcon(FontAwesomeIcons.github, size: 18),
onPressed: () {
launchUrlString(
"https://github.com/StarCitizenToolBox/StarCitizenBoxBrowserEx");
},
),
],
)
],
),
),
),
const SizedBox(height: 16),
makeActivityBanner(context, model, width, homeState),
],
),
if (homeState.webLocalizationVersionsData == null)
Positioned.fill(
child: Container(
decoration: BoxDecoration(
color: Colors.black.withOpacity(.3),
borderRadius: BorderRadius.circular(12)),
child: const Center(
child: ProgressRing(),
),
))
],
);
}
Widget makeNewsCard(
BuildContext context, HomeUIModel model, HomeUIModelState homeState) {
return ScrollConfiguration(
behavior: ScrollConfiguration.of(context).copyWith(scrollbars: false),
child: Container(
width: 316,
height: 386,
decoration: BoxDecoration(
color: Colors.white.withOpacity(.1),
borderRadius: BorderRadius.circular(12)),
child: SingleChildScrollView(
child: Column(
children: [
SizedBox(
height: 190,
width: 316,
child: Tilt(
shadowConfig: const ShadowConfig(maxIntensity: .3),
borderRadius: const BorderRadius.only(
topLeft: Radius.circular(12),
topRight: Radius.circular(12),
),
child: homeState.rssVideoItems == null
? Container(
decoration: BoxDecoration(
color: Colors.white.withOpacity(.1)),
child: makeLoading(context),
)
: Swiper(
2024-03-31 15:05:06 +08:00
itemCount: getMinNumber(
[homeState.rssVideoItems?.length ?? 0, 6]),
2024-03-09 21:53:37 +08:00
itemBuilder: (context, index) {
final item = homeState.rssVideoItems![index];
return GestureDetector(
onTap: () {
if (item.link != null) {
launchUrlString(item.link!);
}
},
child: CacheNetImage(
url: model.getRssImage(item),
fit: BoxFit.cover,
),
);
},
autoplay: true,
),
)),
const SizedBox(height: 1),
if (homeState.rssTextItems == null)
makeLoading(context)
else
ListView.builder(
physics: const NeverScrollableScrollPhysics(),
shrinkWrap: true,
itemBuilder: (BuildContext context, int index) {
final item = homeState.rssTextItems![index];
return Tilt(
shadowConfig: const ShadowConfig(maxIntensity: .3),
borderRadius: BorderRadius.circular(12),
child: GestureDetector(
onTap: () {
if (item.link != null) {
launchUrlString(item.link!);
}
},
child: Padding(
padding: const EdgeInsets.only(
left: 12, right: 12, top: 4, bottom: 4),
child: Row(
children: [
getRssIcon(item.link ?? ""),
const SizedBox(width: 6),
Expanded(
child: Text(
model.handleTitle(item.title),
textAlign: TextAlign.start,
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: const TextStyle(fontSize: 12.2),
),
),
const SizedBox(width: 12),
Icon(
FluentIcons.chevron_right,
size: 12,
color: Colors.white.withOpacity(.4),
)
],
),
),
));
},
itemCount: homeState.rssTextItems?.length,
),
const SizedBox(height: 12),
],
),
)),
);
}
Widget getRssIcon(String url) {
if (url.startsWith("https://tieba.baidu.com")) {
return SvgPicture.asset("assets/tieba.svg", width: 14, height: 14);
}
if (url.startsWith("https://www.bilibili.com")) {
return const FaIcon(
FontAwesomeIcons.bilibili,
size: 14,
color: Color.fromRGBO(0, 161, 214, 1),
);
}
return const FaIcon(FontAwesomeIcons.rss, size: 14);
}
2024-03-10 21:25:03 +08:00
Widget makeIndexActionLists(BuildContext context, HomeUIModel model,
HomeUIModelState homeState, WidgetRef ref) {
2024-03-09 21:53:37 +08:00
final items = [
2024-03-16 19:13:49 +08:00
_HomeItemData(
"game_doctor",
S.current.home_action_one_click_diagnosis,
S.current.home_action_info_one_click_diagnosis_star_citizen,
2024-03-09 21:53:37 +08:00
FluentIcons.auto_deploy_settings),
_HomeItemData(
2024-03-16 19:13:49 +08:00
"localization",
S.current.home_action_localization_management,
S.current.home_action_info_quick_install_localization_resources,
FluentIcons.locale_language),
_HomeItemData(
"performance",
S.current.home_action_performance_optimization,
S.current.home_action_info_engine_config_optimization,
2024-03-09 21:53:37 +08:00
FluentIcons.process_meta_task),
];
return Padding(
padding: const EdgeInsets.all(24),
child: AlignedGridView.count(
crossAxisCount: 3,
mainAxisSpacing: 12,
crossAxisSpacing: 12,
itemCount: items.length,
shrinkWrap: true,
itemBuilder: (context, index) {
final item = items.elementAt(index);
return HoverButton(
2024-03-10 21:25:03 +08:00
onPressed: () => _onMenuTap(context, item.key, homeState, ref),
2024-07-07 17:16:47 +08:00
builder: (BuildContext context, Set<WidgetState> states) {
2024-03-09 21:53:37 +08:00
return Container(
width: 300,
height: 120,
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(12),
2024-07-07 17:16:47 +08:00
color: states.isHovered
2024-03-09 21:53:37 +08:00
? FluentTheme.of(context).cardColor.withOpacity(.1)
: FluentTheme.of(context).cardColor,
),
child: Padding(
padding: const EdgeInsets.all(12),
child: Row(
children: [
Container(
decoration: BoxDecoration(
color: Colors.white.withOpacity(.2),
borderRadius: BorderRadius.circular(1000)),
child: Padding(
2024-03-13 22:40:28 +08:00
padding: const EdgeInsets.all(12),
2024-03-09 21:53:37 +08:00
child: Icon(
item.icon,
2024-03-13 22:40:28 +08:00
size: 24,
2024-03-09 21:53:37 +08:00
),
),
),
const SizedBox(width: 24),
Expanded(
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
item.name,
style: const TextStyle(fontSize: 18),
),
const SizedBox(height: 4),
2024-03-17 12:44:50 +08:00
Text(
item.infoString,
style: TextStyle(
fontSize: 14,
color: Colors.white.withOpacity(.6)),
),
2024-03-09 21:53:37 +08:00
],
)),
if (item.key == "localization" &&
homeState.localizationUpdateInfo != null)
Container(
padding: const EdgeInsets.only(
top: 3, bottom: 3, left: 8, right: 8),
decoration: BoxDecoration(
color: Colors.red,
borderRadius: BorderRadius.circular(12)),
child: Text(
homeState.localizationUpdateInfo?.key ?? " "),
),
const SizedBox(width: 12),
const Icon(
FluentIcons.chevron_right,
size: 16,
)
],
),
),
);
},
);
}),
);
}
Widget makeWebViewButton(BuildContext context, HomeUIModel model,
{required Widget icon,
required String name,
required String webTitle,
required String webURL,
required bool useLocalization,
required double width,
String? info,
String? touchKey}) {
return Tilt(
shadowConfig: const ShadowConfig(maxIntensity: .3),
borderRadius: BorderRadius.circular(12),
child: GestureDetector(
onTap: () {
if (touchKey != null) {
AnalyticsApi.touch(touchKey);
}
2024-03-10 12:07:30 +08:00
model.goWebView(context, webTitle, webURL, useLocalization: true);
2024-03-09 21:53:37 +08:00
},
child: Container(
width: width,
decoration: BoxDecoration(
color: FluentTheme.of(context).cardColor,
),
child: Padding(
padding: const EdgeInsets.all(8.0),
child: Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
icon,
Text(
name,
style: const TextStyle(fontSize: 14),
),
],
),
if (info != null)
Padding(
padding: const EdgeInsets.only(top: 4),
child: Text(
info,
2024-03-17 12:44:50 +08:00
maxLines: 1,
2024-03-09 21:53:37 +08:00
style: TextStyle(
fontSize: 12,
color: Colors.white.withOpacity(.6)),
),
)
],
),
),
const SizedBox(width: 12),
Icon(
FluentIcons.chevron_right,
size: 14,
color: Colors.white.withOpacity(.6),
)
],
),
),
),
),
);
}
Widget makeGameStatusCard(BuildContext context, HomeUIModel model,
double width, HomeUIModelState homeState) {
2024-03-15 00:01:06 +08:00
final statusCnName = {
"Platform": S.current.home_action_rsi_status_platform,
2024-03-16 19:13:49 +08:00
"Persistent Universe":
S.current.home_action_rsi_status_persistent_universe,
2024-03-15 00:01:06 +08:00
"Electronic Access": S.current.home_action_rsi_status_electronic_access,
"Arena Commander": S.current.home_action_rsi_status_arena_commander
2024-03-09 21:53:37 +08:00
};
return Tilt(
shadowConfig: const ShadowConfig(maxIntensity: .2),
borderRadius: BorderRadius.circular(12),
child: GestureDetector(
onTap: () {
2024-03-16 19:13:49 +08:00
model.goWebView(
context,
S.current.home_action_rsi_status_rsi_server_status,
2024-03-10 12:07:30 +08:00
"https://status.robertsspaceindustries.com/",
2024-03-09 21:53:37 +08:00
useLocalization: true);
},
child: Container(
width: width,
decoration: BoxDecoration(
color: FluentTheme.of(context).cardColor,
),
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(children: [
if (homeState.scServerStatus == null)
makeLoading(context, width: 20)
else
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
2024-03-15 00:01:06 +08:00
Text(S.current.home_action_rsi_status_status),
2024-03-09 21:53:37 +08:00
for (final item in homeState.scServerStatus ?? [])
Row(
children: [
SizedBox(
height: 14,
child: Center(
child: Icon(
FontAwesomeIcons.solidCircle,
color: model.isRSIServerStatusOK(item)
? Colors.green
: Colors.red,
size: 12,
),
),
),
const SizedBox(width: 5),
Text(
"${statusCnName[item["name"]] ?? item["name"]}",
style: const TextStyle(fontSize: 13),
),
],
),
Icon(
FluentIcons.chevron_right,
size: 12,
color: Colors.white.withOpacity(.4),
)
],
)
]),
),
),
),
);
}
Widget makeActivityBanner(BuildContext context, HomeUIModel model,
double width, HomeUIModelState homeState) {
return Tilt(
borderRadius: BorderRadius.circular(12),
shadowConfig: const ShadowConfig(disable: true),
child: GestureDetector(
2024-03-10 12:17:43 +08:00
onTap: () => _onTapFestival(context),
2024-03-09 21:53:37 +08:00
child: Container(
width: width + 24,
decoration: BoxDecoration(color: FluentTheme.of(context).cardColor),
child: Padding(
padding:
const EdgeInsets.only(left: 12, right: 12, top: 8, bottom: 8),
child: (homeState.countdownFestivalListData == null)
? SizedBox(
width: width,
height: 62,
child: const Center(
child: ProgressRing(),
),
)
: SizedBox(
width: width,
height: 62,
child: Swiper(
2024-03-31 15:05:06 +08:00
itemCount: getMinNumber(
[homeState.countdownFestivalListData!.length, 6]),
2024-03-09 21:53:37 +08:00
autoplay: true,
autoplayDelay: 5000,
itemBuilder: (context, index) {
final item =
homeState.countdownFestivalListData![index];
return Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
if (item.icon != null && item.icon != "") ...[
ClipRRect(
borderRadius: BorderRadius.circular(1000),
child: Image.asset(
"assets/countdown/${item.icon}",
width: 48,
height: 48,
fit: BoxFit.cover,
),
),
],
Column(
children: [
Text(
item.name ?? "",
style: const TextStyle(fontSize: 15),
),
const SizedBox(height: 3),
CountdownTimeText(
targetTime:
DateTime.fromMillisecondsSinceEpoch(
item.time ?? 0),
),
],
),
const SizedBox(width: 12),
Icon(
FluentIcons.chevron_right,
size: 14,
color: Colors.white.withOpacity(.6),
)
],
);
},
),
),
)),
),
);
}
2024-03-10 12:17:43 +08:00
_showPlacard(BuildContext context, HomeUIModelState homeState) {
switch (homeState.appPlacardData?.linkType) {
case "external":
launchUrlString(homeState.appPlacardData?.link);
return;
case "doc":
showDialog(
context: context,
builder: (context) {
return HomeMdContentDialogUI(
2024-03-16 19:13:49 +08:00
title: homeState.appPlacardData?.title ??
S.current.home_announcement_details,
url: homeState.appPlacardData?.link,
);
});
2024-03-10 12:17:43 +08:00
return;
}
}
2024-03-09 21:53:37 +08:00
2024-03-10 12:17:43 +08:00
_onTapFestival(BuildContext context) {
showDialog(
context: context, builder: (context) => const HomeCountdownDialogUI());
}
2024-03-10 14:18:30 +08:00
2024-03-10 21:25:03 +08:00
_onMenuTap(BuildContext context, String key, HomeUIModelState homeState,
WidgetRef ref) async {
2024-03-15 00:01:06 +08:00
String gameInstallReqInfo =
2024-03-16 19:13:49 +08:00
S.current.home_action_info_valid_install_location_required;
switch (key) {
case "localization":
if (homeState.scInstalledPath == "not_install") {
2024-09-04 17:18:13 +08:00
// TODO
// ToolsUIModel.English(context, showNotGameInstallMsg: true);
break;
}
2024-03-10 21:25:03 +08:00
final model = ref.watch(homeUIModelProvider.notifier);
model.checkLocalizationUpdate();
await showDialog(
2024-07-07 17:49:15 +08:00
context: context,
dismissWithEsc: false,
builder: (BuildContext context) => const LocalizationDialogUI(),
);
2024-03-10 21:25:03 +08:00
model.checkLocalizationUpdate(skipReload: true);
break;
case "performance":
if (homeState.scInstalledPath == "not_install") {
showToast(context, gameInstallReqInfo);
break;
}
context.push("/index/$key");
break;
default:
context.push("/index/$key");
}
2024-03-10 14:18:30 +08:00
}
2024-03-30 15:07:32 +08:00
2024-07-07 17:16:47 +08:00
Color? _getRunButtonColor(Set<WidgetState> states) {
if (states.isPressed) {
2024-03-30 15:07:32 +08:00
return const Color.fromRGBO(49, 227, 88, .5);
}
2024-07-07 17:16:47 +08:00
if (states.isPressed) {
2024-03-30 15:07:32 +08:00
return const Color.fromRGBO(47, 213, 84, 1.0);
}
return const Color.fromRGBO(49, 227, 88, .8);
}
2024-03-09 21:53:37 +08:00
}
class _HomeItemData {
String key;
_HomeItemData(this.key, this.name, this.infoString, this.icon);
String name;
String infoString;
IconData icon;
2024-03-16 19:13:49 +08:00
}