mirror of
https://github.com/simon-ding/polaris.git
synced 2026-06-09 11:39:46 +08:00
feat: activity page
This commit is contained in:
@@ -28,7 +28,7 @@ type History struct {
|
|||||||
// TargetDir holds the value of the "target_dir" field.
|
// TargetDir holds the value of the "target_dir" field.
|
||||||
TargetDir string `json:"target_dir,omitempty"`
|
TargetDir string `json:"target_dir,omitempty"`
|
||||||
// Completed holds the value of the "completed" field.
|
// Completed holds the value of the "completed" field.
|
||||||
Completed bool `json:"completed,omitempty"`
|
Completed bool `json:"completed"`
|
||||||
// Saved holds the value of the "saved" field.
|
// Saved holds the value of the "saved" field.
|
||||||
Saved string `json:"saved,omitempty"`
|
Saved string `json:"saved,omitempty"`
|
||||||
selectValues sql.SelectValues
|
selectValues sql.SelectValues
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ func (History) Fields() []ent.Field {
|
|||||||
field.String("source_title"),
|
field.String("source_title"),
|
||||||
field.Time("date"),
|
field.Time("date"),
|
||||||
field.String("target_dir"),
|
field.String("target_dir"),
|
||||||
field.Bool("completed").Default(false),
|
field.Bool("completed").Default(false).StructTag("json:\"completed\""),
|
||||||
field.String("saved").Optional(),
|
field.String("saved").Optional(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"polaris/ent"
|
||||||
"polaris/log"
|
"polaris/log"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
@@ -8,10 +9,27 @@ import (
|
|||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type Activity struct {
|
||||||
|
*ent.History
|
||||||
|
InBackgroud bool `json:"in_backgroud"`
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Server) GetAllActivities(c *gin.Context) (interface{}, error) {
|
func (s *Server) GetAllActivities(c *gin.Context) (interface{}, error) {
|
||||||
his := s.db.GetHistories()
|
his := s.db.GetHistories()
|
||||||
|
var activities = make([]Activity, 0, len(his))
|
||||||
|
for _, h := range his {
|
||||||
|
a := Activity{
|
||||||
|
History: h,
|
||||||
|
}
|
||||||
|
for id, task := range s.tasks {
|
||||||
|
if h.ID == id && task.Processing {
|
||||||
|
a.InBackgroud = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
activities = append(activities, a)
|
||||||
|
}
|
||||||
|
|
||||||
return his, nil
|
return activities, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) RemoveActivity(c *gin.Context) (interface{}, error) {
|
func (s *Server) RemoveActivity(c *gin.Context) (interface{}, error) {
|
||||||
|
|||||||
@@ -131,7 +131,7 @@ func (s *Server) searchAndDownload(seriesId, seasonNum, episodeNum int) (*string
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "save record")
|
return nil, errors.Wrap(err, "save record")
|
||||||
}
|
}
|
||||||
s.tasks[history.ID] = torrent
|
s.tasks[history.ID] = &Task{Torrent: torrent}
|
||||||
|
|
||||||
log.Infof("success add %s to download task", r1.Name)
|
log.Infof("success add %s to download task", r1.Name)
|
||||||
return &r1.Name, nil
|
return &r1.Name, nil
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"polaris/db"
|
"polaris/db"
|
||||||
"polaris/log"
|
"polaris/log"
|
||||||
|
"polaris/pkg"
|
||||||
"polaris/pkg/storage"
|
"polaris/pkg/storage"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
@@ -28,9 +29,14 @@ func (s *Server) checkTasks() {
|
|||||||
log.Infof("task no longer exists: %v", id)
|
log.Infof("task no longer exists: %v", id)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if t.Processing {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
log.Infof("task (%s) percentage done: %d%%", t.Name(), t.Progress())
|
log.Infof("task (%s) percentage done: %d%%", t.Name(), t.Progress())
|
||||||
if t.Progress() == 100 {
|
if t.Progress() == 100 {
|
||||||
log.Infof("task is done: %v", t.Name())
|
log.Infof("task is done: %v", t.Name())
|
||||||
|
t.Processing = true
|
||||||
go func() {
|
go func() {
|
||||||
if err := s.moveCompletedTask(id); err != nil {
|
if err := s.moveCompletedTask(id); err != nil {
|
||||||
log.Infof("post tasks for id %v fail: %v", id, err)
|
log.Infof("post tasks for id %v fail: %v", id, err)
|
||||||
@@ -43,9 +49,6 @@ func (s *Server) checkTasks() {
|
|||||||
func (s *Server) moveCompletedTask(id int) error {
|
func (s *Server) moveCompletedTask(id int) error {
|
||||||
torrent := s.tasks[id]
|
torrent := s.tasks[id]
|
||||||
r := s.db.GetHistory(id)
|
r := s.db.GetHistory(id)
|
||||||
s.db.SetHistoryComplete(r.ID)
|
|
||||||
|
|
||||||
delete(s.tasks, r.ID)
|
|
||||||
|
|
||||||
series := s.db.GetSeriesDetails(r.SeriesID)
|
series := s.db.GetSeriesDetails(r.SeriesID)
|
||||||
st := s.db.GetStorage(series.StorageID)
|
st := s.db.GetStorage(series.StorageID)
|
||||||
@@ -68,5 +71,12 @@ func (s *Server) moveCompletedTask(id int) error {
|
|||||||
}
|
}
|
||||||
log.Infof("move downloaded files to target dir success, file: %v, target dir: %v", torrent.Name(), r.TargetDir)
|
log.Infof("move downloaded files to target dir success, file: %v, target dir: %v", torrent.Name(), r.TargetDir)
|
||||||
torrent.Remove()
|
torrent.Remove()
|
||||||
|
delete(s.tasks, r.ID)
|
||||||
|
s.db.SetHistoryComplete(r.ID)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Task struct {
|
||||||
|
Processing bool
|
||||||
|
pkg.Torrent
|
||||||
|
}
|
||||||
@@ -3,7 +3,6 @@ package server
|
|||||||
import (
|
import (
|
||||||
"polaris/db"
|
"polaris/db"
|
||||||
"polaris/log"
|
"polaris/log"
|
||||||
"polaris/pkg"
|
|
||||||
"polaris/pkg/tmdb"
|
"polaris/pkg/tmdb"
|
||||||
"polaris/pkg/transmission"
|
"polaris/pkg/transmission"
|
||||||
"polaris/ui"
|
"polaris/ui"
|
||||||
@@ -21,7 +20,7 @@ func NewServer(db *db.Client) *Server {
|
|||||||
r: r,
|
r: r,
|
||||||
db: db,
|
db: db,
|
||||||
cron: cron.New(),
|
cron: cron.New(),
|
||||||
tasks: make(map[int]pkg.Torrent),
|
tasks: make(map[int]*Task),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -30,7 +29,7 @@ type Server struct {
|
|||||||
db *db.Client
|
db *db.Client
|
||||||
cron *cron.Cron
|
cron *cron.Cron
|
||||||
language string
|
language string
|
||||||
tasks map[int]pkg.Torrent
|
tasks map[int]*Task
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) Serve() error {
|
func (s *Server) Serve() error {
|
||||||
@@ -118,6 +117,6 @@ func (s *Server) reloadTasks() {
|
|||||||
log.Errorf("relaod task %s failed: %v", t.SourceTitle, err)
|
log.Errorf("relaod task %s failed: %v", t.SourceTitle, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
s.tasks[t.ID] = torrent
|
s.tasks[t.ID] = &Task{Torrent: torrent}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,51 @@
|
|||||||
|
import 'dart:js_interop_unsafe';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
|
import 'package:ui/providers/activity.dart';
|
||||||
|
|
||||||
class ActivityPage extends ConsumerWidget {
|
class ActivityPage extends ConsumerWidget {
|
||||||
|
static const route = "/activities";
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
// TODO: implement build
|
var activitiesWatcher = ref.watch(activitiesDataProvider);
|
||||||
throw UnimplementedError();
|
|
||||||
|
return activitiesWatcher.when(
|
||||||
|
data: (activities) {
|
||||||
|
return SingleChildScrollView(
|
||||||
|
child: DataTable(
|
||||||
|
columns: const [
|
||||||
|
DataColumn(label: Text("id"), numeric: true),
|
||||||
|
DataColumn(label: Text("名称")),
|
||||||
|
// DataColumn(label: Text("目标路径")),
|
||||||
|
DataColumn(label: Text("是否完成")),
|
||||||
|
DataColumn(label: Text("后台操作")),
|
||||||
|
DataColumn(label: Text("操作"))
|
||||||
|
],
|
||||||
|
rows: List<DataRow>.generate(activities.length, (i) {
|
||||||
|
var activity = activities[i];
|
||||||
|
|
||||||
|
return DataRow(cells: [
|
||||||
|
DataCell(Text("${activity.id}")),
|
||||||
|
DataCell(Text("${activity.sourceTitle}")),
|
||||||
|
//DataCell(Text("${activity.targetDir}")),
|
||||||
|
DataCell(Text("${activity.completed}")),
|
||||||
|
DataCell(Text("${activity.inBackgroud}")),
|
||||||
|
DataCell(IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
ref
|
||||||
|
.read(activitiesDataProvider.notifier)
|
||||||
|
.deleteActivity(activity.id!);
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.delete)))
|
||||||
|
]);
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
error: (err, trace) => Text("$err"),
|
||||||
|
loading: () => const Center(
|
||||||
|
child: SizedBox(
|
||||||
|
width: 30, height: 30, child: CircularProgressIndicator())));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
import 'package:flutter/gestures.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:go_router/go_router.dart';
|
import 'package:go_router/go_router.dart';
|
||||||
|
import 'package:ui/activity.dart';
|
||||||
import 'package:ui/login_page.dart';
|
import 'package:ui/login_page.dart';
|
||||||
import 'package:ui/navdrawer.dart';
|
import 'package:ui/navdrawer.dart';
|
||||||
import 'package:ui/providers/APIs.dart';
|
import 'package:ui/providers/APIs.dart';
|
||||||
@@ -34,12 +36,19 @@ class MyApp extends StatelessWidget {
|
|||||||
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
|
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
|
||||||
// Here we take the value from the MyHomePage object that was created by
|
// Here we take the value from the MyHomePage object that was created by
|
||||||
// the App.build method, and use it to set our appbar title.
|
// the App.build method, and use it to set our appbar title.
|
||||||
title: const Text("Polaris追剧"),
|
title: Row(
|
||||||
|
children: [
|
||||||
|
const Text("Polaris 追剧"),
|
||||||
|
const SizedBox(
|
||||||
|
width: 100,
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
tooltip: "搜索剧集",
|
||||||
|
onPressed: () => context.go(SearchPage.route),
|
||||||
|
icon: const Icon(Icons.search)),
|
||||||
|
],
|
||||||
|
),
|
||||||
actions: [
|
actions: [
|
||||||
IconButton(
|
|
||||||
tooltip: "搜索剧集",
|
|
||||||
onPressed: () => context.go(SearchPage.route),
|
|
||||||
icon: const Icon(Icons.search)),
|
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () => context.go(SystemSettingsPage.route),
|
onPressed: () => context.go(SystemSettingsPage.route),
|
||||||
icon: const Icon(Icons.settings))
|
icon: const Icon(Icons.settings))
|
||||||
@@ -77,6 +86,10 @@ class MyApp extends StatelessWidget {
|
|||||||
builder: (context, state) =>
|
builder: (context, state) =>
|
||||||
TvDetailsPage(seriesId: state.pathParameters['id']!),
|
TvDetailsPage(seriesId: state.pathParameters['id']!),
|
||||||
),
|
),
|
||||||
|
GoRoute(
|
||||||
|
path: ActivityPage.route,
|
||||||
|
builder: (context, state) => ActivityPage(),
|
||||||
|
)
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -87,7 +100,7 @@ class MyApp extends StatelessWidget {
|
|||||||
_shellRoute,
|
_shellRoute,
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: LoginScreen.route,
|
path: LoginScreen.route,
|
||||||
builder: (context, state) =>const LoginScreen(),
|
builder: (context, state) => const LoginScreen(),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:go_router/go_router.dart';
|
import 'package:go_router/go_router.dart';
|
||||||
|
import 'package:ui/activity.dart';
|
||||||
import 'package:ui/search.dart';
|
import 'package:ui/search.dart';
|
||||||
import 'package:ui/system_settings.dart';
|
import 'package:ui/system_settings.dart';
|
||||||
import 'package:ui/weclome.dart';
|
import 'package:ui/weclome.dart';
|
||||||
@@ -34,7 +35,7 @@ class _NavDrawerState extends State<NavDrawer> {
|
|||||||
if (value == 0) {
|
if (value == 0) {
|
||||||
context.go(WelcomePage.route);
|
context.go(WelcomePage.route);
|
||||||
} else if (value == 1) {
|
} else if (value == 1) {
|
||||||
context.go(SearchPage.route);
|
context.go(ActivityPage.route);
|
||||||
} else if (value == 2) {
|
} else if (value == 2) {
|
||||||
context.go(SystemSettingsPage.route);
|
context.go(SystemSettingsPage.route);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ class APIs {
|
|||||||
static final storageUrl = "$_baseUrl/api/v1/storage/";
|
static final storageUrl = "$_baseUrl/api/v1/storage/";
|
||||||
static final loginUrl = "$_baseUrl/api/login";
|
static final loginUrl = "$_baseUrl/api/login";
|
||||||
static final loginSettingUrl = "$_baseUrl/api/v1/setting/auth";
|
static final loginSettingUrl = "$_baseUrl/api/v1/setting/auth";
|
||||||
|
static final activityUrl = "$_baseUrl/api/v1/activity/";
|
||||||
|
|
||||||
static const tmdbImgBaseUrl = "https://image.tmdb.org/t/p/w500/";
|
static const tmdbImgBaseUrl = "https://image.tmdb.org/t/p/w500/";
|
||||||
|
|
||||||
|
|||||||
74
ui/lib/providers/activity.dart
Normal file
74
ui/lib/providers/activity.dart
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
|
||||||
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
|
import 'package:ui/providers/APIs.dart';
|
||||||
|
import 'package:ui/providers/server_response.dart';
|
||||||
|
|
||||||
|
var activitiesDataProvider = AsyncNotifierProvider.autoDispose<ActivityData, List<Activity>>(
|
||||||
|
ActivityData.new);
|
||||||
|
|
||||||
|
class ActivityData extends AutoDisposeAsyncNotifier<List<Activity>> {
|
||||||
|
@override
|
||||||
|
FutureOr<List<Activity>> build() async {
|
||||||
|
final dio = await APIs.getDio();
|
||||||
|
var resp = await dio.get(APIs.activityUrl);
|
||||||
|
final sp = ServerResponse.fromJson(resp.data);
|
||||||
|
if (sp.code != 0) {
|
||||||
|
throw sp.message;
|
||||||
|
}
|
||||||
|
List<Activity> activities = List.empty(growable: true);
|
||||||
|
for (final a in sp.data as List) {
|
||||||
|
activities.add(Activity.fromJson(a));
|
||||||
|
}
|
||||||
|
return activities;
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> deleteActivity(int id) async {
|
||||||
|
final dio = await APIs.getDio();
|
||||||
|
var resp = await dio.delete("${APIs.activityUrl}$id");
|
||||||
|
final sp = ServerResponse.fromJson(resp.data);
|
||||||
|
if (sp.code != 0) {
|
||||||
|
throw sp.message;
|
||||||
|
}
|
||||||
|
ref.invalidateSelf();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Activity {
|
||||||
|
Activity({
|
||||||
|
required this.id,
|
||||||
|
required this.seriesId,
|
||||||
|
required this.episodeId,
|
||||||
|
required this.sourceTitle,
|
||||||
|
required this.date,
|
||||||
|
required this.targetDir,
|
||||||
|
required this.completed,
|
||||||
|
required this.saved,
|
||||||
|
required this.inBackgroud,
|
||||||
|
});
|
||||||
|
|
||||||
|
final int? id;
|
||||||
|
final int? seriesId;
|
||||||
|
final int? episodeId;
|
||||||
|
final String? sourceTitle;
|
||||||
|
final DateTime? date;
|
||||||
|
final String? targetDir;
|
||||||
|
final bool? completed;
|
||||||
|
final String? saved;
|
||||||
|
final bool? inBackgroud;
|
||||||
|
|
||||||
|
factory Activity.fromJson(Map<String, dynamic> json){
|
||||||
|
return Activity(
|
||||||
|
id: json["id"],
|
||||||
|
seriesId: json["series_id"],
|
||||||
|
episodeId: json["episode_id"],
|
||||||
|
sourceTitle: json["source_title"],
|
||||||
|
date: DateTime.tryParse(json["date"] ?? ""),
|
||||||
|
targetDir: json["target_dir"],
|
||||||
|
completed: json["completed"],
|
||||||
|
saved: json["saved"],
|
||||||
|
inBackgroud: json["in_backgroud"],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user