First designs for ui

This commit is contained in:
verboomp
2026-01-27 09:57:53 +01:00
parent f48bfe2107
commit 3d456128b1
16 changed files with 487 additions and 155 deletions

View File

@@ -2,22 +2,15 @@ import 'package:fotodocumentation/controller/base_controller.dart';
import 'package:fotodocumentation/dto/customer_dto.dart';
abstract interface class PictureController {
Future<PictureDto?> get({required int id});
Future<bool> delete(PictureDto dto);
}
class PictureControllerImpl extends BaseController implements PictureController {
final String path = "picture";
@override
Future<PictureDto?> get({required int id}) {
String uriStr = '${uriUtils.getBaseUrl()}$path/$id';
return runGetWithAuth(uriStr, (json) => PictureDto.fromJson(json));
}
@override
Future<bool> delete(PictureDto dto) {
// TODO: implement delete
throw UnimplementedError();
String uriStr = '${uriUtils.getBaseUrl()}$path/${dto.id}';
return runDeleteWithAuth(uriStr);
}
}

View File

@@ -6,7 +6,7 @@ class CustomerListDto {
final String customerNumber;
final DateTime? lastUpdateDate;
CustomerListDto({required this.id, required this.name, required this.customerNumber, required this.lastUpdateDate});
CustomerListDto({required this.id, required this.name, required this.customerNumber, this.lastUpdateDate});
/// Create from JSON response
factory CustomerListDto.fromJson(Map<String, dynamic> json) {
@@ -45,8 +45,10 @@ class PictureDto {
final String image;
final DateTime pictureDate;
final String? username;
final CustomerListDto customerListDto;
PictureDto({required this.id, required this.comment, required this.category, required this.image, required this.pictureDate, required this.username});
PictureDto(
{required this.id, required this.comment, required this.category, required this.image, required this.pictureDate, required this.username, required this.customerListDto});
/// Create from JSON response
factory PictureDto.fromJson(Map<String, dynamic> json) {
@@ -57,6 +59,7 @@ PictureDto({required this.id, required this.comment, required this.category, req
image: json['image'] as String,
pictureDate: DateTimeUtils.toDateTime(json['pictureDate']) ?? DateTime.now(),
username: json['username'] as String?,
customerListDto: CustomerListDto.fromJson(json['customer']),
);
}
}

View File

@@ -1,6 +1,8 @@
import 'dart:convert' show base64Decode;
import 'package:flutter/material.dart';
import 'package:fotodocumentation/controller/picture_controller.dart';
import 'package:fotodocumentation/pages/customer/picture_delete_dialog.dart';
import 'package:go_router/go_router.dart';
@@ -16,6 +18,7 @@ import 'package:fotodocumentation/pages/ui_utils/component/waiting_widget.dart';
import 'package:fotodocumentation/pages/ui_utils/general_style.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/global_router.dart';
import 'package:fotodocumentation/pages/customer/picture_widget.dart';
class CustomerWidget extends StatefulWidget {
final int customerId;
@@ -27,6 +30,7 @@ class CustomerWidget extends StatefulWidget {
class _CustomerWidgetState extends State<CustomerWidget> {
CustomerController get _customerController => DiContainer.get();
PictureController get _pictureController => DiContainer.get();
GeneralStyle get _generalStyle => DiContainer.get();
late Future<CustomerDto?> _dto;
@@ -103,8 +107,10 @@ class _CustomerWidgetState extends State<CustomerWidget> {
);
}
Widget _customerWidget(CustomerDto dto) {
var dtos = dto.pictures;
Widget _customerWidget(CustomerDto customerDto) {
var pictureDtos = customerDto.pictures;
pictureDtos.sort((a, b) => b.pictureDate.compareTo(a.pictureDate));
return Card(
margin: EdgeInsets.zero,
@@ -118,9 +124,9 @@ class _CustomerWidgetState extends State<CustomerWidget> {
Expanded(
child: ListView.separated(
padding: const EdgeInsets.only(top: 8.0),
itemCount: dtos.length,
itemCount: pictureDtos.length,
itemBuilder: (BuildContext context, int index) {
return _tableDataRow(context, dtos[index]);
return _tableDataRow(context, customerDto, pictureDtos[index]);
},
separatorBuilder: (BuildContext context, int index) => Divider(color: _generalStyle.secondaryWidgetBackgroundColor),
),
@@ -173,21 +179,22 @@ class _CustomerWidgetState extends State<CustomerWidget> {
),
),
),
const SizedBox(width: 48),
],
),
);
}
Widget _tableDataRow(BuildContext context, PictureDto dto) {
Widget _tableDataRow(BuildContext context, CustomerDto customerDto, PictureDto pictureDto) {
final dataStyle = TextStyle(
fontFamily: _generalStyle.fontFamily,
fontSize: 16.0,
color: _generalStyle.secondaryTextLabelColor,
);
final dateStr = _dateFormat.format(dto.pictureDate);
final dateStr = _dateFormat.format(pictureDto.pictureDate);
return InkWell(
onTap: () => _actionSelect(context, dto),
onTap: () => _actionSelect(context, customerDto, pictureDto),
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
child: Row(
@@ -201,7 +208,7 @@ class _CustomerWidgetState extends State<CustomerWidget> {
child: ConstrainedBox(
constraints: const BoxConstraints(maxWidth: 70, maxHeight: 70),
child: Image.memory(
base64Decode(dto.image),
base64Decode(pictureDto.image),
fit: BoxFit.contain,
),
),
@@ -211,7 +218,7 @@ class _CustomerWidgetState extends State<CustomerWidget> {
flex: 3,
child: Align(
alignment: Alignment.centerLeft,
child: Text(dto.comment ?? "", style: dataStyle),
child: Text(pictureDto.comment ?? "", style: dataStyle),
),
),
Expanded(
@@ -221,14 +228,56 @@ class _CustomerWidgetState extends State<CustomerWidget> {
child: Text(dateStr, style: dataStyle),
),
),
SizedBox(
width: 48,
child: IconButton(
icon: Icon(
Icons.delete_outline,
color: _generalStyle.errorColor,
),
onPressed: () => _actionDelete(context, customerDto, pictureDto),
),
),
],
),
),
);
}
Future<void> _actionSelect(BuildContext context, PictureDto dto) async {
context.go("${GlobalRouter.pathPicture}/${dto.id}");
Future<void> _actionDelete(BuildContext context, CustomerDto customerDto, PictureDto pictureDto) async {
final confirmed = await showDialog<bool>(
context: context,
builder: (BuildContext context) {
return PictureDeleteDialog();
},
);
if (confirmed == true) {
_pictureController.delete(pictureDto);
setState(() {
_dto = _customerController.get(id: widget.customerId);
});
}
}
Future<void> _actionSelect(BuildContext context, CustomerDto customerDto, PictureDto pictureDto) async {
showDialog(
context: context,
builder: (BuildContext context) {
return Dialog(
backgroundColor: _generalStyle.pageBackgroundColor,
insetPadding: const EdgeInsets.all(24),
child: ClipRRect(
borderRadius: BorderRadius.circular(12),
child: SizedBox(
width: MediaQuery.of(context).size.width * 0.9,
height: MediaQuery.of(context).size.height * 0.9,
child: PictureWidget(customerDto: customerDto, pictureDto: pictureDto),
),
),
);
},
);
}
Widget _backButton(BuildContext context) {

View File

@@ -0,0 +1,53 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/pages/ui_utils/general_style.dart';
import 'package:fotodocumentation/utils/di_container.dart';
class PictureDeleteDialog extends StatelessWidget {
GeneralStyle get _generalStyle => DiContainer.get();
const PictureDeleteDialog({super.key});
@override
Widget build(BuildContext context) {
return AlertDialog(
title: Text(
AppLocalizations.of(context)!.deleteDialogTitle,
style: TextStyle(
fontFamily: _generalStyle.fontFamily,
fontWeight: FontWeight.bold,
color: _generalStyle.secondaryWidgetBackgroundColor,
),
),
content: Text(
AppLocalizations.of(context)!.deleteDialogText,
style: TextStyle(
fontFamily: _generalStyle.fontFamily,
color: _generalStyle.secondaryTextLabelColor,
),
),
actions: [
TextButton(
onPressed: () => Navigator.of(context).pop(false),
child: Text(
AppLocalizations.of(context)!.deleteDialogButtonCancel,
style: TextStyle(
fontFamily: _generalStyle.fontFamily,
color: _generalStyle.secondaryTextLabelColor,
),
),
),
TextButton(
onPressed: () => Navigator.of(context).pop(true),
child: Text(
AppLocalizations.of(context)!.deleteDialogButtonApprove,
style: TextStyle(
fontFamily: _generalStyle.fontFamily,
color: _generalStyle.errorColor,
),
),
),
],
);
}
}

View File

@@ -0,0 +1,54 @@
import 'dart:convert' show base64Decode;
import 'package:flutter/material.dart';
import 'package:fotodocumentation/dto/customer_dto.dart';
import 'package:fotodocumentation/pages/ui_utils/general_style.dart';
import 'package:fotodocumentation/utils/di_container.dart';
class PictureFullscreenDialog extends StatelessWidget {
GeneralStyle get _generalStyle => DiContainer.get();
final PictureDto dto;
const PictureFullscreenDialog({super.key, required this.dto});
@override
Widget build(BuildContext context) {
return Dialog(
backgroundColor: _generalStyle.pageBackgroundColor,
insetPadding: const EdgeInsets.all(24),
child: ClipRRect(
borderRadius: BorderRadius.circular(12),
child: SizedBox(
width: MediaQuery.of(context).size.width * 0.9,
height: MediaQuery.of(context).size.height * 0.9,
child: Stack(
fit: StackFit.expand,
children: [
InteractiveViewer(
panEnabled: true,
scaleEnabled: true,
minScale: 0.5,
maxScale: 5.0,
child: Center(
child: Image.memory(
base64Decode(dto.image),
fit: BoxFit.contain,
),
),
),
Positioned(
top: 16,
right: 16,
child: IconButton(
icon: Icon(Icons.close, color: _generalStyle.primaryButtonBackgroundColor, size: 32),
onPressed: () => Navigator.of(context).pop(),
),
),
],
),
),
),
);
}
}

View File

@@ -1,31 +1,25 @@
import 'dart:convert' show base64Decode;
import 'package:flutter/material.dart';
import 'package:fotodocumentation/controller/picture_controller.dart';
import 'package:intl/intl.dart';
import 'package:fotodocumentation/controller/base_controller.dart';
import 'package:fotodocumentation/dto/customer_dto.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/pages/ui_utils/component/general_error_widget.dart';
import 'package:fotodocumentation/pages/ui_utils/component/waiting_widget.dart';
import 'package:fotodocumentation/pages/customer/picture_fullscreen_dialog.dart';
import 'package:fotodocumentation/pages/ui_utils/general_style.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:intl/intl.dart';
class PictureWidget extends StatefulWidget {
final int id;
const PictureWidget({super.key, required this.id});
final CustomerDto customerDto;
final PictureDto pictureDto;
const PictureWidget({super.key, required this.customerDto, required this.pictureDto});
@override
State<PictureWidget> createState() => _PictureWidgetState();
}
class _PictureWidgetState extends State<PictureWidget> {
PictureController get _pictureController => DiContainer.get();
GeneralStyle get _generalStyle => DiContainer.get();
late Future<PictureDto?> _dto;
late PictureDto _selectedPicture;
late DateFormat _dateFormat;
final ScrollController _commentScrollController = ScrollController();
@@ -33,7 +27,7 @@ class _PictureWidgetState extends State<PictureWidget> {
void initState() {
super.initState();
_dateFormat = DateFormat('dd MMMM yyyy');
_dto = _pictureController.get(id: widget.id);
_selectedPicture = widget.pictureDto;
}
@override
@@ -44,91 +38,164 @@ class _PictureWidgetState extends State<PictureWidget> {
@override
Widget build(BuildContext context) {
return Scaffold(
body: Container(
color: _generalStyle.pageBackgroundColor,
child: Padding(
padding: const EdgeInsets.only(top: 8.0, left: 50.0, right: 50.0, bottom: 8.0),
child: _body(context),
final pictures = widget.customerDto.pictures;
final currentIndex = pictures.indexWhere((p) => p.id == _selectedPicture.id);
final hasPrevious = currentIndex > 0;
final hasNext = currentIndex < pictures.length - 1;
return Stack(
children: [
Container(
width: double.infinity,
height: double.infinity,
color: _generalStyle.pageBackgroundColor,
child: Padding(
padding: const EdgeInsets.only(top: 50.0, left: 50.0, right: 50.0, bottom: 8.0),
child: _body(context),
),
),
),
// Left navigation button
if (hasPrevious)
Positioned(
left: 0,
top: 0,
bottom: 0,
child: GestureDetector(
onTap: () => _navigateToPicture(currentIndex - 1),
child: MouseRegion(
cursor: SystemMouseCursors.click,
child: Container(
width: 50,
color: Colors.transparent,
child: Center(
child: Container(
decoration: BoxDecoration(
color: _generalStyle.primaryButtonBackgroundColor,
shape: BoxShape.circle,
),
child: Padding(
padding: const EdgeInsets.all(8.0),
child: Icon(
Icons.chevron_left,
color: _generalStyle.primaryButtonTextColor,
size: 32,
),
),
),
),
),
),
),
),
// Right navigation button
if (hasNext)
Positioned(
right: 0,
top: 0,
bottom: 0,
child: GestureDetector(
onTap: () => _navigateToPicture(currentIndex + 1),
child: MouseRegion(
cursor: SystemMouseCursors.click,
child: Container(
width: 50,
color: Colors.transparent,
child: Center(
child: Container(
decoration: BoxDecoration(
color: _generalStyle.primaryButtonBackgroundColor,
shape: BoxShape.circle,
),
child: Padding(
padding: const EdgeInsets.all(8.0),
child: Icon(
Icons.chevron_right,
color: _generalStyle.primaryButtonTextColor,
size: 32,
),
),
),
),
),
),
),
),
// Close button
Positioned(
top: 16,
right: 16,
child: Container(
decoration: BoxDecoration(
color: _generalStyle.primaryButtonBackgroundColor,
shape: BoxShape.circle,
),
child: IconButton(
icon: Icon(Icons.close, color: _generalStyle.primaryButtonTextColor, size: 24),
onPressed: () => Navigator.of(context).pop(),
),
),
),
],
);
}
Widget _body(BuildContext context) {
return FutureBuilder<PictureDto?>(
future: _dto,
builder: (BuildContext context, AsyncSnapshot<PictureDto?> snapshot) {
if (snapshot.connectionState != ConnectionState.done) {
return const WaitingWidget();
}
if (snapshot.hasData) {
PictureDto? dto = snapshot.data;
void _navigateToPicture(int index) {
final pictures = widget.customerDto.pictures;
if (index >= 0 && index < pictures.length) {
setState(() {
_selectedPicture = pictures[index];
});
}
}
return _mainWidget(dto);
} else if (snapshot.hasError) {
var error = snapshot.error;
return (error is ServerError) ? GeneralErrorWidget.fromServerError(error) : GeneralErrorWidget(error: snapshot.error.toString());
}
return const WaitingWidget();
Widget _body(BuildContext context) {
return LayoutBuilder(
builder: (context, constraints) {
final isNarrow = constraints.maxWidth < 800;
return SingleChildScrollView(
child: isNarrow
? Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_imageWidget(_selectedPicture),
const SizedBox(height: 32),
_contentWidget(_selectedPicture),
],
)
: Row(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_imageWidget(_selectedPicture),
const SizedBox(width: 32),
Expanded(child: _contentWidget(_selectedPicture)),
],
),
);
},
);
}
Widget _mainWidget(PictureDto? dto) {
if (dto == null) {
return Text(
AppLocalizations.of(context)!.customerWidgetNotFound,
style: TextStyle(
fontFamily: _generalStyle.fontFamily,
fontWeight: FontWeight.bold,
fontSize: 20,
color: _generalStyle.secondaryWidgetBackgroundColor,
),
);
}
return Card(
margin: EdgeInsets.zero,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
child: Padding(
padding: const EdgeInsets.all(24.0),
child: LayoutBuilder(
builder: (context, constraints) {
final isNarrow = constraints.maxWidth < 800;
return SingleChildScrollView(
child: isNarrow
? Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_imageWidget(dto),
const SizedBox(height: 32),
_contentWidget(dto),
],
)
: Row(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_imageWidget(dto),
const SizedBox(width: 32),
_contentWidget(dto),
],
),
);
},
Widget _imageWidget(PictureDto dto) {
return GestureDetector(
onTap: () => _showFullscreenImage(dto),
child: MouseRegion(
cursor: SystemMouseCursors.click,
child: Image.memory(
base64Decode(dto.image),
fit: BoxFit.contain,
),
),
);
}
Widget _imageWidget(PictureDto dto) {
return Image.memory(
base64Decode(dto.image),
fit: BoxFit.contain,
void _showFullscreenImage(PictureDto dto) {
showDialog(
context: context,
builder: (BuildContext context) {
return PictureFullscreenDialog(dto: dto);
},
);
}
@@ -167,7 +234,7 @@ class _PictureWidgetState extends State<PictureWidget> {
Padding(
padding: const EdgeInsets.only(top: 4.0),
child: Text(
"Name of apotheke",
dto.customerListDto.name,
style: contentStyle,
),
),
@@ -181,7 +248,7 @@ class _PictureWidgetState extends State<PictureWidget> {
Padding(
padding: const EdgeInsets.only(top: 4.0),
child: Text(
"123445587474873",
dto.customerListDto.customerNumber,
style: contentStyle,
),
),
@@ -209,7 +276,7 @@ class _PictureWidgetState extends State<PictureWidget> {
Padding(
padding: const EdgeInsets.only(top: 4.0),
child: Container(
width: 300,
width: double.infinity,
height: 150,
decoration: BoxDecoration(
border: Border.all(color: _generalStyle.secondaryTextLabelColor.withValues(alpha: 0.3)),

View File

@@ -14,6 +14,8 @@ abstract interface class GeneralStyle {
Color get pageBackgroundColor;
Color get primaryCardColor;
Color get errorColor;
String get fontFamily;
@@ -44,6 +46,9 @@ class GeneralStyleImpl implements GeneralStyle {
@override
Color get primaryButtonTextColor => Colors.white;
@override
Color get primaryCardColor => Colors.white;
@override
Color get pageBackgroundColor => const Color(0xFFF5F5F5);

View File

@@ -3,7 +3,6 @@ import 'package:flutter/material.dart';
import 'package:fotodocumentation/main.dart';
import 'package:fotodocumentation/pages/customer/customer_list_widget.dart';
import 'package:fotodocumentation/pages/customer/customer_widget.dart';
import 'package:fotodocumentation/pages/customer/picture_widget.dart';
import 'package:fotodocumentation/pages/login/login_widget.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
@@ -17,7 +16,6 @@ class GlobalRouter {
static final String pathHome = "/home";
static final String pathCustomer = "/customer";
static final String pathPicture = "/picture";
static final String pathLogin = "/login";
static final GoRouter router = createRouter(pathHome);
@@ -47,14 +45,6 @@ class GlobalRouter {
return CustomerWidget(customerId: id ?? -1);
},
),
GoRoute(
path: "$pathPicture/:id",
builder: (context, state) {
var idStr = state.pathParameters['id'];
var id = idStr == null ? null : int.tryParse(idStr);
return PictureWidget(id: id ?? -1);
},
),
],
redirect: (context, state) {
var uriStr = state.uri.toString();