This commit is contained in:
2026-01-24 19:22:03 +01:00
parent 5c948d2010
commit 11140a748a
4 changed files with 47 additions and 40 deletions

View File

@@ -17,15 +17,16 @@ class ApiClient {
Future<void> login(String username, String password) async { Future<void> login(String username, String password) async {
log.info("Logging in..."); log.info("Logging in...");
final url = '$baseUrl/login'; final url = '$baseUrl/auth/login';
final response = await http.post( final response = await http.post(
Uri.parse(url), Uri.parse(url),
headers: {'Content-Type': 'application/json'}, headers: {'Content-Type': 'application/json'},
body: jsonEncode({'email': username, 'password': password}), body: jsonEncode({'username': username, 'password': password}),
); );
if (response.statusCode == 200) { if (response.statusCode == 200) {
token = jsonDecode(response.body); final responseData = jsonDecode(response.body);
token = responseData['token'];
log.info('Login successful'); log.info('Login successful');
} else { } else {
throw Exception( throw Exception(
@@ -55,6 +56,9 @@ class ApiClient {
if (!throwExceptionIfStatusCodeNot200 || response.statusCode == 200) { if (!throwExceptionIfStatusCodeNot200 || response.statusCode == 200) {
return response; return response;
} else { } else {
log.warning(
"Failed get request to '$url'! StatusCode: ${response.statusCode}\nResponseBody: ${response.body}",
);
throw Exception( throw Exception(
'GET request failed: ${response.statusCode} ${response.body}', 'GET request failed: ${response.statusCode} ${response.body}',
); );
@@ -120,19 +124,19 @@ class ApiClient {
Future<List<Sheet>> fetchSheets() async { Future<List<Sheet>> fetchSheets() async {
final response = await get( final response = await get(
"/list/sheets", "/api/sheets/list",
throwExceptionIfStatusCodeNot200: true, throwExceptionIfStatusCodeNot200: true,
); );
final data = jsonDecode(response.body); final data = jsonDecode(response.body);
return (data as List<dynamic>) return (data['sheets'] as List<dynamic>)
.map((sheet) => Sheet.fromJson(sheet as Map<String, dynamic>)) .map((sheet) => Sheet.fromJson(sheet as Map<String, dynamic>))
.toList(); .toList();
} }
Future<Uint8List> fetchPdfFileData(String sheetUuid) async { Future<Uint8List> fetchPdfFileData(String sheetUuid) async {
final response = await get( final response = await get(
'/sheet/pdf/$sheetUuid', '/api/sheets/get/$sheetUuid',
isBinary: true, isBinary: true,
throwExceptionIfStatusCodeNot200: true, throwExceptionIfStatusCodeNot200: true,
); );

View File

@@ -58,37 +58,36 @@ class _MyHomePageState extends State<MyHomePage> with FullScreenListener {
Future<List<Sheet>> acquireSheets() async { Future<List<Sheet>> acquireSheets() async {
final url = await _storageHelper.readSecure(SecureStorageKey.url); final url = await _storageHelper.readSecure(SecureStorageKey.url);
final jwt = await _storageHelper.readSecure(SecureStorageKey.jwt); final jwt = await _storageHelper.readSecure(SecureStorageKey.jwt);
apiClient = ApiClient(baseUrl: "${url!}/api", token: jwt); apiClient = ApiClient(baseUrl: url!, token: jwt);
// TODO: check if really logged in // TODO: check if really logged in
final sheets = await apiClient!.fetchSheets(); final sheets = await apiClient!.fetchSheets();
log.info("${sheets.length} sheets fetched"); log.info("${sheets.length} sheets fetched");
final sheetsSorted = await sortSheetsByAccessTime(sheets); final sheetsSorted = await sortSheetsByRecency(sheets);
log.info("${sheetsSorted.length} sheets sorted"); log.info("${sheetsSorted.length} sheets sorted");
final changeQueue = await _storageHelper.readChangeQueue(); // TODO: make work
changeQueue.applyToSheets(sheetsSorted); // final changeQueue = await _storageHelper.readChangeQueue();
log.info("${changeQueue.length()} changes applied"); // changeQueue.applyToSheets(sheetsSorted);
// log.info("${changeQueue.length()} changes applied");
return sheetsSorted; return sheetsSorted;
} }
Future<List<Sheet>> sortSheetsByAccessTime(List<Sheet> sheets) async { Future<List<Sheet>> sortSheetsByRecency(List<Sheet> sheets) async {
final accessTimes = await _storageHelper.readSheetAccessTimes(); final accessTimes = await _storageHelper.readSheetAccessTimes();
sheets.sort((a, b) { sheets.sort((a, b) {
final dateA = accessTimes[a.uuid]; var dateA = accessTimes[a.uuid];
final dateB = accessTimes[b.uuid]; var dateB = accessTimes[b.uuid];
if (dateB == null) { if (dateA == null || a.updatedAt.isAfter(dateA)) {
// b has no date, sort below a dateA = a.updatedAt;
return -1;
} else if (dateA == null) {
// a has no date, sort below b
return 1;
} else {
// compare both and sort by date
return dateB.compareTo(dateA);
} }
if (dateB == null || b.updatedAt.isAfter(dateB)) {
dateB = b.updatedAt;
}
return dateB.compareTo(dateA);
}); });
return sheets; return sheets;
@@ -109,7 +108,7 @@ class _MyHomePageState extends State<MyHomePage> with FullScreenListener {
if (newState) { if (newState) {
(await sheets).shuffle(); (await sheets).shuffle();
} else { } else {
sheets = sortSheetsByAccessTime(await sheets); sheets = sortSheetsByRecency(await sheets);
} }
setState(() { setState(() {

View File

@@ -18,7 +18,7 @@ class _LoginPageState extends State<LoginPage> {
final TextEditingController _urlController = TextEditingController( final TextEditingController _urlController = TextEditingController(
text: "https://sheetable.julian-mutter.de", text: "https://sheetable.julian-mutter.de",
); );
final TextEditingController _emailController = TextEditingController(); final TextEditingController _usernameController = TextEditingController();
final TextEditingController _passwordController = TextEditingController(); final TextEditingController _passwordController = TextEditingController();
final _formKey = GlobalKey<FormState>(); final _formKey = GlobalKey<FormState>();
@@ -40,12 +40,12 @@ class _LoginPageState extends State<LoginPage> {
return; return;
} }
final url = await _storageHelper.readSecure(SecureStorageKey.url); final url = await _storageHelper.readSecure(SecureStorageKey.url);
final email = await _storageHelper.readSecure(SecureStorageKey.email); final username = await _storageHelper.readSecure(SecureStorageKey.email);
if (url != null) { if (url != null) {
_urlController.text = url; _urlController.text = url;
} }
if (email != null) { if (username != null) {
_emailController.text = email; _usernameController.text = username;
} }
} }
@@ -58,17 +58,18 @@ class _LoginPageState extends State<LoginPage> {
} }
} }
Future<void> _login(String serverUrl, String email, String password) async { Future<void> _login(
String serverUrl, String username, String password) async {
setState(() { setState(() {
_error = null; _error = null;
}); });
final apiClient = ApiClient(baseUrl: "$serverUrl/api"); final apiClient = ApiClient(baseUrl: serverUrl);
try { try {
await apiClient.login(email, password); await apiClient.login(username, password);
await _storageHelper.writeSecure(SecureStorageKey.url, serverUrl); await _storageHelper.writeSecure(SecureStorageKey.url, serverUrl);
await _storageHelper.writeSecure(SecureStorageKey.jwt, apiClient.token!); await _storageHelper.writeSecure(SecureStorageKey.jwt, apiClient.token!);
await _storageHelper.writeSecure(SecureStorageKey.email, email); await _storageHelper.writeSecure(SecureStorageKey.email, username);
await _navigateToMainPage(); await _navigateToMainPage();
} catch (e) { } catch (e) {
setState(() { setState(() {
@@ -98,7 +99,7 @@ class _LoginPageState extends State<LoginPage> {
if (_formKey.currentState!.validate()) { if (_formKey.currentState!.validate()) {
await _login( await _login(
_urlController.text, _urlController.text,
_emailController.text, _usernameController.text,
_passwordController.text, _passwordController.text,
); );
} }
@@ -123,9 +124,9 @@ class _LoginPageState extends State<LoginPage> {
textInputAction: TextInputAction.next, textInputAction: TextInputAction.next,
), ),
TextFormField( TextFormField(
controller: _emailController, controller: _usernameController,
validator: validateNotEmpty, validator: validateNotEmpty,
decoration: InputDecoration(labelText: 'Email'), decoration: InputDecoration(labelText: 'Username'),
textInputAction: TextInputAction.next, textInputAction: TextInputAction.next,
), ),
TextFormField( TextFormField(

View File

@@ -1,6 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:ui'; import 'dart:ui';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:sheetless/edit_bottom_sheet.dart'; import 'package:sheetless/edit_bottom_sheet.dart';
import 'package:sheetless/storage_helper.dart'; import 'package:sheetless/storage_helper.dart';
@@ -11,21 +10,25 @@ class Sheet {
String name; String name;
String composerUuid; String composerUuid;
String composerName; String composerName;
DateTime updatedAt;
Sheet({ Sheet({
required this.uuid, required this.uuid,
required this.name, required this.name,
required this.composerUuid, required this.composerUuid,
required this.composerName, required this.composerName,
required this.updatedAt,
}); });
// Factory constructor for creating a Sheet from JSON // Factory constructor for creating a Sheet from JSON
factory Sheet.fromJson(Map<String, dynamic> json) { factory Sheet.fromJson(Map<String, dynamic> json) {
final composer = json['composer'] as Map<String, dynamic>?;
return Sheet( return Sheet(
uuid: json['uuid'], uuid: json['uuid'].toString(),
name: json['sheet_name'], name: json['title'],
composerUuid: json['composer_uuid'], composerUuid: json['composer_uuid']?.toString() ?? '',
composerName: json['composer_name'], composerName: composer?['name'] ?? 'Unknown',
updatedAt: DateTime.parse(json['updated_at']),
); );
} }
} }