This commit is contained in:
2026-03-07 17:24:59 +08:00
parent 4418ebecac
commit b0ec8ab4bd
417 changed files with 42546 additions and 2 deletions

View File

@@ -0,0 +1,22 @@
import 'package:flutter_dmzj/app/controller/base_controller.dart';
import 'package:flutter_dmzj/models/novel/category_model.dart';
import 'package:flutter_dmzj/requests/novel_request.dart';
import 'package:flutter_dmzj/routes/app_navigator.dart';
class NovelCategoryController extends BasePageController<NovelCategoryModel> {
final NovelRequest request = NovelRequest();
@override
Future<List<NovelCategoryModel>> getData(int page, int pageSize) async {
if (page > 1) {
return [];
}
var ls = await request.categores();
return ls;
}
void toDetail(NovelCategoryModel item) {
AppNavigator.toNovelCategoryDetail(item.tagId);
}
}

View File

@@ -0,0 +1,61 @@
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/app_style.dart';
import 'package:flutter_dmzj/modules/novel/home/category/novel_category_controller.dart';
import 'package:flutter_dmzj/widgets/keep_alive_wrapper.dart';
import 'package:flutter_dmzj/widgets/net_image.dart';
import 'package:flutter_dmzj/widgets/page_grid_view.dart';
import 'package:flutter_dmzj/widgets/shadow_card.dart';
import 'package:get/get.dart';
class NovelCategoryView extends StatelessWidget {
final NovelCategoryController controller;
NovelCategoryView({Key? key})
: controller = Get.put(NovelCategoryController()),
super(key: key);
@override
Widget build(BuildContext context) {
return LayoutBuilder(builder: (context, constraints) {
var count = constraints.maxWidth ~/ 160;
if (count < 3) count = 3;
return KeepAliveWrapper(
child: PageGridView(
pageController: controller,
firstRefresh: true,
loadMore: false,
crossAxisCount: count,
padding: AppStyle.edgeInsetsH12.copyWith(bottom: 12),
mainAxisSpacing: 12,
crossAxisSpacing: 12,
itemBuilder: (context, i) {
var item = controller.list[i];
return ShadowCard(
onTap: () {
controller.toDetail(item);
},
child: Column(
children: [
AspectRatio(
aspectRatio: 1.0,
child: NetImage(
item.cover,
borderRadius: 8,
),
),
Padding(
padding: AppStyle.edgeInsetsA8,
child: Text(
item.title,
textAlign: TextAlign.center,
style: const TextStyle(height: 1),
),
),
],
),
);
},
),
);
});
}
}

View File

@@ -0,0 +1,14 @@
import 'package:flutter_dmzj/app/controller/base_controller.dart';
import 'package:flutter_dmzj/models/novel/latest_model.dart';
import 'package:flutter_dmzj/requests/novel_request.dart';
class NovelLatestController extends BasePageController<NovelLatestModel> {
final NovelRequest request = NovelRequest();
@override
Future<List<NovelLatestModel>> getData(int page, int pageSize) async {
var ls = await request.latest(page: page);
return ls;
}
}

View File

@@ -0,0 +1,121 @@
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/app_constant.dart';
import 'package:flutter_dmzj/app/app_style.dart';
import 'package:flutter_dmzj/models/novel/latest_model.dart';
import 'package:flutter_dmzj/modules/novel/home/latest/novel_latest_controller.dart';
import 'package:flutter_dmzj/routes/app_navigator.dart';
import 'package:flutter_dmzj/services/user_service.dart';
import 'package:flutter_dmzj/widgets/keep_alive_wrapper.dart';
import 'package:flutter_dmzj/widgets/net_image.dart';
import 'package:flutter_dmzj/widgets/page_list_view.dart';
import 'package:get/get.dart';
class NovelLatestView extends StatelessWidget {
final NovelLatestController controller;
NovelLatestView({Key? key})
: controller = Get.put(NovelLatestController()),
super(key: key);
@override
Widget build(BuildContext context) {
return KeepAliveWrapper(
child: PageListView(
pageController: controller,
firstRefresh: true,
showPageLoadding: false,
separatorBuilder: (context, i) => Divider(
endIndent: 12,
indent: 12,
color: Colors.grey.withOpacity(.2),
height: 1,
),
itemBuilder: (context, i) {
var item = controller.list[i];
return buildItem(item);
},
),
);
}
Widget buildItem(NovelLatestModel item) {
return InkWell(
onTap: () {
AppNavigator.toNovelDetail(item.id);
},
child: Container(
padding: AppStyle.edgeInsetsA12,
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
NetImage(
item.cover ?? "",
width: 80,
height: 110,
borderRadius: 4,
),
AppStyle.hGap12,
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Text(
item.title,
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
const SizedBox(height: 2),
Text.rich(
TextSpan(children: [
const WidgetSpan(
child: Icon(
Icons.account_circle,
color: Colors.grey,
size: 18,
)),
const TextSpan(
text: " ",
),
TextSpan(
text: item.authors,
style:
const TextStyle(color: Colors.grey, fontSize: 14))
]),
),
const SizedBox(height: 2),
Text(item.types ?? "",
style: const TextStyle(color: Colors.grey, fontSize: 14)),
const SizedBox(height: 2),
Text(item.lastName ?? "",
style: const TextStyle(color: Colors.grey, fontSize: 14)),
],
),
),
Center(
child: Obx(
() => UserService.instance.subscribedNovelIds.contains(item.id)
? IconButton(
icon: const Icon(Icons.favorite),
onPressed: () {
UserService.instance.cancelSubscribe(
[item.id],
AppConstant.kTypeNovel,
);
},
)
: IconButton(
icon: const Icon(Icons.favorite_border),
onPressed: () {
UserService.instance.addSubscribe(
[item.id],
AppConstant.kTypeNovel,
);
},
),
),
)
],
),
),
);
}
}

View File

@@ -0,0 +1,52 @@
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/controller/base_controller.dart';
import 'package:flutter_dmzj/app/event_bus.dart';
import 'package:flutter_dmzj/modules/novel/home/category/novel_category_controller.dart';
import 'package:flutter_dmzj/modules/novel/home/latest/novel_latest_controller.dart';
import 'package:flutter_dmzj/modules/novel/home/rank/novel_rank_controller.dart';
import 'package:flutter_dmzj/modules/novel/home/recommend/novel_recommend_controller.dart';
import 'package:flutter_dmzj/routes/app_navigator.dart';
import 'package:get/get.dart';
class NovelHomeController extends GetxController
with GetTickerProviderStateMixin {
late TabController tabController;
StreamSubscription<dynamic>? streamSubscription;
@override
void onInit() {
streamSubscription = EventBus.instance.listen(
EventBus.kBottomNavigationBarClicked,
(index) {
if (index == 2) {
refreshOrScrollTop();
}
},
);
tabController = TabController(length: 3, vsync: this);
super.onInit();
}
void refreshOrScrollTop() {
var tabIndex = tabController.index;
BasePageController? controller;
if (tabIndex == 0) {
controller = Get.find<NovelRecommendController>();
} else if (tabIndex == 1) {
controller = Get.find<NovelLatestController>();
} else if (tabIndex == 2) {
controller = Get.find<NovelCategoryController>();
} else if (tabIndex == 3) {
controller = Get.find<NovelRankController>();
}
controller?.scrollToTopOrRefresh();
}
void search() {
AppNavigator.toNovelSearch();
}
}

View File

@@ -0,0 +1,57 @@
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/platform_utils.dart';
import 'package:flutter_dmzj/modules/novel/home/category/novel_category_view.dart';
import 'package:flutter_dmzj/modules/novel/home/latest/novel_latest_view.dart';
import 'package:flutter_dmzj/modules/novel/home/novel_home_controller.dart';
import 'package:flutter_dmzj/modules/novel/home/recommend/novel_recommend_view.dart';
import 'package:flutter_dmzj/widgets/tab_appbar.dart';
import 'package:flutter_dmzj/widgets/windows_tab_page.dart';
import 'package:get/get.dart';
class NovelHomePage extends GetView<NovelHomeController> {
const NovelHomePage({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
if (PlatformUtils.isWindows) {
return WindowsTabPage(
tabs: [
WindowsTabItem(label: '推荐', body: NovelRecommendView()),
WindowsTabItem(label: '更新', body: NovelLatestView()),
WindowsTabItem(label: '分类', body: NovelCategoryView()),
],
headerAction: IconButton(
onPressed: controller.search,
icon: const Icon(Icons.search),
),
);
}
return Scaffold(
appBar: TabAppBar(
tabs: const [
Tab(text: "推荐"),
Tab(text: "更新"),
Tab(text: "分类"),
//Tab(text: "排行"),
],
controller: controller.tabController,
action: IconButton(
onPressed: controller.search,
icon: const Icon(
Icons.search,
),
),
),
body: TabBarView(
controller: controller.tabController,
children: [
NovelRecommendView(),
NovelLatestView(),
NovelCategoryView(),
//NovelRankView(),
],
),
);
}
}

View File

@@ -0,0 +1,44 @@
import 'package:flutter_dmzj/app/controller/base_controller.dart';
import 'package:flutter_dmzj/models/novel/rank_model.dart';
import 'package:flutter_dmzj/requests/novel_request.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
class NovelRankController extends BasePageController<NovelRankModel> {
final NovelRequest request = NovelRequest();
RxMap<int, String> tags = {
0: "全部分类",
}.obs;
var tag = 0.obs;
Map<int, String> rankTypes = {
0: "人气排行",
1: "订阅排行",
};
var rankType = 0.obs;
@override
void onInit() {
loadFilter();
super.onInit();
}
void loadFilter() async {
try {
tags.value = await request.rankFilter();
} catch (e) {
SmartDialog.showToast(e.toString());
}
}
@override
Future<List<NovelRankModel>> getData(int page, int pageSize) async {
var ls = await request.rank(
tagId: tag.value,
sort: rankType.value,
page: page - 1,
);
return ls;
}
}

View File

@@ -0,0 +1,191 @@
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/app_constant.dart';
import 'package:flutter_dmzj/app/app_style.dart';
import 'package:flutter_dmzj/app/utils.dart';
import 'package:flutter_dmzj/models/novel/rank_model.dart';
import 'package:flutter_dmzj/modules/novel/home/rank/novel_rank_controller.dart';
import 'package:flutter_dmzj/routes/app_navigator.dart';
import 'package:flutter_dmzj/services/user_service.dart';
import 'package:flutter_dmzj/widgets/keep_alive_wrapper.dart';
import 'package:flutter_dmzj/widgets/net_image.dart';
import 'package:flutter_dmzj/widgets/page_list_view.dart';
import 'package:get/get.dart';
class NovelRankView extends StatelessWidget {
final NovelRankController controller;
NovelRankView({Key? key})
: controller = Get.put(NovelRankController()),
super(key: key);
@override
Widget build(BuildContext context) {
return KeepAliveWrapper(
child: Column(
children: [
Obx(
() => Row(
children: [
buildFilter(
// ignore: invalid_use_of_protected_member
types: controller.tags.value,
value: controller.tag.value,
onSelected: (e) {
controller.tag.value = e;
controller.refreshData();
},
),
buildFilter(
types: controller.rankTypes,
value: controller.rankType.value,
onSelected: (e) {
controller.rankType.value = e;
controller.refreshData();
},
),
],
),
),
AppStyle.vGap12,
Expanded(
child: PageListView(
pageController: controller,
firstRefresh: true,
showPageLoadding: false,
separatorBuilder: (context, i) => Divider(
endIndent: 12,
indent: 12,
color: Colors.grey.withOpacity(.2),
height: 1,
),
itemBuilder: (context, i) {
var item = controller.list[i];
return buildItem(item);
},
),
),
],
),
);
}
Widget buildFilter({
required Map<int, String> types,
required int value,
required Function(int) onSelected,
}) {
return Expanded(
child: PopupMenuButton<int>(
onSelected: onSelected,
itemBuilder: (c) => types.keys
.map(
(k) => CheckedPopupMenuItem<int>(
value: k,
checked: k == value,
child: Text(types[k] ?? ""),
),
)
.toList(),
child: SizedBox(
height: 36,
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Text(
types[value] ?? "",
),
const Icon(
Icons.arrow_drop_down,
color: Colors.grey,
)
],
),
),
),
);
}
Widget buildItem(NovelRankModel item) {
return InkWell(
onTap: () {
AppNavigator.toNovelDetail(item.id);
},
child: Container(
padding: AppStyle.edgeInsetsA12,
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
NetImage(
item.cover,
width: 80,
height: 110,
borderRadius: 4,
),
AppStyle.hGap12,
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Text(
item.name,
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
const SizedBox(height: 2),
Text.rich(
TextSpan(children: [
const WidgetSpan(
child: Icon(
Icons.account_circle,
color: Colors.grey,
size: 18,
)),
const TextSpan(
text: " ",
),
TextSpan(
text: item.authors,
style:
const TextStyle(color: Colors.grey, fontSize: 14))
]),
),
const SizedBox(height: 2),
Text(item.types.join("/"),
style: const TextStyle(color: Colors.grey, fontSize: 14)),
const SizedBox(height: 2),
Text(item.lastUpdateChapterName,
style: const TextStyle(color: Colors.grey, fontSize: 14)),
const SizedBox(height: 2),
Text("更新于${Utils.formatTimestamp(item.lastUpdateTime)}",
style: const TextStyle(color: Colors.grey, fontSize: 14)),
],
),
),
Center(
child: Obx(
() => UserService.instance.subscribedNovelIds.contains(item.id)
? IconButton(
icon: const Icon(Icons.favorite),
onPressed: () {
UserService.instance.cancelSubscribe(
[item.id],
AppConstant.kTypeNovel,
);
},
)
: IconButton(
icon: const Icon(Icons.favorite_border),
onPressed: () {
UserService.instance.addSubscribe(
[item.id],
AppConstant.kTypeNovel,
);
},
),
),
)
],
),
),
);
}
}

View File

@@ -0,0 +1,68 @@
import 'dart:async';
import 'package:flutter_dmzj/app/controller/base_controller.dart';
import 'package:flutter_dmzj/models/novel/recommend_model.dart';
import 'package:flutter_dmzj/modules/novel/home/novel_home_controller.dart';
import 'package:flutter_dmzj/requests/novel_request.dart';
import 'package:flutter_dmzj/routes/app_navigator.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:url_launcher/url_launcher_string.dart';
class NovelRecommendController extends BasePageController<NovelRecommendModel> {
final NovelRequest request = NovelRequest();
@override
Future<List<NovelRecommendModel>> getData(int page, int pageSize) async {
var ls = await request.recommend();
return ls;
}
void openDetail(NovelRecommendItemModel item) {
//漫画=1
if (item.type == null || item.type == 2) {
AppNavigator.toNovelDetail(
item.objId ?? item.id ?? 0,
);
} else if (item.type == 1) {
//专题=5
AppNavigator.toComicDetail(
item.objId ?? 0,
);
} else if (item.type == 5) {
//专题=5
AppNavigator.toSpecialDetail(
item.objId ?? 0,
);
} else if (item.type == 6) {
//网页=6
AppNavigator.toWebView(item.url ?? "");
} else if (item.type == 7) {
//新闻=7
AppNavigator.toNewsDetail(
url: item.url ?? "",
newsId: item.objId ?? 0,
title: item.title,
);
} else if (item.type == 8) {
//作者=8
AppNavigator.toComicAuthorDetail(item.objId ?? 0);
} else if (item.type == 13) {
//社区=13
//直接跳转至网页
launchUrlString(
"http://m.forum.idmzj.com/thread/detail?tid=${item.objId}");
} else {
SmartDialog.showToast("未知类型,无法跳转");
}
}
void toLatest() {
var homeController = Get.find<NovelHomeController>();
homeController.tabController.animateTo(1);
}
void toMySubscribe() {}
}

View File

@@ -0,0 +1,229 @@
import 'package:flutter/material.dart';
import 'package:flutter_dmzj/app/app_style.dart';
import 'package:flutter_dmzj/models/novel/recommend_model.dart';
import 'package:flutter_dmzj/modules/novel/home/recommend/novel_recommend_controller.dart';
import 'package:flutter_dmzj/widgets/keep_alive_wrapper.dart';
import 'package:flutter_dmzj/widgets/net_image.dart';
import 'package:flutter_dmzj/widgets/page_list_view.dart';
import 'package:flutter_dmzj/widgets/refresh_until_widget.dart';
import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart';
import 'package:flutter_swiper_view/flutter_swiper_view.dart';
import 'package:get/get.dart';
class NovelRecommendView extends StatelessWidget {
final NovelRecommendController controller;
NovelRecommendView({Key? key})
: controller = Get.put(NovelRecommendController()),
super(key: key);
@override
Widget build(BuildContext context) {
return KeepAliveWrapper(
child: PageListView(
pageController: controller,
padding: AppStyle.edgeInsetsH12,
firstRefresh: true,
loadMore: false,
showPageLoadding: true,
itemBuilder: (context, i) {
var item = controller.list[i];
if (item.categoryId == 57) {
return buildBanner(item);
}
Widget? action;
if (item.categoryId == 58) {
action = buildShowMore(onTap: controller.toLatest);
}
return buildCard(
context,
child: buildTreeColumnGridView(item.data),
title: item.title.toString(),
action: action,
);
},
),
);
}
Widget buildCard(
BuildContext context, {
required Widget child,
required String title,
Widget? action,
}) {
return Padding(
padding: AppStyle.edgeInsetsB8,
child: Container(
decoration: BoxDecoration(
borderRadius: AppStyle.radius8,
),
child: Column(
children: [
Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(
child: Text(
title,
style: const TextStyle(
fontSize: 16, height: 1.0, fontWeight: FontWeight.bold),
),
),
SizedBox(
height: 48,
child: action,
),
],
),
child,
],
),
),
);
}
Widget buildShowMore({required Function() onTap}) {
return GestureDetector(
onTap: onTap,
child: const Row(
children: [
Text(
"查看更多",
style: TextStyle(fontSize: 14, color: Colors.grey),
),
Icon(Icons.chevron_right, size: 18, color: Colors.grey),
],
),
);
}
Widget buildRefresh({required Future Function() onRefresh}) {
return RefreshUntilWidget(onRefresh: onRefresh, text: "换一批");
}
Widget buildBanner(NovelRecommendModel item) {
return Padding(
padding: AppStyle.edgeInsetsB12,
child: ClipRRect(
borderRadius: AppStyle.radius4,
child: AspectRatio(
aspectRatio: 7.5 / 4,
child: Swiper(
itemWidth: 750,
itemHeight: 400,
autoplay: true,
itemCount: item.data.length,
itemBuilder: (_, i) => NetImage(
item.data[i].cover,
width: 750,
height: 400,
),
onTap: (i) {
controller.openDetail(item.data[i]);
},
pagination: SwiperCustomPagination(
builder: (BuildContext context, SwiperPluginConfig config) {
return Align(
alignment: Alignment.bottomCenter,
child: Container(
padding: const EdgeInsets.only(
left: 8,
right: 12,
top: 4,
bottom: 4,
),
//color: Colors.black12,
decoration: const BoxDecoration(
gradient: LinearGradient(
begin: Alignment.bottomCenter,
end: Alignment.topCenter,
colors: [
Colors.black38,
Colors.transparent,
],
),
),
child: Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(
child: Text(
item.data[config.activeIndex].title,
overflow: TextOverflow.ellipsis,
style: const TextStyle(
fontSize: 14, color: Colors.white),
),
),
AppStyle.hGap8,
PageIndicator(
controller: config.pageController!,
count: config.itemCount,
size: 10,
layout: PageIndicatorLayout.SCALE,
),
],
),
),
);
},
),
),
),
),
);
}
Widget buildTreeColumnGridView(List<NovelRecommendItemModel> items) {
return MasonryGridView.count(
padding: EdgeInsets.zero,
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
crossAxisCount: 3,
mainAxisSpacing: 12,
crossAxisSpacing: 12,
itemCount: items.length,
itemBuilder: (_, i) {
var item = items[i];
return InkWell(
onTap: () => controller.openDetail(item),
borderRadius: AppStyle.radius4,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
ClipRRect(
borderRadius: AppStyle.radius4,
child: AspectRatio(
aspectRatio: 27 / 36,
child: NetImage(
item.cover,
width: 270,
height: 360,
),
),
),
AppStyle.vGap8,
Text(
item.title,
maxLines: 1,
style: const TextStyle(height: 1.2),
overflow: TextOverflow.ellipsis,
),
Text(
item.subTitle ?? item.status ?? '',
maxLines: 1,
style: const TextStyle(
height: 1.2,
fontSize: 12,
color: Colors.grey,
overflow: TextOverflow.ellipsis,
),
),
AppStyle.vGap8,
],
),
);
},
);
}
}