mirror of
https://github.com/simon-ding/polaris.git
synced 2026-06-10 03:57:39 +08:00
feat: better support for small screen
This commit is contained in:
@@ -32,6 +32,7 @@ class _TvDetailsPageState extends ConsumerState<TvDetailsPage> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
final screenWidth = MediaQuery.of(context).size.width;
|
||||||
var seriesDetails = ref.watch(mediaDetailsProvider(widget.seriesId));
|
var seriesDetails = ref.watch(mediaDetailsProvider(widget.seriesId));
|
||||||
return seriesDetails.when(
|
return seriesDetails.when(
|
||||||
data: (details) {
|
data: (details) {
|
||||||
@@ -118,6 +119,41 @@ class _TvDetailsPageState extends ConsumerState<TvDetailsPage> {
|
|||||||
}
|
}
|
||||||
List<ExpansionTile> list = List.empty(growable: true);
|
List<ExpansionTile> list = List.empty(growable: true);
|
||||||
for (final k in m.keys.toList().reversed) {
|
for (final k in m.keys.toList().reversed) {
|
||||||
|
final seasonEpisodes = DataTable(columns: [
|
||||||
|
const DataColumn(label: Text("#")),
|
||||||
|
const DataColumn(
|
||||||
|
label: Text("标题"),
|
||||||
|
),
|
||||||
|
const DataColumn(label: Text("播出时间")),
|
||||||
|
const DataColumn(label: Text("状态")),
|
||||||
|
DataColumn(
|
||||||
|
label: Row(
|
||||||
|
children: [
|
||||||
|
LoadingIconButton(
|
||||||
|
tooltip: "搜索下载全部剧集",
|
||||||
|
onPressed: () async {
|
||||||
|
await ref
|
||||||
|
.read(
|
||||||
|
mediaDetailsProvider(widget.seriesId).notifier)
|
||||||
|
.searchAndDownload(widget.seriesId, k, 0)
|
||||||
|
.then((v) => showSnakeBar("开始下载: $v"));
|
||||||
|
//showLoadingWithFuture(f);
|
||||||
|
},
|
||||||
|
icon: Icons.download),
|
||||||
|
const SizedBox(
|
||||||
|
width: 10,
|
||||||
|
),
|
||||||
|
Tooltip(
|
||||||
|
message: "查看可用资源",
|
||||||
|
child: IconButton(
|
||||||
|
onPressed: () =>
|
||||||
|
showAvailableTorrents(widget.seriesId, k, 0),
|
||||||
|
icon: const Icon(Icons.manage_search)),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
))
|
||||||
|
], rows: m[k]!);
|
||||||
|
|
||||||
var seasonList = ExpansionTile(
|
var seasonList = ExpansionTile(
|
||||||
tilePadding: const EdgeInsets.fromLTRB(10, 0, 10, 0),
|
tilePadding: const EdgeInsets.fromLTRB(10, 0, 10, 0),
|
||||||
//childrenPadding: const EdgeInsets.fromLTRB(50, 0, 50, 0),
|
//childrenPadding: const EdgeInsets.fromLTRB(50, 0, 50, 0),
|
||||||
@@ -125,40 +161,12 @@ class _TvDetailsPageState extends ConsumerState<TvDetailsPage> {
|
|||||||
title: k == 0 ? const Text("特别篇") : Text("第 $k 季"),
|
title: k == 0 ? const Text("特别篇") : Text("第 $k 季"),
|
||||||
expandedCrossAxisAlignment: CrossAxisAlignment.stretch,
|
expandedCrossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
children: [
|
children: [
|
||||||
DataTable(columns: [
|
screenWidth < 600
|
||||||
const DataColumn(label: Text("#")),
|
? SingleChildScrollView(
|
||||||
const DataColumn(
|
scrollDirection: Axis.horizontal,
|
||||||
label: Text("标题"),
|
child: seasonEpisodes,
|
||||||
),
|
|
||||||
const DataColumn(label: Text("播出时间")),
|
|
||||||
const DataColumn(label: Text("状态")),
|
|
||||||
DataColumn(
|
|
||||||
label: Row(
|
|
||||||
children: [
|
|
||||||
LoadingIconButton(
|
|
||||||
tooltip: "搜索下载全部剧集",
|
|
||||||
onPressed: () async {
|
|
||||||
await ref
|
|
||||||
.read(mediaDetailsProvider(widget.seriesId)
|
|
||||||
.notifier)
|
|
||||||
.searchAndDownload(widget.seriesId, k, 0)
|
|
||||||
.then((v) => showSnakeBar("开始下载: $v"));
|
|
||||||
//showLoadingWithFuture(f);
|
|
||||||
},
|
|
||||||
icon: Icons.download),
|
|
||||||
const SizedBox(
|
|
||||||
width: 10,
|
|
||||||
),
|
|
||||||
Tooltip(
|
|
||||||
message: "查看可用资源",
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () =>
|
|
||||||
showAvailableTorrents(widget.seriesId, k, 0),
|
|
||||||
icon: const Icon(Icons.manage_search)),
|
|
||||||
)
|
)
|
||||||
],
|
: seasonEpisodes
|
||||||
))
|
|
||||||
], rows: m[k]!),
|
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
list.add(seasonList);
|
list.add(seasonList);
|
||||||
|
|||||||
@@ -52,10 +52,13 @@ class WelcomePage extends ConsumerWidget {
|
|||||||
|
|
||||||
class MediaCard extends StatelessWidget {
|
class MediaCard extends StatelessWidget {
|
||||||
final MediaDetail item;
|
final MediaDetail item;
|
||||||
|
static const double smallWidth = 110;
|
||||||
|
static const double largeWidth = 140;
|
||||||
|
|
||||||
const MediaCard({super.key, required this.item});
|
const MediaCard({super.key, required this.item});
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
|
||||||
return Card(
|
return Card(
|
||||||
shape: ContinuousRectangleBorder(
|
shape: ContinuousRectangleBorder(
|
||||||
borderRadius: BorderRadius.circular(10.0),
|
borderRadius: BorderRadius.circular(10.0),
|
||||||
@@ -75,8 +78,8 @@ class MediaCard extends StatelessWidget {
|
|||||||
child: Column(
|
child: Column(
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
SizedBox(
|
SizedBox(
|
||||||
width: 140,
|
width: cardWidth(context),
|
||||||
height: 210,
|
height: cardWidth(context) / 2 * 3,
|
||||||
child: Ink.image(
|
child: Ink.image(
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
image: NetworkImage(
|
image: NetworkImage(
|
||||||
@@ -84,7 +87,7 @@ class MediaCard extends StatelessWidget {
|
|||||||
)),
|
)),
|
||||||
),
|
),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
width: 140,
|
width: cardWidth(context),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
LinearProgressIndicator(
|
LinearProgressIndicator(
|
||||||
@@ -107,4 +110,12 @@ class MediaCard extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
double cardWidth(BuildContext context) {
|
||||||
|
final screenWidth = MediaQuery.of(context).size.width;
|
||||||
|
if (screenWidth < 600) {
|
||||||
|
return smallWidth;
|
||||||
|
}
|
||||||
|
return largeWidth;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import 'dart:math';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:ui/providers/APIs.dart';
|
import 'package:ui/providers/APIs.dart';
|
||||||
|
import 'dart:io' show Platform;
|
||||||
|
|
||||||
class Utils {
|
class Utils {
|
||||||
static Future<void> showAlertDialog(BuildContext context, String msg) async {
|
static Future<void> showAlertDialog(BuildContext context, String msg) async {
|
||||||
@@ -31,18 +32,17 @@ class Utils {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
showSnakeBar(String msg) {
|
showSnakeBar(String msg) {
|
||||||
final context = APIs.navigatorKey.currentContext;
|
final context = APIs.navigatorKey.currentContext;
|
||||||
if (context != null) {
|
if (context != null) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
||||||
content: Text(msg),
|
content: Text(msg),
|
||||||
showCloseIcon: true,
|
showCloseIcon: true,
|
||||||
));
|
));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
extension FileFormatter on num {
|
extension FileFormatter on num {
|
||||||
String readableFileSize({bool base1024 = false}) {
|
String readableFileSize({bool base1024 = false}) {
|
||||||
@@ -53,3 +53,8 @@ extension FileFormatter on num {
|
|||||||
return "${NumberFormat("#,##0.#").format(this / pow(base, digitGroups))} ${units[digitGroups]}";
|
return "${NumberFormat("#,##0.#").format(this / pow(base, digitGroups))} ${units[digitGroups]}";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool isDesktop() {
|
||||||
|
return Platform.isLinux || Platform.isWindows || Platform.isMacOS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user