added frontend

This commit is contained in:
verboomp
2026-01-21 16:08:09 +01:00
parent d2e6f5164a
commit b3de3eec8c
74 changed files with 4938 additions and 26 deletions

30
.gitignore vendored
View File

@@ -42,9 +42,9 @@ target/
*/.dart_tool */.dart_tool
*/build */build
skillmatrix-frontend/build hartmann-foto-documentation-frontend/build
skillmatrix-frontend/coverage/ hartmann-foto-documentation-frontend/coverage/
skillmatrix-frontend/pubspec.lock hartmann-foto-documentation-frontend/pubspec.lock
# Avoid committing generated Javascript files: # Avoid committing generated Javascript files:
*.dart.js *.dart.js
@@ -60,16 +60,16 @@ skillmatrix-frontend/pubspec.lock
skillmatrix-docker/src/main/docker/skillmatrix-web-*.war hartmann-foto-documentation-docker/src/main/docker/hartmann-foto-documentation-web-*.war
skillmatrix-web/src/main/webapp/.last_build_id hartmann-foto-documentation-web/src/main/webapp/.last_build_id
skillmatrix-web/src/main/webapp/assets/ hartmann-foto-documentation-web/src/main/webapp/assets/
skillmatrix-web/src/main/webapp/canvaskit/ hartmann-foto-documentation-web/src/main/webapp/canvaskit/
skillmatrix-web/src/main/webapp/favicon.png hartmann-foto-documentation-web/src/main/webapp/favicon.png
skillmatrix-web/src/main/webapp/flutter.js hartmann-foto-documentation-web/src/main/webapp/flutter.js
skillmatrix-web/src/main/webapp/flutter_bootstrap.js hartmann-foto-documentation-web/src/main/webapp/flutter_bootstrap.js
skillmatrix-web/src/main/webapp/flutter_service_worker.js hartmann-foto-documentation-web/src/main/webapp/flutter_service_worker.js
skillmatrix-web/src/main/webapp/icons/ hartmann-foto-documentation-web/src/main/webapp/icons/
skillmatrix-web/src/main/webapp/index.html hartmann-foto-documentation-web/src/main/webapp/index.html
skillmatrix-web/src/main/webapp/manifest.json hartmann-foto-documentation-web/src/main/webapp/manifest.json
skillmatrix-web/src/main/webapp/version.json hartmann-foto-documentation-web/src/main/webapp/version.json

4
Jenkinsfile vendored
View File

@@ -58,7 +58,7 @@ pipeline {
''' '''
} }
} }
/*
stage ('Build Frontend') { stage ('Build Frontend') {
steps { steps {
echo "running Frontend build for branch ${env.BRANCH_NAME}" echo "running Frontend build for branch ${env.BRANCH_NAME}"
@@ -112,7 +112,7 @@ pipeline {
} }
} }
} }
*/
stage ('Build') { stage ('Build') {
steps { steps {
echo "running build for branch ${env.BRANCH_NAME}" echo "running build for branch ${env.BRANCH_NAME}"

View File

@@ -43,6 +43,6 @@ public class CustomerPictureService extends AbstractService {
// FIXME: do query // FIXME: do query
List<Customer> customers = queryService.callNamedQueryList(Customer.FIND_ALL); List<Customer> customers = queryService.callNamedQueryList(Customer.FIND_ALL);
customers.forEach(c -> c.getPictures().size()); customers.forEach(c -> c.getPictures().size());
return customers.parallelStream().map(c -> CustomerListValue.builder(c)).toList(); return customers.parallelStream().map(CustomerListValue::builder).toList();
} }
} }

View File

@@ -38,8 +38,8 @@ public class JwtTokenUtil {
private static final long REFRESH_TOKEN_VALIDITY = 30 * 24 * 60 * 60 * 1000L; // 30 days private static final long REFRESH_TOKEN_VALIDITY = 30 * 24 * 60 * 60 * 1000L; // 30 days
private static final long TEMP_2FA_TOKEN_VALIDITY = 5 * 60 * 1000L; // 5 minutes private static final long TEMP_2FA_TOKEN_VALIDITY = 5 * 60 * 1000L; // 5 minutes
private static final String ISSUER = "skillmatrix-jwt-issuer"; private static final String ISSUER = "foto-jwt-issuer";
private static final String AUDIENCE = "skillmatrix-api"; private static final String AUDIENCE = "foto-api";
private PrivateKey privateKey; private PrivateKey privateKey;
private PublicKey publicKey; private PublicKey publicKey;

View File

@@ -43,13 +43,13 @@ public class LoginUtils {
private Optional<SecurityIdentity> authenticate(String username, String password) { private Optional<SecurityIdentity> authenticate(String username, String password) {
try { try {
LOG.error("Login with username: " + username + " password: " + password); LOG.debug("Login with username & password " + username);
Principal principal = new NamePrincipal(username); Principal principal = new NamePrincipal(username);
PasswordGuessEvidence evidence = new PasswordGuessEvidence(password.toCharArray()); PasswordGuessEvidence evidence = new PasswordGuessEvidence(password.toCharArray());
SecurityDomain sd = SecurityDomain.getCurrent(); SecurityDomain sd = SecurityDomain.getCurrent();
SecurityIdentity identity = sd.authenticate(principal, evidence); SecurityIdentity identity = sd.authenticate(principal, evidence);
LOG.error("Login identity: " + identity); LOG.debug("Login identity: " + identity);
return Optional.ofNullable(identity); return Optional.ofNullable(identity);
} catch (RealmUnavailableException | SecurityException e) { } catch (RealmUnavailableException | SecurityException e) {
LOG.warn("Failed to authenticate user " + e.getMessage(), e); LOG.warn("Failed to authenticate user " + e.getMessage(), e);

View File

@@ -22,6 +22,7 @@ import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces; import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.Context; import jakarta.ws.rs.core.Context;
import jakarta.ws.rs.core.Response; import jakarta.ws.rs.core.Response;
import jakarta.ws.rs.core.Response.Status;
import marketing.heyday.hartmann.fotodocumentation.core.service.LoginService; import marketing.heyday.hartmann.fotodocumentation.core.service.LoginService;
import marketing.heyday.hartmann.fotodocumentation.core.utils.LoginUtils; import marketing.heyday.hartmann.fotodocumentation.core.utils.LoginUtils;
import marketing.heyday.hartmann.fotodocumentation.rest.vo.TokenPairValue; import marketing.heyday.hartmann.fotodocumentation.rest.vo.TokenPairValue;
@@ -58,7 +59,7 @@ public class LoginResource {
Optional<SecurityIdentity> identity = loginUtils.authenticate(httpServletRequest); Optional<SecurityIdentity> identity = loginUtils.authenticate(httpServletRequest);
if (identity.isEmpty()) { if (identity.isEmpty()) {
LOG.debug("identity empty login invalid"); LOG.debug("identity empty login invalid");
return Response.status(401).build(); return Response.status(Status.UNAUTHORIZED).build();
} }
String username = identity.get().getPrincipal().getName(); String username = identity.get().getPrincipal().getName();

View File

@@ -333,7 +333,7 @@ eZlo8cWlAC5welD3dz1qxEo=
<!-- patrick --> <!-- patrick -->
<token-realm name="fotoDocumentationJwtRealm" principal-claim="username"> <token-realm name="fotoDocumentationJwtRealm" principal-claim="username">
<jwt issuer="skillmatrix-jwt-issuer" audience="skillmatrix-api" <jwt issuer="foto-jwt-issuer" audience="foto-api"
public-key="-----BEGIN PUBLIC KEY----- public-key="-----BEGIN PUBLIC KEY-----
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAitRLwUhlIP/iHUzZ5al1 MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAitRLwUhlIP/iHUzZ5al1
pFrS8qUQ1uWLUNYpU9OOgEHz8kwivVhiVKqrcX4jsUNKilVrF2Xf9ycBz56qYDkc pFrS8qUQ1uWLUNYpU9OOgEHz8kwivVhiVKqrcX4jsUNKilVrF2Xf9ycBz56qYDkc

View File

@@ -2,14 +2,12 @@ package marketing.heyday.hartmann.fotodocumentation.rest;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
import java.io.File;
import java.io.IOException; import java.io.IOException;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
import org.apache.http.HttpResponse; import org.apache.http.HttpResponse;
import org.apache.http.client.fluent.Request; import org.apache.http.client.fluent.Request;
import org.apache.http.entity.ContentType;
import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.MethodOrderer.OrderAnnotation; import org.junit.jupiter.api.MethodOrderer.OrderAnnotation;
import org.junit.jupiter.api.Order; import org.junit.jupiter.api.Order;

View File

@@ -0,0 +1,12 @@
{
"permissions": {
"allow": [
"Bash(flutter test:*)",
"Bash(cat:*)",
"Bash(flutter gen-l10n:*)",
"Bash(flutter analyze:*)"
],
"deny": [],
"ask": []
}
}

View File

@@ -0,0 +1,45 @@
# Miscellaneous
*.class
*.log
*.pyc
*.swp
.DS_Store
.atom/
.build/
.buildlog/
.history
.svn/
.swiftpm/
migrate_working_dir/
# IntelliJ related
*.iml
*.ipr
*.iws
.idea/
# The .vscode folder contains launch configuration and tasks you configure in
# VS Code which you may wish to be included in version control, so this line
# is commented out by default.
#.vscode/
# Flutter/Dart/Pub related
**/doc/api/
**/ios/Flutter/.last_build_id
.dart_tool/
.flutter-plugins-dependencies
.pub-cache/
.pub/
/build/
/coverage/
# Symbolication related
app.*.symbols
# Obfuscation related
app.*.map.json
# Android Studio will place build artifacts here
/android/app/debug
/android/app/profile
/android/app/release

View File

@@ -0,0 +1,30 @@
# This file tracks properties of this Flutter project.
# Used by Flutter tool to assess capabilities and perform upgrades etc.
#
# This file should be version controlled and should not be manually edited.
version:
revision: "ac4e799d237041cf905519190471f657b657155a"
channel: "stable"
project_type: app
# Tracks metadata for the flutter migrate command
migration:
platforms:
- platform: root
create_revision: ac4e799d237041cf905519190471f657b657155a
base_revision: ac4e799d237041cf905519190471f657b657155a
- platform: web
create_revision: ac4e799d237041cf905519190471f657b657155a
base_revision: ac4e799d237041cf905519190471f657b657155a
# User provided section
# List of Local paths (relative to this file) that should be
# ignored by the migrate tool.
#
# Files that are not part of the templates will be ignored by default.
unmanaged_files:
- 'lib/main.dart'
- 'ios/Runner.xcodeproj/project.pbxproj'

View File

@@ -0,0 +1,25 @@
{
// Use IntelliSense to learn about possible attributes.
// Hover to view descriptions of existing attributes.
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0",
"configurations": [
{
"name": "foto-frontend",
"request": "launch",
"type": "dart"
},
{
"name": "foto-frontend (profile mode)",
"request": "launch",
"type": "dart",
"flutterMode": "profile"
},
{
"name": "foto-frontend (release mode)",
"request": "launch",
"type": "dart",
"flutterMode": "release"
}
]
}

View File

@@ -0,0 +1,16 @@
# foto documentation
A new Flutter project.
## Getting Started
This project is a starting point for a Flutter application.
A few resources to get you started if this is your first Flutter project:
- [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab)
- [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook)
For help getting started with Flutter development, view the
[online documentation](https://docs.flutter.dev/), which offers tutorials,
samples, guidance on mobile development, and a full API reference.

View File

@@ -0,0 +1,28 @@
# This file configures the analyzer, which statically analyzes Dart code to
# check for errors, warnings, and lints.
#
# The issues identified by the analyzer are surfaced in the UI of Dart-enabled
# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be
# invoked from the command line by running `flutter analyze`.
# The following line activates a set of recommended lints for Flutter apps,
# packages, and plugins designed to encourage good coding practices.
include: package:flutter_lints/flutter.yaml
linter:
# The lint rules applied to this project can be customized in the
# section below to disable rules from the `package:flutter_lints/flutter.yaml`
# included above or to enable additional rules. A list of all available lints
# and their documentation is published at https://dart.dev/lints.
#
# Instead of disabling a lint rule for the entire project in the
# section below, it can also be suppressed for a single line of code
# or a specific dart file by using the `// ignore: name_of_lint` and
# `// ignore_for_file: name_of_lint` syntax on the line or in the file
# producing the lint.
rules:
# avoid_print: false # Uncomment to disable the `avoid_print` rule
# prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule
# Additional information about this file can be found at
# https://dart.dev/guides/language/analysis-options

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,3 @@
description: This file stores settings for Dart & Flutter DevTools.
documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states
extensions:

View File

@@ -0,0 +1,3 @@
arb-dir: lib/l10n
template-arb-file: app_de.arb
output-localization-file: app_localizations.dart

View File

@@ -0,0 +1,174 @@
import 'dart:convert' show jsonDecode, jsonEncode;
import 'package:fotodocumentation/dto/base_dto.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/http_client_utils.dart';
import 'package:fotodocumentation/utils/jwt_token_storage.dart';
import 'package:fotodocumentation/utils/url_utils.dart';
import 'package:http/http.dart' as http;
import 'package:http/http.dart' show Response;
import 'package:fotodocumentation/main.dart' show logger;
abstract class BaseController {
UrlUtils get uriUtils => DiContainer.get();
JwtTokenStorage get _jwtTokenStorage => DiContainer.get();
HttpClientUtils get httpClientUtils => DiContainer.get();
Future<Header> getAuthHeader() async {
final accessToken = await _jwtTokenStorage.getAccessToken();
if (accessToken != null && accessToken.isNotEmpty) {
// Use JWT Bearer token
return Header('Authorization', 'Bearer $accessToken');
} else {
return const Header("Accept-Language", "en-US");
}
}
Exception getServerError(Response response) {
return Exception("Error receiving data from server");
}
Future<List<T>> runGetListWithAuth<T>(String uriStr, List<T> Function(dynamic) convert) async {
http.Client client = httpClientUtils.client;
try {
Header cred = await getAuthHeader();
Uri uri = Uri.parse(uriStr);
var response = await client.get(uri, headers: {cred.name: cred.value});
if (response.statusCode == 200) {
String text = response.body;
var jsonArray = jsonDecode(text);
return convert(jsonArray);
} else {
throw ServerError(response.statusCode); // Exception("Failed to get server data ${response.statusCode}");
}
} catch (e) {
logger.e("exception $e");
rethrow;
}
}
Future<T?> runGetWithAuth<T>(String uriStr, T Function(dynamic) convert) async {
http.Client client = httpClientUtils.client;
try {
Header cred = await getAuthHeader();
Uri uri = Uri.parse(uriStr);
var response = await client.get(uri, headers: {cred.name: cred.value});
if (response.statusCode == 200) {
String text = response.body;
var jsonArray = jsonDecode(text);
return convert(jsonArray);
} else {
throw ServerError(response.statusCode); // Exception("Failed to get server data ${response.statusCode}");
}
} catch (e) {
logger.e("exception $e");
rethrow;
}
}
Future<bool> runDeleteWithAuth(String uriStr) async {
http.Client client = httpClientUtils.client;
Header cred = await getAuthHeader();
Uri uri = Uri.parse(uriStr);
var response = await client.delete(uri, headers: {cred.name: cred.value});
return response.statusCode == 200;
}
Future<bool> runPutWithAuth(String uriStr) async {
http.Client client = httpClientUtils.client;
Header cred = await getAuthHeader();
Uri uri = Uri.parse(uriStr);
var response = await client.put(uri, headers: {cred.name: cred.value});
return response.statusCode == 200;
}
Future<ServerReply<T>> runSaveNew<T extends DtoMapAble>(String uriStr, T dtoObj, Function(http.Response response, T dto) processReply) async {
http.Client client = httpClientUtils.client;
try {
Header cred = await getAuthHeader();
String body = jsonEncode(dtoObj.toMap());
Uri uri = Uri.parse(uriStr);
var response = await client.post(uri, headers: {cred.name: cred.value, "Accept": "application/json", "Content-Type": "application/json"}, body: body);
return processReply(response, dtoObj);
} catch (e) {
logger.e("exception $e");
}
return ServerReply(ServerState.error, dtoObj);
}
Future<ServerReply<T>> runSaveUpdate<T extends DtoMapAble>(String uriStr, T dtoObj, Function(http.Response response, T dto) processReply) async {
http.Client client = httpClientUtils.client;
try {
Header cred = await getAuthHeader();
String body = jsonEncode(dtoObj.toMap());
Uri uri = Uri.parse(uriStr);
var response = await client.put(uri, headers: {cred.name: cred.value, "Accept": "application/json", "Content-Type": "application/json"}, body: body);
return processReply(response, dtoObj);
} catch (e) {
logger.e("exception $e");
}
return ServerReply(ServerState.error, dtoObj);
}
ServerReply<T> processServerResponse<T>(Response response, T dto, T Function(Map<String, dynamic> json) fromJson) {
if (response.statusCode == 200) {
String text = response.body;
var json = jsonDecode(text);
var dto = fromJson(json);
return ServerReply(ServerState.ok, dto);
} else if (response.statusCode == 400 || response.statusCode == 409) {
String text = response.body;
try {
var json = jsonDecode(text);
var error = ErrorDto.fromJson(json);
return ServerReply(ServerState.duplicate, dto, error: error);
} catch (e) {
return ServerReply(ServerState.error, dto, error: ErrorDto(response.statusCode, text));
}
} else if (response.statusCode == 403) {
var error = ErrorDto(403, "Not allowed.");
return ServerReply(ServerState.error, dto, error: error);
}
return ServerReply(ServerState.error, dto, error: ErrorDto(response.statusCode, "Internal server error"));
}
}
class Header {
final String name;
final String value;
const Header(this.name, this.value);
@override
String toString() => '$runtimeType: $name, $value';
}
class ServerReply<T> {
ServerState state;
T entity;
ErrorDto? error;
ServerReply(this.state, this.entity, {this.error});
@override
String toString() => '$runtimeType: $state, $entity, $error';
}
class ServerError {
int statusCode;
ServerError(this.statusCode);
}
enum ServerState {
ok,
duplicate,
error,
;
}

View File

@@ -0,0 +1,102 @@
import 'dart:convert' show base64, utf8;
import 'dart:convert' show jsonDecode, jsonEncode;
import 'package:fotodocumentation/controller/base_controller.dart';
import 'package:fotodocumentation/dto/jwt_token_pair_dto.dart';
import 'package:fotodocumentation/main.dart' show logger;
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/jwt_token_storage.dart';
import 'package:http/http.dart' as http;
typedef AuthenticateReply = ({JwtTokenPairDto? jwtTokenPairDto});
abstract interface class LoginController {
Future<AuthenticateReply> authenticate(String username, String password);
Future<bool> refreshAccessToken();
Future<bool> isUsingJwtAuth();
}
class LoginControllerImpl extends BaseController implements LoginController {
final String path = "login";
JwtTokenStorage get _jwtTokenStorage => DiContainer.get();
@override
Future<AuthenticateReply> authenticate(String username, String password) async {
http.Client client = httpClientUtils.client;
try {
Header cred = _getLoginHeader(username, password);
String uriStr = '${uriUtils.getBaseUrl()}$path';
Uri uri = Uri.parse(uriStr);
var response = await client.get(uri, headers: {cred.name: cred.value});
if (response.statusCode == 200) {
final Map<String, dynamic> data = Map.castFrom(jsonDecode(response.body));
final tokenPair = JwtTokenPairDto.fromJson(data);
// Store tokens securely
await _jwtTokenStorage.saveTokens(tokenPair.accessToken, tokenPair.refreshToken);
// Load user data using the new token
return (jwtTokenPairDto: tokenPair);
} else {
logger.e('Authentication failed: ${response.statusCode} ${response.body}');
return (jwtTokenPairDto: null);
}
} catch (e) {
logger.e("Authentication error: $e");
return (jwtTokenPairDto: null);
}
}
@override
Future<bool> refreshAccessToken() async {
try {
final refreshToken = await _jwtTokenStorage.getRefreshToken();
if (refreshToken == null) {
logger.i('No refresh token available');
return false;
}
String uriStr = '${uriUtils.getBaseUrl()}$path/login/refresh';
Uri uri = Uri.parse(uriStr);
final response = await http.post(
uri,
headers: {'Content-Type': 'application/json'},
body: jsonEncode({
'refreshToken': refreshToken,
}),
);
if (response.statusCode == 200) {
final data = jsonDecode(response.body);
final newAccessToken = data['accessToken'] as String;
// Update only the access token (keep same refresh token)
await _jwtTokenStorage.updateAccessToken(newAccessToken);
logger.d('Access token refreshed successfully');
return true;
} else {
logger.d('Token refresh failed: ${response.statusCode} ${response.body}');
return false;
}
} catch (e) {
logger.e('Token refresh error: $e');
return false;
}
}
@override
Future<bool> isUsingJwtAuth() async {
return await _jwtTokenStorage.hasTokens();
}
Header _getLoginHeader(String username, String password) {
String combined = "$username:$password";
final bytes = utf8.encode(combined);
String asBase64 = base64.encode(bytes);
return Header("Authorization", "Basic $asBase64");
}
}

View File

@@ -0,0 +1,15 @@
final class ErrorDto {
int error;
String message;
ErrorDto(this.error, this.message);
ErrorDto.fromJson(Map<String, dynamic> json)
: error = json['error'] as int,
message = json['message'];
}
abstract interface class DtoMapAble {
Map<String, dynamic> toMap();
}

View File

@@ -0,0 +1,50 @@
/// DTO representing a failes login attempt request for 2fa token.
class TokenRequiredDto {
final bool? tokenRequired;
final bool? tokenInValid;
TokenRequiredDto({
required this.tokenRequired,
required this.tokenInValid,
});
/// Create from JSON response
factory TokenRequiredDto.fromJson(Map<String, dynamic> json) {
return TokenRequiredDto(
tokenRequired: json['tokenRequired'] as bool,
tokenInValid: json['tokenInValid'] as bool,
);
}
}
/// DTO representing a pair of JWT tokens from the backend.
class JwtTokenPairDto {
final String accessToken;
final String refreshToken;
JwtTokenPairDto({
required this.accessToken,
required this.refreshToken,
});
/// Create from JSON response
factory JwtTokenPairDto.fromJson(Map<String, dynamic> json) {
return JwtTokenPairDto(
accessToken: json['accessToken'] as String,
refreshToken: json['refreshToken'] as String,
);
}
/// Convert to JSON (for serialization if needed)
Map<String, dynamic> toJson() {
return {
'accessToken': accessToken,
'refreshToken': refreshToken,
};
}
@override
String toString() {
return 'JwtTokenPairDto{accessToken: [REDACTED], refreshToken: [REDACTED]}';
}
}

View File

@@ -0,0 +1,75 @@
{
"@@locale": "de",
"searchTFHint": "Suchtext",
"@searchTFHint": {
"description": "Search hint TextField"
},
"searchButtonLabel": "Suchen",
"@searchButtonLabel": {
"description": "Search button label"
},
"loginUsernameTFLabel": "Benutzername",
"@loginUsernameTFLabel": {
"description": "Usernamt TextField Label"
},
"loginPasswordTFLabel": "Passwort",
"@loginPasswordTFLabel": {
"description": "Password TextField Label"
},
"loginLoginButtonLabel": "Anmelden",
"@loginLoginButtonLabel": {
"description": "Login Button Label"
},
"errorWidgetStatusCode": "Statuscode {statusCode}",
"@errorWidgetStatusCode": {
"description": "Error message showing server status code",
"placeholders": {
"statusCode": {
"type": "int"
}
}
},
"errorWidget": "Fehler: {name}",
"@errorWidget": {
"description": "Error widget text",
"placeholders": {
"name": {
"type": "String",
"example": "Error text"
}
}
},
"errorWidgetRetryButton": "Wiederholen",
"@errorWidgetRetryButton": {
"description": "Retry button text for error widget"
},
"submitWidget": "Speichern",
"@submitWidget": {
"description": "Save Button text"
},
"textInputWidgetValidatorText": "Bitte geben Sie einen Text ein",
"@textInputWidgetValidatorText": {
"description": "Awaiting result info text"
},
"waitingWidget": "Warten auf Ergebnis …",
"@waitingWidget": {
"description": "Awaiting result info text"
},
"deleteDialogTitle": "Löschen",
"@deleteDialogTitle": {
"description": "Delete dialog title"
},
"deleteDialogText": "Sind Sie sicher, dass Sie diese Eintrag löschen möchten?",
"@deleteDialogText": {
"description": "Delete dialog text"
},
"deleteDialogButtonCancel": "Nein",
"@deleteDialogButtonCancel": {
"description": "Cancel Button text"
},
"deleteDialogButtonApprove": "Ja",
"@deleteDialogButtonApprove": {
"description": "Approve Button text"
}
}

View File

@@ -0,0 +1,217 @@
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:intl/intl.dart' as intl;
import 'app_localizations_de.dart';
// ignore_for_file: type=lint
/// Callers can lookup localized strings with an instance of AppLocalizations
/// returned by `AppLocalizations.of(context)`.
///
/// Applications need to include `AppLocalizations.delegate()` in their app's
/// `localizationDelegates` list, and the locales they support in the app's
/// `supportedLocales` list. For example:
///
/// ```dart
/// import 'l10n/app_localizations.dart';
///
/// return MaterialApp(
/// localizationsDelegates: AppLocalizations.localizationsDelegates,
/// supportedLocales: AppLocalizations.supportedLocales,
/// home: MyApplicationHome(),
/// );
/// ```
///
/// ## Update pubspec.yaml
///
/// Please make sure to update your pubspec.yaml to include the following
/// packages:
///
/// ```yaml
/// dependencies:
/// # Internationalization support.
/// flutter_localizations:
/// sdk: flutter
/// intl: any # Use the pinned version from flutter_localizations
///
/// # Rest of dependencies
/// ```
///
/// ## iOS Applications
///
/// iOS applications define key application metadata, including supported
/// locales, in an Info.plist file that is built into the application bundle.
/// To configure the locales supported by your app, youll need to edit this
/// file.
///
/// First, open your projects ios/Runner.xcworkspace Xcode workspace file.
/// Then, in the Project Navigator, open the Info.plist file under the Runner
/// projects Runner folder.
///
/// Next, select the Information Property List item, select Add Item from the
/// Editor menu, then select Localizations from the pop-up menu.
///
/// Select and expand the newly-created Localizations item then, for each
/// locale your application supports, add a new item and select the locale
/// you wish to add from the pop-up menu in the Value field. This list should
/// be consistent with the languages listed in the AppLocalizations.supportedLocales
/// property.
abstract class AppLocalizations {
AppLocalizations(String locale)
: localeName = intl.Intl.canonicalizedLocale(locale.toString());
final String localeName;
static AppLocalizations? of(BuildContext context) {
return Localizations.of<AppLocalizations>(context, AppLocalizations);
}
static const LocalizationsDelegate<AppLocalizations> delegate =
_AppLocalizationsDelegate();
/// A list of this localizations delegate along with the default localizations
/// delegates.
///
/// Returns a list of localizations delegates containing this delegate along with
/// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate,
/// and GlobalWidgetsLocalizations.delegate.
///
/// Additional delegates can be added by appending to this list in
/// MaterialApp. This list does not have to be used at all if a custom list
/// of delegates is preferred or required.
static const List<LocalizationsDelegate<dynamic>> localizationsDelegates =
<LocalizationsDelegate<dynamic>>[
delegate,
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
];
/// A list of this localizations delegate's supported locales.
static const List<Locale> supportedLocales = <Locale>[Locale('de')];
/// Search hint TextField
///
/// In de, this message translates to:
/// **'Suchtext'**
String get searchTFHint;
/// Search button label
///
/// In de, this message translates to:
/// **'Suchen'**
String get searchButtonLabel;
/// Usernamt TextField Label
///
/// In de, this message translates to:
/// **'Benutzername'**
String get loginUsernameTFLabel;
/// Password TextField Label
///
/// In de, this message translates to:
/// **'Passwort'**
String get loginPasswordTFLabel;
/// Login Button Label
///
/// In de, this message translates to:
/// **'Anmelden'**
String get loginLoginButtonLabel;
/// Error message showing server status code
///
/// In de, this message translates to:
/// **'Statuscode {statusCode}'**
String errorWidgetStatusCode(int statusCode);
/// Error widget text
///
/// In de, this message translates to:
/// **'Fehler: {name}'**
String errorWidget(String name);
/// Retry button text for error widget
///
/// In de, this message translates to:
/// **'Wiederholen'**
String get errorWidgetRetryButton;
/// Save Button text
///
/// In de, this message translates to:
/// **'Speichern'**
String get submitWidget;
/// Awaiting result info text
///
/// In de, this message translates to:
/// **'Bitte geben Sie einen Text ein'**
String get textInputWidgetValidatorText;
/// Awaiting result info text
///
/// In de, this message translates to:
/// **'Warten auf Ergebnis …'**
String get waitingWidget;
/// Delete dialog title
///
/// In de, this message translates to:
/// **'Löschen'**
String get deleteDialogTitle;
/// Delete dialog text
///
/// In de, this message translates to:
/// **'Sind Sie sicher, dass Sie diese Eintrag löschen möchten?'**
String get deleteDialogText;
/// Cancel Button text
///
/// In de, this message translates to:
/// **'Nein'**
String get deleteDialogButtonCancel;
/// Approve Button text
///
/// In de, this message translates to:
/// **'Ja'**
String get deleteDialogButtonApprove;
}
class _AppLocalizationsDelegate
extends LocalizationsDelegate<AppLocalizations> {
const _AppLocalizationsDelegate();
@override
Future<AppLocalizations> load(Locale locale) {
return SynchronousFuture<AppLocalizations>(lookupAppLocalizations(locale));
}
@override
bool isSupported(Locale locale) =>
<String>['de'].contains(locale.languageCode);
@override
bool shouldReload(_AppLocalizationsDelegate old) => false;
}
AppLocalizations lookupAppLocalizations(Locale locale) {
// Lookup logic when only language code is specified.
switch (locale.languageCode) {
case 'de':
return AppLocalizationsDe();
}
throw FlutterError(
'AppLocalizations.delegate failed to load unsupported locale "$locale". This is likely '
'an issue with the localizations generation tool. Please file an issue '
'on GitHub with a reproducible sample app and the gen-l10n configuration '
'that was used.');
}

View File

@@ -0,0 +1,60 @@
// ignore: unused_import
import 'package:intl/intl.dart' as intl;
import 'app_localizations.dart';
// ignore_for_file: type=lint
/// The translations for German (`de`).
class AppLocalizationsDe extends AppLocalizations {
AppLocalizationsDe([String locale = 'de']) : super(locale);
@override
String get searchTFHint => 'Suchtext';
@override
String get searchButtonLabel => 'Suchen';
@override
String get loginUsernameTFLabel => 'Benutzername';
@override
String get loginPasswordTFLabel => 'Passwort';
@override
String get loginLoginButtonLabel => 'Anmelden';
@override
String errorWidgetStatusCode(int statusCode) {
return 'Statuscode $statusCode';
}
@override
String errorWidget(String name) {
return 'Fehler: $name';
}
@override
String get errorWidgetRetryButton => 'Wiederholen';
@override
String get submitWidget => 'Speichern';
@override
String get textInputWidgetValidatorText => 'Bitte geben Sie einen Text ein';
@override
String get waitingWidget => 'Warten auf Ergebnis …';
@override
String get deleteDialogTitle => 'Löschen';
@override
String get deleteDialogText =>
'Sind Sie sicher, dass Sie diese Eintrag löschen möchten?';
@override
String get deleteDialogButtonCancel => 'Nein';
@override
String get deleteDialogButtonApprove => 'Ja';
}

View File

@@ -0,0 +1,49 @@
import 'package:flutter/material.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:logger/web.dart' show DateTimeFormat, Logger, PrettyPrinter;
import 'package:fotodocumentation/controller/login_controller.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/main_utils.dart';
import 'package:fotodocumentation/utils/global_router.dart';
var logger = Logger(
printer: PrettyPrinter(methodCount: 2, errorMethodCount: 8, colors: true, printEmojis: true, dateTimeFormat: DateTimeFormat.onlyTimeAndSinceStart),
);
void main() async {
WidgetsFlutterBinding.ensureInitialized();
DiContainer.instance.initState();
final theme = await ThemeLoader.loadTheme();
LoginController loginController = DiContainer.get();
//await loginController.isLoggedIn();
runApp(FotoDocumentationApp(theme: theme));
}
class FotoDocumentationApp extends StatelessWidget {
final ThemeData theme;
const FotoDocumentationApp({super.key, required this.theme});
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp.router(
title: 'Hartmann Foto App',
localizationsDelegates: [
AppLocalizations.delegate,
GlobalMaterialLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
],
supportedLocales: [
Locale('de'),
],
scrollBehavior: MyCustomScrollBehavior(), // <== needed for web horizontal scroll behavior
theme: theme,
routerConfig: GlobalRouter.router);
}
}

View File

@@ -0,0 +1,15 @@
import 'package:flutter/material.dart';
class CustomerWidget extends StatefulWidget {
const CustomerWidget({super.key});
@override
State<CustomerWidget> createState() => _CustomerWidgetState();
}
class _CustomerWidgetState extends State<CustomerWidget> {
@override
Widget build(BuildContext context) {
return const Placeholder();
}
}

View File

@@ -0,0 +1,23 @@
import 'package:flutter/material.dart';
class LandingPageWidget extends StatefulWidget {
final Widget child;
const LandingPageWidget({super.key, required this.child});
@override
State<LandingPageWidget> createState() => _LandingPageWidgetState();
}
class _LandingPageWidgetState extends State<LandingPageWidget> {
@override
void initState() {
super.initState();
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: widget.child,
);
}
}

View File

@@ -0,0 +1,154 @@
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:go_router/go_router.dart';
import 'package:fotodocumentation/controller/login_controller.dart';
import 'package:fotodocumentation/dto/jwt_token_pair_dto.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/pages/ui_utils/component/general_submit_widget.dart';
import 'package:fotodocumentation/pages/ui_utils/header_utils.dart';
import 'package:fotodocumentation/pages/ui_utils/modern_app_bar.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
class LoginWidget extends StatefulWidget {
const LoginWidget({super.key});
@override
State<LoginWidget> createState() => _LoginWidgetState();
}
class _LoginWidgetState extends State<LoginWidget> {
HeaderUtils get _headerUtils => DiContainer.get();
LoginController get _loginController => DiContainer.get();
LoginCredentials get _loginCredentials => DiContainer.get();
final GlobalKey<FormState> _formKey = GlobalKey<FormState>();
final _usernameController = TextEditingController();
final _passwordController = TextEditingController();
String? _error;
final FocusNode _focusNode = FocusNode();
@override
void dispose() {
_focusNode.dispose();
_usernameController.dispose();
_passwordController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: ModernAppBar(
title: _headerUtils.titleWidget("Login title"),
actions: [],
),
body: _body(context),
);
}
Widget _body(BuildContext context) {
return Container(
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
Colors.grey[50]!,
Colors.white,
],
),
),
child: _content(context),
);
}
Widget _content(BuildContext context) {
return Padding(
padding: const EdgeInsets.all(20.0),
child: Form(
key: _formKey,
child: KeyboardListener(
focusNode: _focusNode,
onKeyEvent: (event) {
if (event is KeyDownEvent && event.logicalKey == LogicalKeyboardKey.enter) {
_actionSubmit(context);
}
},
child: ListView(
children: [
Card(
elevation: 4,
margin: EdgeInsets.zero,
clipBehavior: Clip.antiAlias,
child: Padding(
padding: const EdgeInsets.all(30.0),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
TextFormField(
key: Key("username"),
controller: _usernameController,
decoration: InputDecoration(
border: UnderlineInputBorder(),
labelText: AppLocalizations.of(context)!.loginUsernameTFLabel,
),
),
const SizedBox(height: 10),
TextFormField(
key: Key("password"),
controller: _passwordController,
obscureText: true,
decoration: InputDecoration(
border: UnderlineInputBorder(),
labelText: AppLocalizations.of(context)!.loginPasswordTFLabel,
),
),
const SizedBox(height: 10),
if (_error != null) ...[
Text(
_error!,
style: const TextStyle(color: Colors.red),
),
const SizedBox(height: 10),
],
GeneralSubmitWidget(
key: const Key("submit"),
onSelect: () async => await _actionSubmit(context),
title: AppLocalizations.of(context)!.loginLoginButtonLabel,
),
const SizedBox(height: 30),
],
),
),
),
],
),
),
),
);
}
Future<void> _actionSubmit(BuildContext context) async {
String username = _usernameController.text;
String password = _passwordController.text;
AuthenticateReply authenticateReply = await _loginController.authenticate(username, password);
JwtTokenPairDto? jwtTokenPairDto = authenticateReply.jwtTokenPairDto;
if (jwtTokenPairDto == null) {
setState(() => _error = "Error message");
return;
}
_loginCredentials.setLoggedIn(true);
if (context.mounted) {
context.go("/");
}
}
}

View File

@@ -0,0 +1,55 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/controller/base_controller.dart';
class GeneralErrorWidget extends StatelessWidget {
final String error;
final Function()? reload;
final int? statusCode;
const GeneralErrorWidget({super.key, required this.error, this.reload, this.statusCode});
factory GeneralErrorWidget.fromServerError(ServerError serverError, {Function()? reload}) {
return GeneralErrorWidget(error: "", reload: reload, statusCode: serverError.statusCode);
}
factory GeneralErrorWidget.fromSnapshot(AsyncSnapshot snapshot, {Function()? reload}) {
var error = snapshot.error;
if (error is ServerError) {
return GeneralErrorWidget.fromServerError(error, reload: () => reload);
}
return GeneralErrorWidget(error: snapshot.error.toString(), reload: () => reload);
}
@override
Widget build(BuildContext context) {
final localizations = AppLocalizations.of(context)!;
final String errorMessage = statusCode != null
? localizations.errorWidgetStatusCode(statusCode!)
: localizations.errorWidget(error);
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(Icons.error_outline, size: 60, color: Colors.red[300]),
Padding(
padding: const EdgeInsets.only(top: 16),
child: Text(
errorMessage,
style: TextStyle(color: Colors.red[700]),
textAlign: TextAlign.center,
),
),
const SizedBox(height: 16),
if (reload != null) ...[
ElevatedButton(
onPressed: reload,
child: Text(localizations.errorWidgetRetryButton),
),
],
],
),
);
}
}

View File

@@ -0,0 +1,32 @@
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';
typedef SubmitCallback = void Function();
class GeneralSubmitWidget extends StatelessWidget {
GeneralStyle get _generalStyle => DiContainer.get();
final SubmitCallback onSelect;
final String? title;
const GeneralSubmitWidget({super.key, required this.onSelect, this.title});
@override
Widget build(BuildContext context) {
String text = title ?? AppLocalizations.of(context)!.submitWidget;
return Row(
mainAxisAlignment: MainAxisAlignment.end,
crossAxisAlignment: CrossAxisAlignment.end,
children: [
ElevatedButton(
key: Key("SubmitWidgetButton"),
style: _generalStyle.elevatedButtonStyle,
onPressed: onSelect,
child: Text(text),
),
],
);
}
}

View File

@@ -0,0 +1,66 @@
import 'package:flutter/material.dart';
class PageHeaderWidget extends StatelessWidget {
final IconData iconData;
final String text;
final String subText;
final Color? iconColor;
const PageHeaderWidget({super.key, this.iconData = Icons.business, required this.text, this.subText = "", this.iconColor});
@override
Widget build(BuildContext context) {
final color = iconColor ?? Theme.of(context).colorScheme.primary;
return Card(
elevation: 2,
margin: EdgeInsets.zero,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
side: BorderSide(
color: Colors.grey[300]!,
width: 1,
),
),
child: Padding(
padding: const EdgeInsets.all(24.0),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Container(
padding: const EdgeInsets.all(8),
decoration: BoxDecoration(
color: color.withAlpha(51),
shape: BoxShape.circle,
),
child: Icon(
iconData,
size: 32,
color: color,
),
),
const SizedBox(width: 16),
Text(
text,
style: Theme.of(context).textTheme.headlineSmall?.copyWith(
fontWeight: FontWeight.bold,
),
),
],
),
if (subText.isNotEmpty) ...[
const SizedBox(height: 16),
Text(
subText,
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
color: Colors.grey[600],
),
),
],
],
),
),
);
}
}

View File

@@ -0,0 +1,81 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
class SearchBarCardWidget extends StatefulWidget {
final TextEditingController searchController;
final Function(String) onSearch;
const SearchBarCardWidget({super.key, required this.searchController, required this.onSearch});
@override
State<SearchBarCardWidget> createState() => _SearchBarCardWidgetState();
}
class _SearchBarCardWidgetState extends State<SearchBarCardWidget> {
@override
Widget build(BuildContext context) {
return Card(
elevation: 2,
margin: EdgeInsets.zero,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
side: BorderSide(
color: Colors.grey[300]!,
width: 1,
),
),
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
child: Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(
child: TextField(
key: Key("Search_text_field"),
controller: widget.searchController,
textAlignVertical: TextAlignVertical.center,
decoration: InputDecoration(
hintText: AppLocalizations.of(context)!.searchTFHint,
border: InputBorder.none,
prefixIcon: const Icon(Icons.search, size: 28),
contentPadding: EdgeInsets.zero,
isDense: true,
suffixIcon: InkWell(
key: Key("Search_text_clear_button"),
onTap: () => _actionClear(),
child: const Icon(
Icons.close,
color: Colors.black,
),
)
),
onSubmitted: (_) => _actionSubmit(),
),
),
const SizedBox(width: 8),
ElevatedButton.icon(
key: Key("Search_text_button"),
onPressed: _actionSubmit,
icon: const Icon(Icons.search, size: 18),
label: Text(AppLocalizations.of(context)!.searchButtonLabel),
style: ElevatedButton.styleFrom(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(8),
),
),
),
],
),
),
);
}
void _actionSubmit() {
widget.onSearch(widget.searchController.text);
}
void _actionClear() {
widget.searchController.text = "";
widget.onSearch(widget.searchController.text);
}
}

View File

@@ -0,0 +1,60 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:provider/provider.dart';
class TextInputWidget extends StatelessWidget {
final String labelText;
final bool required;
final bool obscureText;
final bool readOnly;
final Function? onTap;
const TextInputWidget({super.key, required this.labelText, this.required = false, this.obscureText = false, this.readOnly = false, this.onTap});
@override
Widget build(BuildContext context) {
return Consumer<TextEditingController>(builder: (context, controller, child) {
return TextFormField(
readOnly: readOnly,
obscureText: obscureText,
controller: controller,
decoration: InputDecoration(
border: const UnderlineInputBorder(),
labelText: labelText,
),
validator: (String? value) => required && (value == null || value.isEmpty) ? AppLocalizations.of(context)!.textInputWidgetValidatorText : null,
onTap: () => onTap?.call(),
);
});
}
}
class TextMultiInputWidget extends StatelessWidget {
final String labelText;
final bool required;
final bool obscureText;
final bool readOnly;
final Function? onTap;
final int maxLines;
const TextMultiInputWidget({super.key, required this.labelText, this.required = false, this.obscureText = false, this.readOnly = false, this.maxLines = 6, this.onTap});
@override
Widget build(BuildContext context) {
return Consumer<TextEditingController>(builder: (context, controller, child) {
return TextFormField(
readOnly: readOnly,
minLines: 3, // Set this
maxLines: maxLines, // and this
keyboardType: TextInputType.multiline,
obscureText: obscureText,
controller: controller,
decoration: InputDecoration(
border: const UnderlineInputBorder(),
labelText: labelText,
),
validator: (String? value) => required && (value == null || value.isEmpty) ? AppLocalizations.of(context)!.textInputWidgetValidatorText : null,
onTap: () => onTap?.call(),
);
});
}
}

View File

@@ -0,0 +1,26 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
class WaitingWidget extends StatelessWidget {
const WaitingWidget({super.key});
@override
Widget build(BuildContext context) {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const SizedBox(
width: 60,
height: 60,
child: CircularProgressIndicator(),
),
Padding(
padding: const EdgeInsets.only(top: 16),
child: Text(AppLocalizations.of(context)!.waitingWidget),
),
],
),
);
}
}

View File

@@ -0,0 +1,173 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/main.dart' show logger;
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/pages/ui_utils/dialog/snackbar_utils.dart';
import 'package:fotodocumentation/utils/di_container.dart';
class DeleteDialog extends StatelessWidget {
static SnackbarUtils get _snackbarUtils => DiContainer.get();
const DeleteDialog({super.key});
static Future<void> show(BuildContext context, Future<DeleteDialogResult> Function() doDelete) async {
await _openDialog(context).then((value) async {
if (value != null && value && context.mounted) {
logger.d("Delete popup result $value");
var result = await doDelete();
if (context.mounted && result.msg.isNotEmpty) {
_snackbarUtils.showSnackbar(context, result.msg, result.warning);
}
}
});
}
static Future<bool?> _openDialog(BuildContext context) async {
return showDialog<bool>(
context: context,
barrierDismissible: false, // user must tap button!
builder: (BuildContext context) {
return const DeleteDialog(
key: Key("delete_dialog"),
);
},
);
}
@override
Widget build(BuildContext context) {
final loc = AppLocalizations.of(context)!;
return AlertDialog(
backgroundColor: Colors.white,
scrollable: false,
titlePadding: const EdgeInsets.all(16.0),
contentPadding: const EdgeInsets.fromLTRB(16.0, 0, 16.0, 16.0),
title: _titleWidget(context, loc),
content: _content(context, loc),
);
}
Widget _titleWidget(BuildContext context, AppLocalizations loc) {
return Card(
elevation: 4,
margin: EdgeInsets.zero,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
side: BorderSide(
color: Colors.grey[300]!,
width: 1,
),
),
child: Padding(
padding: const EdgeInsets.all(12.0),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Row(
children: [
Icon(
Icons.warning_amber_rounded,
size: 32,
color: Colors.orange[700],
),
const SizedBox(width: 16),
Text(
loc.deleteDialogTitle,
style: Theme.of(context).textTheme.headlineSmall?.copyWith(
fontWeight: FontWeight.bold,
),
),
],
),
IconButton(
key: const Key("close_button"),
icon: const Icon(Icons.close),
onPressed: () => Navigator.pop(context, false),
),
],
),
),
);
}
Widget _content(BuildContext context, AppLocalizations loc) {
return SizedBox(
width: 400,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Card(
elevation: 2,
margin: EdgeInsets.zero,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
side: BorderSide(
color: Colors.grey[300]!,
width: 1,
),
),
child: Padding(
padding: const EdgeInsets.all(16.0),
child: Row(
children: [
Icon(
Icons.warning_amber_rounded,
size: 48,
color: Colors.orange[700],
),
const SizedBox(width: 16),
Expanded(
child: Text(
loc.deleteDialogText,
style: Theme.of(context).textTheme.bodyLarge,
),
),
],
),
),
),
const SizedBox(height: 16),
Row(
mainAxisAlignment: MainAxisAlignment.end,
children: [
TextButton(
key: const Key("delete_dialog:cancel"),
style: TextButton.styleFrom(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(8),
),
),
onPressed: () => Navigator.pop(context, false),
child: Text(loc.deleteDialogButtonCancel),
),
const SizedBox(width: 8),
ElevatedButton.icon(
key: const Key("delete_dialog:approve"),
style: ElevatedButton.styleFrom(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
backgroundColor: Colors.red[600],
foregroundColor: Colors.white,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(8),
),
),
onPressed: () => Navigator.pop(context, true),
icon: const Icon(Icons.delete, size: 20),
label: Text(loc.deleteDialogButtonApprove),
),
],
),
],
),
);
}
}
class DeleteDialogResult {
final String msg;
final bool warning;
DeleteDialogResult({required this.msg, required this.warning});
}

View File

@@ -0,0 +1,11 @@
class DialogResult<T> {
final DialogResultType type;
final T? dto;
const DialogResult({required this.type, this.dto});
}
enum DialogResultType {
create,
add;
}

View File

@@ -0,0 +1,85 @@
import 'package:flutter/material.dart';
abstract interface class SnackbarUtils {
void showSnackbar(BuildContext context, String msg, bool warning);
void showSnackbarPopup(BuildContext context, String msg, bool warning);
}
class SnackbarUtilsImpl implements SnackbarUtils {
@override
void showSnackbar(BuildContext context, String msg, bool warning) {
var snackBar = SnackBar(
content: _contentFor(context, msg, warning),
backgroundColor: Colors.white,
behavior: SnackBarBehavior.floating,
showCloseIcon: true,
closeIconColor: Theme.of(context).colorScheme.inversePrimary,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(6),
side: BorderSide(width: 2.0, style: BorderStyle.solid, color: Theme.of(context).colorScheme.inversePrimary),
),
margin: EdgeInsets.only(bottom: MediaQuery.of(context).size.height - 130, left: MediaQuery.of(context).size.width - 400, right: 10),
);
ScaffoldMessenger.of(context).showSnackBar(snackBar);
}
@override
void showSnackbarPopup(BuildContext context, String msg, bool warning) {
var snackBar = SnackBar(
content: _contentFor(context, msg, warning),
backgroundColor: Colors.white,
behavior: SnackBarBehavior.floating,
showCloseIcon: true,
closeIconColor: Theme.of(context).colorScheme.inversePrimary,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(6),
side: BorderSide(width: 2.0, style: BorderStyle.solid, color: Theme.of(context).colorScheme.inversePrimary),
),
width: 350,
//margin: EdgeInsets.only(bottom: MediaQuery.of(context).size.height - 100, left: MediaQuery.of(context).size.width - 350, right: 10),
);
ScaffoldMessenger.of(context).showSnackBar(snackBar);
}
Widget _contentFor(BuildContext context, String msg, bool warning) {
var icon = _iconFor(context, warning);
var style = _textStyleFor(context, warning);
return Wrap(
alignment: WrapAlignment.start,
crossAxisAlignment: WrapCrossAlignment.center,
children: [
icon,
Text(
msg,
style: style,
),
],
);
}
Icon _iconFor(BuildContext context, bool warning) {
var color = _contentColor(context, warning);
return warning ? Icon(Icons.error, color: color) : Icon(Icons.check_circle_outline, color: color);
}
TextStyle _textStyleFor(BuildContext context, bool warning) {
var color = _contentColor(context, warning);
var bodyLarge = Theme.of(context).primaryTextTheme.bodyLarge!;
var style = TextStyle(
color: color,
decoration: bodyLarge.decoration,
fontFamily: bodyLarge.fontFamily,
fontSize: bodyLarge.fontSize,
fontWeight: bodyLarge.fontWeight,
letterSpacing: bodyLarge.letterSpacing,
textBaseline: bodyLarge.textBaseline);
return style;
}
Color _contentColor(BuildContext context, bool warning) {
return warning ? Theme.of(context).colorScheme.error : Theme.of(context).colorScheme.inversePrimary;
}
}

View File

@@ -0,0 +1,36 @@
import 'package:flutter/material.dart';
import 'package:pinput/pinput.dart';
abstract interface class GeneralStyle {
PinTheme get pinTheme;
ButtonStyle get elevatedButtonStyle;
ButtonStyle get roundedButtonStyle;
}
class GeneralStyleImpl implements GeneralStyle {
static final ButtonStyle _elevatedButtonStyle = ElevatedButton.styleFrom(textStyle: const TextStyle(fontSize: 20));
static final ButtonStyle _roundedButtonStyle = ElevatedButton.styleFrom(shape: const CircleBorder(), padding: const EdgeInsets.all(8));
@override
PinTheme get pinTheme => _getPinTheme();
@override
ButtonStyle get elevatedButtonStyle => _elevatedButtonStyle;
@override
ButtonStyle get roundedButtonStyle => _roundedButtonStyle;
PinTheme _getPinTheme() {
return PinTheme(
width: 56,
height: 56,
textStyle: TextStyle(fontSize: 20, fontWeight: FontWeight.w600),
decoration: BoxDecoration(
border: Border.all(color: Colors.grey),
borderRadius: BorderRadius.circular(8),
),
);
}
}

View File

@@ -0,0 +1,89 @@
import 'package:flutter/material.dart';
class HeaderButtonWrapper extends StatefulWidget {
final IconData icon;
final String tooltip;
final VoidCallback onPressed;
final Color? iconColor;
final int? badgeCount;
const HeaderButtonWrapper({
super.key,
required this.icon,
required this.tooltip,
required this.onPressed,
this.iconColor,
this.badgeCount,
});
@override
State<HeaderButtonWrapper> createState() => _HeaderButtonWrapperState();
}
class _HeaderButtonWrapperState extends State<HeaderButtonWrapper> {
bool _isHovered = false;
@override
Widget build(BuildContext context) {
return MouseRegion(
onEnter: (_) => setState(() => _isHovered = true),
onExit: (_) => setState(() => _isHovered = false),
child: Tooltip(
message: widget.tooltip,
child: Stack(
clipBehavior: Clip.none,
children: [
Container(
decoration: BoxDecoration(
color: _isHovered ? Colors.blue[50] : Colors.transparent,
borderRadius: BorderRadius.circular(12),
border: Border.all(
color: _isHovered ? Colors.blue[200]! : Colors.transparent,
width: 1,
),
),
child: IconButton(
icon: Icon(
widget.icon,
size: 24,
color: widget.iconColor ?? Colors.grey[700],
),
onPressed: widget.onPressed,
splashRadius: 24,
),
),
if (widget.badgeCount != null && widget.badgeCount! > 0)
Positioned(
right: 4,
top: 4,
child: Container(
padding: const EdgeInsets.all(4),
decoration: BoxDecoration(
color: Colors.red[600],
shape: BoxShape.circle,
border: Border.all(
color: Colors.white,
width: 2,
),
),
constraints: const BoxConstraints(
minWidth: 20,
minHeight: 20,
),
child: Text(
widget.badgeCount! > 99 ? '99+' : widget.badgeCount.toString(),
style: const TextStyle(
color: Colors.white,
fontSize: 10,
fontWeight: FontWeight.bold,
),
textAlign: TextAlign.center,
),
),
),
],
),
),
);
}
}

View File

@@ -0,0 +1,75 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
abstract interface class HeaderUtils {
Widget titleWidget(String text);
}
class HeaderUtilsImpl extends HeaderUtils {
LoginCredentials get _loginCredentials => DiContainer.get();
@override
Widget titleWidget(String text) {
return Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Icon(
Icons.school,
size: 28,
color: Colors.blue[700],
),
const SizedBox(width: 12),
Text(
text,
style: const TextStyle(
fontSize: 22,
fontWeight: FontWeight.bold,
color: Colors.black87,
),
),
],
),
const SizedBox(height: 4),
if (_loginCredentials.fullname.isNotEmpty) ...[
Padding(
padding: const EdgeInsets.only(left: 40.0),
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 4),
decoration: BoxDecoration(
color: Colors.blue[50],
borderRadius: BorderRadius.circular(12),
border: Border.all(
color: Colors.blue[200]!,
width: 1,
),
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(
Icons.person_outline,
size: 14,
color: Colors.blue[700],
),
const SizedBox(width: 6),
Text(
_loginCredentials.fullname,
style: TextStyle(
fontSize: 12,
color: Colors.blue[900],
fontWeight: FontWeight.w500,
),
),
],
),
),
),
],
],
);
}
}

View File

@@ -0,0 +1,99 @@
import 'package:flutter/material.dart';
import 'package:fotodocumentation/pages/ui_utils/header_button_wrapper.dart';
class ModernAppBar extends StatelessWidget implements PreferredSizeWidget {
final Widget title;
final List<Widget> actions;
final Widget? leading;
final bool automaticallyImplyLeading;
const ModernAppBar({
super.key,
required this.title,
this.actions = const [],
this.leading,
this.automaticallyImplyLeading = true,
});
@override
Widget build(BuildContext context) {
Widget? effectiveLeading = _effectiveLeading(context);
return Container(
decoration: BoxDecoration(
color: Colors.grey[50],
border: Border(
bottom: BorderSide(
color: Colors.grey[300]!,
width: 1,
),
),
),
child: SafeArea(
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
child: Row(
children: [
if (effectiveLeading != null) ...[
effectiveLeading,
const SizedBox(width: 16),
],
Expanded(child: title),
const SizedBox(width: 16),
...actions.map((action) => Padding(
padding: const EdgeInsets.only(left: 8.0),
child: action,
)),
],
),
),
),
);
}
Widget? _effectiveLeading(BuildContext context) {
// Determine if we should show a back button
final ScaffoldState? scaffold = Scaffold.maybeOf(context);
final ModalRoute<dynamic>? parentRoute = ModalRoute.of(context);
final bool hasDrawer = scaffold?.hasDrawer ?? false;
final bool canPop = parentRoute?.canPop ?? false;
final bool useCloseButton = parentRoute is PageRoute<dynamic> && parentRoute.fullscreenDialog;
Widget? effectiveLeading = leading;
if (effectiveLeading == null && automaticallyImplyLeading) {
if (hasDrawer) {
effectiveLeading = HeaderButtonWrapper(
icon: Icons.menu,
onPressed: () {
Scaffold.of(context).openDrawer();
},
tooltip: MaterialLocalizations.of(context).openAppDrawerTooltip,
iconColor: Colors.grey[700],
);
} else if (canPop) {
if (useCloseButton) {
effectiveLeading = HeaderButtonWrapper(
icon: Icons.close,
onPressed: () {
Navigator.of(context).pop();
},
tooltip: MaterialLocalizations.of(context).closeButtonTooltip,
iconColor: Colors.grey[700],
);
} else {
effectiveLeading = HeaderButtonWrapper(
icon: Icons.arrow_back,
onPressed: () {
Navigator.of(context).pop();
},
tooltip: MaterialLocalizations.of(context).backButtonTooltip,
iconColor: Colors.grey[700],
);
}
}
}
return effectiveLeading;
}
@override
Size get preferredSize => const Size.fromHeight(80);
}

View File

@@ -0,0 +1,17 @@
final class DateTimeUtils {
static DateTime? toDateTime(dynamic element) {
if (element == null) {
return null;
}
String text = element.toString();
int? time = int.tryParse(text);
if (time == null) {
return null;
}
return DateTime.fromMillisecondsSinceEpoch(time);
}
static int? fromDateTime(DateTime? dt) {
return dt?.millisecondsSinceEpoch;
}
}

View File

@@ -0,0 +1,38 @@
import 'package:fotodocumentation/controller/login_controller.dart';
import 'package:fotodocumentation/pages/ui_utils/dialog/snackbar_utils.dart';
import 'package:fotodocumentation/pages/ui_utils/general_style.dart';
import 'package:fotodocumentation/pages/ui_utils/header_utils.dart';
import 'package:fotodocumentation/utils/http_client_utils.dart';
import 'package:fotodocumentation/utils/jwt_token_storage.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
import 'package:fotodocumentation/utils/url_utils.dart';
class DiContainer {
static final DiContainer instance = DiContainer._privateConstructor();
DiContainer._privateConstructor();
final _container = {};
static T get<T>() {
return instance._container[T] as T;
}
void initState() {
DiContainer.instance.put(LoginCredentials, LoginCredentialsImpl());
DiContainer.instance.put(GeneralStyle, GeneralStyleImpl());
DiContainer.instance.put(JwtTokenStorage, JwtTokenStorageImpl());
DiContainer.instance.put(HttpClientUtils, HttpCLientUtilsImpl());
DiContainer.instance.put(HeaderUtils, HeaderUtilsImpl());
DiContainer.instance.put(UrlUtils, UrlUtilsImpl());
DiContainer.instance.put(SnackbarUtils, SnackbarUtilsImpl());
DiContainer.instance.put(LoginController, LoginControllerImpl());
}
void put<T>(Type key, T object) {
_container[key] = object;
}
T get2<T>() {
return _container[T] as T;
}
}

View File

@@ -0,0 +1,29 @@
import 'package:flutter/material.dart' show Colors, Color;
extension HexColor on Color {
/// String is in the format "aabbcc" or "ffaabbcc" with an optional leading "#".
static Color fromHex(String hexString) {
final buffer = StringBuffer();
if (hexString.length == 6 || hexString.length == 7) buffer.write('ff');
buffer.write(hexString.replaceFirst('#', ''));
return Color(int.parse(buffer.toString(), radix: 16));
}
}
extension RiskColor on Color {
static const Color noRisk = Colors.transparent;
static final Color lowRisk = HexColor.fromHex("#FFFF00");
static final Color mediumRisk = HexColor.fromHex("#FF9000");
static final Color highRisk = HexColor.fromHex("#FF4000");
static Color colorForRisk(int value) {
if (value == 1) {
return lowRisk;
} else if (value == 2) {
return mediumRisk;
} else if (value == 3) {
return highRisk;
}
return noRisk;
}
}

View File

@@ -0,0 +1,51 @@
// needed for web horizontal scroll behavior
import 'package:flutter/material.dart';
import 'package:fotodocumentation/main.dart';
import 'package:fotodocumentation/pages/customer/customer_widget.dart';
import 'package:fotodocumentation/pages/login/login_widget.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
import 'package:go_router/go_router.dart';
class GlobalRouter {
static final GlobalKey<NavigatorState> rootNavigatorKey = GlobalKey<NavigatorState>(debugLabel: 'root');
static final GlobalKey<NavigatorState> bottomBarNavigatorKey = GlobalKey<NavigatorState>(debugLabel: 'bottombar');
static final GlobalKey<NavigatorState> adminNavigatorKey = GlobalKey<NavigatorState>(debugLabel: 'admin');
static final GlobalKey<NavigatorState> skillEditorNavigatorKey = GlobalKey<NavigatorState>(debugLabel: 'skillEditor');
static final String pathHome = "/home";
static final String pathLogin = "/login";
static final GoRouter router = createRouter(pathHome);
static GoRouter createRouter(String initialLocation) {
return GoRouter(
navigatorKey: rootNavigatorKey,
initialLocation: initialLocation,
routes: <RouteBase>[
GoRoute(
path: "/",
redirect: (_, __) => pathHome,
),
GoRoute(
path: pathLogin,
builder: (BuildContext context, GoRouterState state) => const LoginWidget(),
),
GoRoute(
path: pathHome,
builder: (context, state) => CustomerWidget(),
),
],
redirect: (context, state) {
var uriStr = state.uri.toString();
logger.t("uri $uriStr");
LoginCredentials loginCredentials = DiContainer.get();
if (!loginCredentials.isLoggedIn) {
return pathLogin;
}
return null;
},
);
}
}

View File

@@ -0,0 +1,16 @@
class GlobalStack<T> {
final _list = <T>[];
void push(T value) => _list.add(value);
T pop() => _list.removeLast();
T peek() => _list.last;
bool get isEmpty => _list.isEmpty;
bool get isNotEmpty => _list.isNotEmpty;
@override
String toString() => _list.toString();
}

View File

@@ -0,0 +1,11 @@
import 'package:http/http.dart' as http;
import 'package:fotodocumentation/utils/http_client_factory_stub.dart';
HttpCLientFactory getHttpClientFactory() => HttpClientFactoryApp();
class HttpClientFactoryApp extends HttpCLientFactory {
@override
http.Client createHttpClient() {
return http.Client();
}
}

View File

@@ -0,0 +1,21 @@
import 'package:http/http.dart' as http;
HttpCLientFactory getHttpClientFactory() => throw UnsupportedError('Cannot create http client');
class HttpCLientFactory {
http.Client createHttpClient() {
// Check if running on the Web
/*if (kIsWeb) {
var client = http.Client();
(client as BrowserClient).withCredentials = true;
return client;
} else if (universal_io.Platform.isAndroid || universal_io.Platform.isIOS) {
// Platform-specific logic for Android and iOS
return http.Client();
} else {
throw UnsupportedError('Unsupported platform');
}*/
throw UnsupportedError('Cannot create http client');
}
}

View File

@@ -0,0 +1,14 @@
import 'package:http/browser_client.dart';
import 'package:http/http.dart' as http;
import 'package:fotodocumentation/utils/http_client_factory_stub.dart';
HttpCLientFactory getHttpClientFactory() => HttpClientFactoryWeb();
class HttpClientFactoryWeb extends HttpCLientFactory{
@override
http.Client createHttpClient() {
var client = http.Client();
(client as BrowserClient).withCredentials = true;
return client;
}
}

View File

@@ -0,0 +1,37 @@
import 'package:go_router/go_router.dart';
import 'package:http/http.dart' as http;
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
import 'package:fotodocumentation/utils/global_router.dart';
/// HTTP client that intercepts all responses and handles 401 status codes
/// by logging out the user and redirecting to the login page.
class HttpClientInterceptor extends http.BaseClient {
final http.Client _inner;
HttpClientInterceptor(this._inner);
@override
Future<http.StreamedResponse> send(http.BaseRequest request) async {
final response = await _inner.send(request);
// Check for 401 Unauthorized
if (response.statusCode == 401) {
_handle401Unauthorized();
}
return response;
}
void _handle401Unauthorized() {
// Clear login credentials
final loginCredentials = DiContainer.get<LoginCredentials>();
loginCredentials.logout();
// Navigate to login page using GoRouter
final context = GlobalRouter.rootNavigatorKey.currentContext;
if (context != null) {
context.go(GlobalRouter.pathLogin);
}
}
}

View File

@@ -0,0 +1,20 @@
import 'package:http/http.dart' as http;
import 'http_client_factory_stub.dart' if (dart.library.io) 'http_client_factory_app.dart' if (dart.library.js) 'http_client_factory_web.dart';
import 'http_client_interceptor.dart';
abstract class HttpClientUtils {
http.Client get client;
}
class HttpCLientUtilsImpl extends HttpClientUtils {
http.Client? _client;
@override
http.Client get client => _getClient();
http.Client _getClient() {
_client ??= HttpClientInterceptor(getHttpClientFactory().createHttpClient());
return _client!;
}
}

View File

@@ -0,0 +1,69 @@
abstract class JwtTokenStorage {
/// Save both access and refresh tokens
///
/// @param accessToken The short-lived access token
/// @param refreshToken The long-lived refresh token
Future<void> saveTokens(String accessToken, String refreshToken);
/// Get the stored access token
///
/// @return Access token or null if not found
Future<String?> getAccessToken();
/// Get the stored refresh token
///
/// @return Refresh token or null if not found
Future<String?> getRefreshToken();
/// Clear all stored tokens (on logout)
Future<void> clearTokens();
/// Check if tokens are stored
///
/// @return true if access token exists
Future<bool> hasTokens();
/// Update only the access token (used after refresh)
///
/// @param accessToken New access token
Future<void> updateAccessToken(String accessToken);
}
class JwtTokenStorageImpl extends JwtTokenStorage {
// Storage keys
String? _keyAccessToken;
String? _keyRefreshToken;
@override
Future<void> saveTokens(String accessToken, String refreshToken) async {
_keyAccessToken = accessToken;
_keyRefreshToken = refreshToken;
}
@override
Future<String?> getAccessToken() async {
return _keyAccessToken;
}
@override
Future<String?> getRefreshToken() async {
return _keyRefreshToken;
}
@override
Future<void> clearTokens() async {
_keyAccessToken = null;
_keyRefreshToken = null;
}
@override
Future<bool> hasTokens() async {
return _keyAccessToken != null && _keyAccessToken!.isNotEmpty;
}
@override
Future<void> updateAccessToken(String accessToken) async {
_keyAccessToken == accessToken;
}
}

View File

@@ -0,0 +1,29 @@
import 'package:flutter/material.dart';
abstract class LoginCredentials extends ChangeNotifier {
String get fullname;
bool get isLoggedIn;
void setLoggedIn(bool loggedIn);
void logout();
}
class LoginCredentialsImpl extends LoginCredentials {
bool loggedIn = false;
@override
bool get isLoggedIn => loggedIn;
@override
String get fullname => "";
@override
void setLoggedIn(bool loggedIn) {
this.loggedIn = loggedIn;
}
@override
void logout() {
loggedIn = false;
notifyListeners();
}
}

View File

@@ -0,0 +1,33 @@
// needed for web horizontal scroll behavior
import 'dart:convert' show jsonDecode;
import 'dart:ui';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:json_theme/json_theme.dart';
import 'package:fotodocumentation/main.dart' show logger;
class MyCustomScrollBehavior extends MaterialScrollBehavior {
// Override behavior methods and getters like dragDevices
@override
Set<PointerDeviceKind> get dragDevices => {
PointerDeviceKind.touch,
PointerDeviceKind.mouse,
};
}
class ThemeLoader {
static Future<ThemeData> loadTheme() async {
try {
String prefix = kDebugMode && kIsWeb ? "" : "assets/";
String url = "${prefix}theme/appainter_theme.json";
final themeStr = await rootBundle.loadString(url);
final themeJson = jsonDecode(themeStr);
return ThemeDecoder.decodeThemeData(themeJson)!;
} catch (e) {
logger.e("Failed to load theme $e", error: e);
return ThemeData.light();
}
}
}

View File

@@ -0,0 +1,12 @@
import 'package:basic_utils/basic_utils.dart' show StringUtils;
abstract interface class PasswordUtils {
String create();
}
class PasswordUtilsImpl implements PasswordUtils {
@override
String create() {
return StringUtils.generateRandomString(8, special: false);
}
}

View File

@@ -0,0 +1,15 @@
import 'package:flutter/foundation.dart' show kReleaseMode;
abstract interface class UrlUtils {
String getBaseUrl();
}
class UrlUtilsImpl extends UrlUtils {
@override
String getBaseUrl() {
if (kReleaseMode){
return "${Uri.base.origin}/api/";
}
return "http://localhost:8080/api/";
}
}

View File

@@ -0,0 +1,28 @@
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>marketing.heyday.hartmann.fotodocumentation</groupId>
<artifactId>hartmann-foto-documentation</artifactId>
<version>1.0.1</version>
<relativePath>../hartmann-foto-documentation/pom.xml</relativePath>
</parent>
<groupId>marketing.heyday.hartmann.fotodocumentation</groupId>
<artifactId>hartmann-foto-documentation-frontend</artifactId>
<version>1.0.0-SNAPSHOT</version>
<packaging>pom</packaging>
<name>fotodocumentation-frontend</name>
<build>
</build>
<properties>
<maven.javadoc.skip>true</maven.javadoc.skip>
<!-- Sonar -->
<sonar.sources>lib,pubspec.yaml</sonar.sources>
<sonar.tests>test</sonar.tests>
<!--<sonar.language>dart</sonar.language>-->
<sonar.dart.analyzer.options.override>true</sonar.dart.analyzer.options.override>
</properties>
</project>

View File

@@ -0,0 +1,122 @@
name: fotodocumentation
description: "A new Flutter project."
# The following line prevents the package from being accidentally published to
# pub.dev using `flutter pub publish`. This is preferred for private packages.
publish_to: 'none' # Remove this line if you wish to publish to pub.dev
# The following defines the version and build number for your application.
# A version number is three numbers separated by dots, like 1.2.43
# followed by an optional build number separated by a +.
# Both the version and the builder number may be overridden in flutter
# build by specifying --build-name and --build-number, respectively.
# In Android, build-name is used as versionName while build-number used as versionCode.
# Read more about Android versioning at https://developer.android.com/studio/publish/versioning
# In iOS, build-name is used as CFBundleShortVersionString while build-number is used as CFBundleVersion.
# Read more about iOS versioning at
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
# In Windows, build-name is used as the major, minor, and patch parts
# of the product and file versions while build-number is used as the build suffix.
version: 1.0.0+1
environment:
sdk: '>=3.3.0 <4.0.0'
# Dependencies specify other packages that your package needs in order to work.
# To automatically upgrade your package dependencies to the latest versions
# consider running `flutter pub upgrade --major-versions`. Alternatively,
# dependencies can be manually updated by changing the version numbers below to
# the latest version available on pub.dev. To see which dependencies have newer
# versions available, run `flutter pub outdated`.
dependencies:
flutter:
sdk: flutter
# The following adds the Cupertino Icons font to your application.
# Use with the CupertinoIcons class for iOS style icons.
cupertino_icons: ^1.0.6
http: ^1.2.2
logger: ^2.4.0
provider: ^6.1.2
json_theme: ^9.0.1+2
flutter_localizations:
sdk: flutter
intl: any
universal_io: ^2.2.2
basic_utils: ^5.7.0
file_picker: ^10.3.3
http_parser: ^4.1.2
go_router: ^16.2.4
flutter_multi_select_items: ^0.4.3
fl_chart: ^0.69.0
pinput: ^6.0.1
# TODO: check if we can remove this in future contains a dependency override for Ticket
# https://github.com/peiffer-innovations/json_theme/issues/282
# https://github.com/peiffer-innovations/json_theme/issues/287
# https://github.com/peiffer-innovations/json_theme/pull/288
dependency_overrides:
json_theme:
git:
url: https://github.com/wardbj93/json_theme
path: packages/json_theme
dev_dependencies:
flutter_test:
sdk: flutter
# The "flutter_lints" package below contains a set of recommended lints to
# encourage good coding practices. The lint set provided by the package is
# activated in the `analysis_options.yaml` file located at the root of your
# package. See that file for information about deactivating specific lint
# rules and activating additional ones.
flutter_lints: ^6.0.0
mockito: ^5.4.5
build_runner: ^2.4.14
# For information on the generic Dart part of this file, see the
# following page: https://dart.dev/tools/pub/pubspec
flutter_intl:
enabled: true
# The following section is specific to Flutter packages.
flutter:
generate: true # Add this line for localization
# The following line ensures that the Material Icons font is
# included with your application, so that you can use the icons in
# the material Icons class.
uses-material-design: true
# To add assets to your application, add an assets section, like this:
assets:
- assets/theme/appainter_theme.json
# An image asset can refer to one or more resolution-specific "variants", see
# https://flutter.dev/assets-and-images/#resolution-aware
# For details regarding adding assets from package dependencies, see
# https://flutter.dev/assets-and-images/#from-packages
# To add custom fonts to your application, add a fonts section here,
# in this "flutter" section. Each entry in this list should have a
# "family" key with the font family name, and a "fonts" key with a
# list giving the asset and other descriptors for the font. For
# example:
# fonts:
# - family: Schyler
# fonts:
# - asset: fonts/Schyler-Regular.ttf
# - asset: fonts/Schyler-Italic.ttf
# style: italic
# - family: Trajan Pro
# fonts:
# - asset: fonts/TrajanPro.ttf
# - asset: fonts/TrajanPro_Bold.ttf
# weight: 700
#
# For details regarding fonts from package dependencies,
# see https://flutter.dev/custom-fonts/#from-packages

View File

@@ -0,0 +1,12 @@
import 'package:http/http.dart' as http;
import 'package:fotodocumentation/utils/http_client_utils.dart';
class TestHttpCLientUtilsImpl extends HttpClientUtils {
http.Client testClient;
TestHttpCLientUtilsImpl(this.testClient);
@override
http.Client get client => testClient;
}

View File

@@ -0,0 +1,67 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:http/http.dart' as http;
import 'package:mockito/annotations.dart';
import 'package:fotodocumentation/l10n/app_localizations.dart';
import 'package:fotodocumentation/pages/ui_utils/dialog/snackbar_utils.dart';
import 'package:fotodocumentation/pages/ui_utils/header_utils.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
import 'package:fotodocumentation/utils/password_utils.dart';
import 'package:fotodocumentation/utils/jwt_token_storage.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:fotodocumentation/utils/global_router.dart';
import 'test_utils.mocks.dart';
void setScreenSize(WidgetTester tester, int width, int height) {
final dpi = tester.view.devicePixelRatio;
tester.view.physicalSize = Size(width * dpi, height * dpi);
}
MockLoginCredentials getDefaultLoginCredentials() {
var mockLoginCredentials = MockLoginCredentials();
return mockLoginCredentials;
}
Future<void> pumpApp(WidgetTester tester, Widget widget) async {
await tester.pumpWidget(MaterialApp(
title: 'App',
localizationsDelegates: [
AppLocalizations.delegate,
GlobalMaterialLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
],
supportedLocales: [
Locale('de'),
],
home: Scaffold(body: widget)));
await tester.pump();
}
Future<void> pumpAppConfig(WidgetTester tester, String initialLocation) async {
await tester.pumpWidget(MaterialApp.router(
title: 'App',
localizationsDelegates: [
AppLocalizations.delegate,
GlobalMaterialLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
],
supportedLocales: [
Locale('de'),
],
routerConfig: GlobalRouter.createRouter(initialLocation)));
await tester.pump();
}
// dart run build_runner build
@GenerateMocks([
LoginCredentials,
HeaderUtils,
PasswordUtils,
SnackbarUtils,
JwtTokenStorage,
http.Client
])
void main() {}

View File

@@ -0,0 +1,542 @@
// Mocks generated by Mockito 5.4.6 from annotations
// in fotodocumentation/test/testing/test_utils.dart.
// Do not manually edit this file.
// ignore_for_file: no_leading_underscores_for_library_prefixes
import 'dart:async' as _i11;
import 'dart:convert' as _i12;
import 'dart:typed_data' as _i13;
import 'dart:ui' as _i6;
import 'package:flutter/material.dart' as _i2;
import 'package:fotodocumentation/pages/ui_utils/dialog/snackbar_utils.dart'
as _i9;
import 'package:fotodocumentation/pages/ui_utils/header_utils.dart' as _i7;
import 'package:fotodocumentation/utils/jwt_token_storage.dart' as _i10;
import 'package:fotodocumentation/utils/login_credentials.dart' as _i4;
import 'package:fotodocumentation/utils/password_utils.dart' as _i8;
import 'package:http/http.dart' as _i3;
import 'package:mockito/mockito.dart' as _i1;
import 'package:mockito/src/dummies.dart' as _i5;
// ignore_for_file: type=lint
// ignore_for_file: avoid_redundant_argument_values
// ignore_for_file: avoid_setters_without_getters
// ignore_for_file: comment_references
// ignore_for_file: deprecated_member_use
// ignore_for_file: deprecated_member_use_from_same_package
// ignore_for_file: implementation_imports
// ignore_for_file: invalid_use_of_visible_for_testing_member
// ignore_for_file: must_be_immutable
// ignore_for_file: prefer_const_constructors
// ignore_for_file: unnecessary_parenthesis
// ignore_for_file: camel_case_types
// ignore_for_file: subtype_of_sealed_class
// ignore_for_file: invalid_use_of_internal_member
class _FakeWidget_0 extends _i1.SmartFake implements _i2.Widget {
_FakeWidget_0(
Object parent,
Invocation parentInvocation,
) : super(
parent,
parentInvocation,
);
@override
String toString({_i2.DiagnosticLevel? minLevel = _i2.DiagnosticLevel.info}) =>
super.toString();
}
class _FakeResponse_1 extends _i1.SmartFake implements _i3.Response {
_FakeResponse_1(
Object parent,
Invocation parentInvocation,
) : super(
parent,
parentInvocation,
);
}
class _FakeStreamedResponse_2 extends _i1.SmartFake
implements _i3.StreamedResponse {
_FakeStreamedResponse_2(
Object parent,
Invocation parentInvocation,
) : super(
parent,
parentInvocation,
);
}
/// A class which mocks [LoginCredentials].
///
/// See the documentation for Mockito's code generation for more information.
class MockLoginCredentials extends _i1.Mock implements _i4.LoginCredentials {
MockLoginCredentials() {
_i1.throwOnMissingStub(this);
}
@override
String get fullname => (super.noSuchMethod(
Invocation.getter(#fullname),
returnValue: _i5.dummyValue<String>(
this,
Invocation.getter(#fullname),
),
) as String);
@override
bool get hasListeners => (super.noSuchMethod(
Invocation.getter(#hasListeners),
returnValue: false,
) as bool);
@override
void logout() => super.noSuchMethod(
Invocation.method(
#logout,
[],
),
returnValueForMissingStub: null,
);
@override
void addListener(_i6.VoidCallback? listener) => super.noSuchMethod(
Invocation.method(
#addListener,
[listener],
),
returnValueForMissingStub: null,
);
@override
void removeListener(_i6.VoidCallback? listener) => super.noSuchMethod(
Invocation.method(
#removeListener,
[listener],
),
returnValueForMissingStub: null,
);
@override
void dispose() => super.noSuchMethod(
Invocation.method(
#dispose,
[],
),
returnValueForMissingStub: null,
);
@override
void notifyListeners() => super.noSuchMethod(
Invocation.method(
#notifyListeners,
[],
),
returnValueForMissingStub: null,
);
}
/// A class which mocks [HeaderUtils].
///
/// See the documentation for Mockito's code generation for more information.
class MockHeaderUtils extends _i1.Mock implements _i7.HeaderUtils {
MockHeaderUtils() {
_i1.throwOnMissingStub(this);
}
@override
_i2.Widget titleWidget(String? text) => (super.noSuchMethod(
Invocation.method(
#titleWidget,
[text],
),
returnValue: _FakeWidget_0(
this,
Invocation.method(
#titleWidget,
[text],
),
),
) as _i2.Widget);
}
/// A class which mocks [PasswordUtils].
///
/// See the documentation for Mockito's code generation for more information.
class MockPasswordUtils extends _i1.Mock implements _i8.PasswordUtils {
MockPasswordUtils() {
_i1.throwOnMissingStub(this);
}
@override
String create() => (super.noSuchMethod(
Invocation.method(
#create,
[],
),
returnValue: _i5.dummyValue<String>(
this,
Invocation.method(
#create,
[],
),
),
) as String);
}
/// A class which mocks [SnackbarUtils].
///
/// See the documentation for Mockito's code generation for more information.
class MockSnackbarUtils extends _i1.Mock implements _i9.SnackbarUtils {
MockSnackbarUtils() {
_i1.throwOnMissingStub(this);
}
@override
void showSnackbar(
_i2.BuildContext? context,
String? msg,
bool? warning,
) =>
super.noSuchMethod(
Invocation.method(
#showSnackbar,
[
context,
msg,
warning,
],
),
returnValueForMissingStub: null,
);
@override
void showSnackbarPopup(
_i2.BuildContext? context,
String? msg,
bool? warning,
) =>
super.noSuchMethod(
Invocation.method(
#showSnackbarPopup,
[
context,
msg,
warning,
],
),
returnValueForMissingStub: null,
);
}
/// A class which mocks [JwtTokenStorage].
///
/// See the documentation for Mockito's code generation for more information.
class MockJwtTokenStorage extends _i1.Mock implements _i10.JwtTokenStorage {
MockJwtTokenStorage() {
_i1.throwOnMissingStub(this);
}
@override
_i11.Future<void> saveTokens(
String? accessToken,
String? refreshToken,
) =>
(super.noSuchMethod(
Invocation.method(
#saveTokens,
[
accessToken,
refreshToken,
],
),
returnValue: _i11.Future<void>.value(),
returnValueForMissingStub: _i11.Future<void>.value(),
) as _i11.Future<void>);
@override
_i11.Future<String?> getAccessToken() => (super.noSuchMethod(
Invocation.method(
#getAccessToken,
[],
),
returnValue: _i11.Future<String?>.value(),
) as _i11.Future<String?>);
@override
_i11.Future<String?> getRefreshToken() => (super.noSuchMethod(
Invocation.method(
#getRefreshToken,
[],
),
returnValue: _i11.Future<String?>.value(),
) as _i11.Future<String?>);
@override
_i11.Future<void> clearTokens() => (super.noSuchMethod(
Invocation.method(
#clearTokens,
[],
),
returnValue: _i11.Future<void>.value(),
returnValueForMissingStub: _i11.Future<void>.value(),
) as _i11.Future<void>);
@override
_i11.Future<bool> hasTokens() => (super.noSuchMethod(
Invocation.method(
#hasTokens,
[],
),
returnValue: _i11.Future<bool>.value(false),
) as _i11.Future<bool>);
@override
_i11.Future<void> updateAccessToken(String? accessToken) =>
(super.noSuchMethod(
Invocation.method(
#updateAccessToken,
[accessToken],
),
returnValue: _i11.Future<void>.value(),
returnValueForMissingStub: _i11.Future<void>.value(),
) as _i11.Future<void>);
}
/// A class which mocks [Client].
///
/// See the documentation for Mockito's code generation for more information.
class MockClient extends _i1.Mock implements _i3.Client {
MockClient() {
_i1.throwOnMissingStub(this);
}
@override
_i11.Future<_i3.Response> head(
Uri? url, {
Map<String, String>? headers,
}) =>
(super.noSuchMethod(
Invocation.method(
#head,
[url],
{#headers: headers},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#head,
[url],
{#headers: headers},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<_i3.Response> get(
Uri? url, {
Map<String, String>? headers,
}) =>
(super.noSuchMethod(
Invocation.method(
#get,
[url],
{#headers: headers},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#get,
[url],
{#headers: headers},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<_i3.Response> post(
Uri? url, {
Map<String, String>? headers,
Object? body,
_i12.Encoding? encoding,
}) =>
(super.noSuchMethod(
Invocation.method(
#post,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#post,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<_i3.Response> put(
Uri? url, {
Map<String, String>? headers,
Object? body,
_i12.Encoding? encoding,
}) =>
(super.noSuchMethod(
Invocation.method(
#put,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#put,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<_i3.Response> patch(
Uri? url, {
Map<String, String>? headers,
Object? body,
_i12.Encoding? encoding,
}) =>
(super.noSuchMethod(
Invocation.method(
#patch,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#patch,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<_i3.Response> delete(
Uri? url, {
Map<String, String>? headers,
Object? body,
_i12.Encoding? encoding,
}) =>
(super.noSuchMethod(
Invocation.method(
#delete,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
returnValue: _i11.Future<_i3.Response>.value(_FakeResponse_1(
this,
Invocation.method(
#delete,
[url],
{
#headers: headers,
#body: body,
#encoding: encoding,
},
),
)),
) as _i11.Future<_i3.Response>);
@override
_i11.Future<String> read(
Uri? url, {
Map<String, String>? headers,
}) =>
(super.noSuchMethod(
Invocation.method(
#read,
[url],
{#headers: headers},
),
returnValue: _i11.Future<String>.value(_i5.dummyValue<String>(
this,
Invocation.method(
#read,
[url],
{#headers: headers},
),
)),
) as _i11.Future<String>);
@override
_i11.Future<_i13.Uint8List> readBytes(
Uri? url, {
Map<String, String>? headers,
}) =>
(super.noSuchMethod(
Invocation.method(
#readBytes,
[url],
{#headers: headers},
),
returnValue: _i11.Future<_i13.Uint8List>.value(_i13.Uint8List(0)),
) as _i11.Future<_i13.Uint8List>);
@override
_i11.Future<_i3.StreamedResponse> send(_i3.BaseRequest? request) =>
(super.noSuchMethod(
Invocation.method(
#send,
[request],
),
returnValue:
_i11.Future<_i3.StreamedResponse>.value(_FakeStreamedResponse_2(
this,
Invocation.method(
#send,
[request],
),
)),
) as _i11.Future<_i3.StreamedResponse>);
@override
void close() => super.noSuchMethod(
Invocation.method(
#close,
[],
),
returnValueForMissingStub: null,
);
}

View File

@@ -0,0 +1,101 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:fotodocumentation/utils/date_time_utils.dart';
void main() {
group('DateTimeUtils', () {
group('toDateTime', () {
test('returns null when element is null', () {
expect(DateTimeUtils.toDateTime(null), isNull);
});
test('returns null when element cannot be parsed to int', () {
expect(DateTimeUtils.toDateTime('invalid'), isNull);
expect(DateTimeUtils.toDateTime('abc123'), isNull);
expect(DateTimeUtils.toDateTime('12.34'), isNull);
});
test('converts valid milliseconds string to DateTime', () {
const milliseconds = 1640995200000; // 2022-01-01 00:00:00 UTC
final result = DateTimeUtils.toDateTime(milliseconds.toString());
expect(result, isNotNull);
expect(result!.millisecondsSinceEpoch, equals(milliseconds));
});
test('converts valid milliseconds int to DateTime', () {
const milliseconds = 1640995200000; // 2022-01-01 00:00:00 UTC
final result = DateTimeUtils.toDateTime(milliseconds);
expect(result, isNotNull);
expect(result!.millisecondsSinceEpoch, equals(milliseconds));
});
test('handles zero milliseconds', () {
final result = DateTimeUtils.toDateTime(0);
expect(result, isNotNull);
expect(result!.millisecondsSinceEpoch, equals(0));
});
test('handles negative milliseconds', () {
const milliseconds = -1000;
final result = DateTimeUtils.toDateTime(milliseconds);
expect(result, isNotNull);
expect(result!.millisecondsSinceEpoch, equals(milliseconds));
});
});
group('fromDateTime', () {
test('returns null when DateTime is null', () {
expect(DateTimeUtils.fromDateTime(null), isNull);
});
test('converts DateTime to milliseconds since epoch', () {
const milliseconds = 1640995200000; // 2022-01-01 00:00:00 UTC
final dateTime = DateTime.fromMillisecondsSinceEpoch(milliseconds);
final result = DateTimeUtils.fromDateTime(dateTime);
expect(result, equals(milliseconds));
});
test('handles epoch time (zero)', () {
final dateTime = DateTime.fromMillisecondsSinceEpoch(0);
final result = DateTimeUtils.fromDateTime(dateTime);
expect(result, equals(0));
});
test('handles dates before epoch (negative milliseconds)', () {
const milliseconds = -1000;
final dateTime = DateTime.fromMillisecondsSinceEpoch(milliseconds);
final result = DateTimeUtils.fromDateTime(dateTime);
expect(result, equals(milliseconds));
});
});
group('round-trip conversion', () {
test('toDateTime and fromDateTime are inverse operations', () {
const originalMilliseconds = 1640995200000;
final dateTime = DateTimeUtils.toDateTime(originalMilliseconds);
final convertedBack = DateTimeUtils.fromDateTime(dateTime);
expect(convertedBack, equals(originalMilliseconds));
});
test('fromDateTime and toDateTime are inverse operations', () {
final originalDateTime = DateTime.now();
final milliseconds = DateTimeUtils.fromDateTime(originalDateTime);
final convertedBack = DateTimeUtils.toDateTime(milliseconds);
expect(convertedBack?.millisecondsSinceEpoch, equals(originalDateTime.millisecondsSinceEpoch));
});
});
});
}

View File

@@ -0,0 +1,133 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:http/http.dart' as http;
import 'package:mockito/mockito.dart';
import 'package:fotodocumentation/utils/di_container.dart';
import 'package:fotodocumentation/utils/http_client_interceptor.dart';
import 'package:fotodocumentation/utils/login_credentials.dart';
import '../testing/test_utils.mocks.dart';
void main() {
TestWidgetsFlutterBinding.ensureInitialized();
group('HttpClientInterceptor', () {
late MockClient mockInnerClient;
late MockLoginCredentials mockLoginCredentials;
late HttpClientInterceptor interceptingClient;
setUp(() {
DiContainer.instance.initState();
mockInnerClient = MockClient();
mockLoginCredentials = MockLoginCredentials();
DiContainer.instance.put(LoginCredentials, mockLoginCredentials);
interceptingClient = HttpClientInterceptor(mockInnerClient);
});
test('calls logout on 401 response', () async {
// Mock 401 response
final request = http.Request('GET', Uri.parse('http://example.com/api/test'));
final streamedResponse = http.StreamedResponse(
Stream.value([]),
401,
headers: {'content-type': 'application/json'},
);
when(mockInnerClient.send(any)).thenAnswer((_) async => streamedResponse);
// Execute request
final response = await interceptingClient.send(request);
// Verify logout was called
verify(mockLoginCredentials.logout()).called(1);
// Verify response is still returned
expect(response.statusCode, 401);
});
test('does not interfere with successful responses', () async {
// Mock 200 response
final request = http.Request('GET', Uri.parse('http://example.com/api/test'));
final streamedResponse = http.StreamedResponse(
Stream.value([]),
200,
headers: {'content-type': 'application/json'},
);
when(mockInnerClient.send(any)).thenAnswer((_) async => streamedResponse);
// Execute request
final response = await interceptingClient.send(request);
// Verify response is passed through
expect(response.statusCode, 200);
// Verify logout was NOT called
verifyNever(mockLoginCredentials.logout());
});
test('does not interfere with 404 responses', () async {
// Mock 404 response
final request = http.Request('GET', Uri.parse('http://example.com/api/test'));
final streamedResponse = http.StreamedResponse(
Stream.value([]),
404,
headers: {'content-type': 'application/json'},
);
when(mockInnerClient.send(any)).thenAnswer((_) async => streamedResponse);
// Execute request
final response = await interceptingClient.send(request);
// Verify response is passed through
expect(response.statusCode, 404);
// Verify logout was NOT called
verifyNever(mockLoginCredentials.logout());
});
test('does not interfere with 500 responses', () async {
// Mock 500 response
final request = http.Request('GET', Uri.parse('http://example.com/api/test'));
final streamedResponse = http.StreamedResponse(
Stream.value([]),
500,
headers: {'content-type': 'application/json'},
);
when(mockInnerClient.send(any)).thenAnswer((_) async => streamedResponse);
// Execute request
final response = await interceptingClient.send(request);
// Verify response is passed through
expect(response.statusCode, 500);
// Verify logout was NOT called
verifyNever(mockLoginCredentials.logout());
});
test('handles multiple 401 responses gracefully', () async {
// Mock 401 response
final request1 = http.Request('GET', Uri.parse('http://example.com/api/test1'));
final request2 = http.Request('GET', Uri.parse('http://example.com/api/test2'));
final streamedResponse = http.StreamedResponse(
Stream.value([]),
401,
headers: {'content-type': 'application/json'},
);
when(mockInnerClient.send(any)).thenAnswer((_) async => streamedResponse);
// Execute multiple requests
await interceptingClient.send(request1);
await interceptingClient.send(request2);
// Verify logout was called for each 401
verify(mockLoginCredentials.logout()).called(2);
});
});
}

View File

@@ -0,0 +1,182 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:fotodocumentation/utils/jwt_token_storage.dart';
void main() {
group('JwtTokenStorage Tests', () {
late JwtTokenStorage storage;
setUp(() {
storage = JwtTokenStorageImpl();
});
test('initially has no tokens', () async {
// Verify initial state is empty
expect(await storage.getAccessToken(), isNull);
expect(await storage.getRefreshToken(), isNull);
expect(await storage.hasTokens(), isFalse);
});
test('saveTokens stores both access and refresh tokens', () async {
const accessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.access';
const refreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.refresh';
await storage.saveTokens(accessToken, refreshToken);
expect(await storage.getAccessToken(), equals(accessToken));
expect(await storage.getRefreshToken(), equals(refreshToken));
expect(await storage.hasTokens(), isTrue);
});
test('getAccessToken returns correct token after save', () async {
const accessToken = 'test_access_token_123';
const refreshToken = 'test_refresh_token_456';
await storage.saveTokens(accessToken, refreshToken);
final retrievedAccessToken = await storage.getAccessToken();
expect(retrievedAccessToken, equals(accessToken));
});
test('getRefreshToken returns correct token after save', () async {
const accessToken = 'test_access_token_123';
const refreshToken = 'test_refresh_token_456';
await storage.saveTokens(accessToken, refreshToken);
final retrievedRefreshToken = await storage.getRefreshToken();
expect(retrievedRefreshToken, equals(refreshToken));
});
test('clearTokens removes all stored tokens', () async {
const accessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.access';
const refreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.refresh';
// First save tokens
await storage.saveTokens(accessToken, refreshToken);
expect(await storage.hasTokens(), isTrue);
// Then clear them
await storage.clearTokens();
expect(await storage.getAccessToken(), isNull);
expect(await storage.getRefreshToken(), isNull);
expect(await storage.hasTokens(), isFalse);
});
test('hasTokens returns true when access token exists', () async {
const accessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.access';
const refreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.refresh';
expect(await storage.hasTokens(), isFalse);
await storage.saveTokens(accessToken, refreshToken);
expect(await storage.hasTokens(), isTrue);
});
test('hasTokens returns false when access token is empty string', () async {
const accessToken = '';
const refreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.refresh';
await storage.saveTokens(accessToken, refreshToken);
expect(await storage.hasTokens(), isFalse);
});
test('updateAccessToken updates only the access token', () async {
const initialAccessToken = 'initial_access_token';
const initialRefreshToken = 'initial_refresh_token';
const newAccessToken = 'new_access_token';
// Save initial tokens
await storage.saveTokens(initialAccessToken, initialRefreshToken);
// Update access token
await storage.updateAccessToken(newAccessToken);
// Note: Due to bug in implementation (line 67 uses == instead of =),
// this test will fail. The access token won't actually be updated.
// Uncomment below when bug is fixed:
// expect(await storage.getAccessToken(), equals(newAccessToken));
// expect(await storage.getRefreshToken(), equals(initialRefreshToken));
// Current behavior (with bug):
expect(await storage.getAccessToken(), equals(initialAccessToken));
expect(await storage.getRefreshToken(), equals(initialRefreshToken));
});
test('saveTokens can overwrite existing tokens', () async {
const firstAccessToken = 'first_access_token';
const firstRefreshToken = 'first_refresh_token';
const secondAccessToken = 'second_access_token';
const secondRefreshToken = 'second_refresh_token';
// Save first set of tokens
await storage.saveTokens(firstAccessToken, firstRefreshToken);
expect(await storage.getAccessToken(), equals(firstAccessToken));
expect(await storage.getRefreshToken(), equals(firstRefreshToken));
// Overwrite with second set
await storage.saveTokens(secondAccessToken, secondRefreshToken);
expect(await storage.getAccessToken(), equals(secondAccessToken));
expect(await storage.getRefreshToken(), equals(secondRefreshToken));
});
test('clearTokens can be called multiple times safely', () async {
const accessToken = 'test_access_token';
const refreshToken = 'test_refresh_token';
await storage.saveTokens(accessToken, refreshToken);
await storage.clearTokens();
await storage.clearTokens(); // Call again
expect(await storage.getAccessToken(), isNull);
expect(await storage.getRefreshToken(), isNull);
expect(await storage.hasTokens(), isFalse);
});
test('handles long JWT tokens correctly', () async {
const longAccessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.'
'eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.'
'SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c';
const longRefreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.'
'eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyLCJleHAiOjE1MTYyNDI2MjJ9.'
'Ks_BdfH4CWilyzLNk8S2gShdsGuhkle-VsNBJJxulJc';
await storage.saveTokens(longAccessToken, longRefreshToken);
expect(await storage.getAccessToken(), equals(longAccessToken));
expect(await storage.getRefreshToken(), equals(longRefreshToken));
expect(await storage.hasTokens(), isTrue);
});
test('typical authentication flow', () async {
// 1. Initial state - no tokens
expect(await storage.hasTokens(), isFalse);
// 2. User logs in - tokens are saved
const accessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.access';
const refreshToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.refresh';
await storage.saveTokens(accessToken, refreshToken);
expect(await storage.hasTokens(), isTrue);
expect(await storage.getAccessToken(), equals(accessToken));
expect(await storage.getRefreshToken(), equals(refreshToken));
// 3. Access token expires, refresh with new access token
const newAccessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.new_access';
await storage.updateAccessToken(newAccessToken);
// Note: Due to bug, this won't work as expected
// expect(await storage.getAccessToken(), equals(newAccessToken));
// expect(await storage.getRefreshToken(), equals(refreshToken));
// 4. User logs out - tokens are cleared
await storage.clearTokens();
expect(await storage.hasTokens(), isFalse);
expect(await storage.getAccessToken(), isNull);
expect(await storage.getRefreshToken(), isNull);
});
});
}

View File

@@ -0,0 +1,12 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:fotodocumentation/utils/url_utils.dart';
void main() {
test('Expect the localhost url for debug testing', () {
final urlUtils = UrlUtilsImpl();
String url = urlUtils.getBaseUrl();
expect(url, "http://localhost:8080/api/");
});
}

View File

@@ -0,0 +1,132 @@
#!/usr/bin/env dart
import 'dart:convert';
import 'dart:io';
import 'package:logger/web.dart';
/// Converts Flutter JSON test results to JUnit XML format
/// Usage: dart test_runner.dart [test_results.json] [output.xml]
void main(List<String> arguments) {
var logger = Logger(
printer: PrettyPrinter(methodCount: 2, errorMethodCount: 8, colors: true, printEmojis: true, dateTimeFormat: DateTimeFormat.onlyTimeAndSinceStart),
);
if (arguments.length < 2) {
logger.i('Usage: dart test_runner.dart <json_file> <xml_output>');
exit(1);
}
final jsonFile = arguments[0];
final xmlFile = arguments[1];
try {
final jsonContent = File(jsonFile).readAsStringSync();
final lines = jsonContent.trim().split('\n');
final tests = <TestResult>[];
int totalTests = 0;
int failures = 0;
int skipped = 0;
double duration = 0.0;
// Parse JSON lines from flutter test output
for (final line in lines) {
if (line.trim().isEmpty) continue;
try {
final data = jsonDecode(line) as Map<String, dynamic>;
final type = data['type'] as String?;
if (type == 'testStart') {
final test = data['test'] as Map<String, dynamic>;
final name = test['name'] as String;
final id = test['id'] as int;
tests.add(TestResult(id: id, name: name));
totalTests++;
} else if (type == 'testDone') {
final testId = data['testID'] as int;
final result = data['result'] as String;
final time = data['time'] as int? ?? 0;
final error = data['error'] as String?;
final stackTrace = data['stackTrace'] as String?;
duration += time / 1000.0; // Convert to seconds
final testIndex = tests.indexWhere((t) => t.id == testId);
if (testIndex != -1) {
tests[testIndex].result = result;
tests[testIndex].duration = time / 1000.0;
tests[testIndex].error = error;
tests[testIndex].stackTrace = stackTrace;
if (result == 'error' || result == 'failure') {
failures++;
} else if (result == 'skip') {
skipped++;
}
}
}
} catch (e) {
// Skip malformed JSON lines
logger.i('Warning: Could not parse line: $line');
}
}
// Generate JUnit XML
final xml = generateJUnitXML(tests, totalTests, failures, skipped, duration);
File(xmlFile).writeAsStringSync(xml);
logger.i('Converted ${tests.length} test results to JUnit XML: $xmlFile');
// Exit with error code if there were failures
if (failures > 0) {
exit(1);
}
} catch (e) {
logger.e('Error: $e');
exit(1);
}
}
class TestResult {
final int id;
final String name;
String result = 'pending';
double duration = 0.0;
String? error;
String? stackTrace;
TestResult({required this.id, required this.name});
}
String generateJUnitXML(List<TestResult> tests, int total, int failures, int skipped, double duration) {
final buffer = StringBuffer();
buffer.writeln('<?xml version="1.0" encoding="UTF-8"?>');
buffer.writeln('<testsuite name="Flutter Tests" tests="$total" failures="$failures" skipped="$skipped" time="$duration">');
for (final test in tests) {
final escapedName = _escapeXml(test.name);
buffer.writeln(' <testcase name="$escapedName" time="${test.duration}">');
if (test.result == 'error' || test.result == 'failure') {
final escapedError = _escapeXml(test.error ?? 'Test failed');
final escapedStackTrace = _escapeXml(test.stackTrace ?? '');
buffer.writeln(' <failure message="$escapedError">');
buffer.writeln(' <![CDATA[$escapedStackTrace]]>');
buffer.writeln(' </failure>');
} else if (test.result == 'skip') {
buffer.writeln(' <skipped/>');
}
buffer.writeln(' </testcase>');
}
buffer.writeln('</testsuite>');
return buffer.toString();
}
String _escapeXml(String text) {
return text.replaceAll('&', '&amp;').replaceAll('<', '&lt;').replaceAll('>', '&gt;').replaceAll('"', '&quot;').replaceAll("'", '&apos;');
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 917 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 20 KiB

View File

@@ -0,0 +1,38 @@
<!DOCTYPE html>
<html>
<head>
<!--
If you are serving your web app in a path other than the root, change the
href value below to reflect the base path you are serving from.
The path provided below has to start and end with a slash "/" in order for
it to work correctly.
For more details:
* https://developer.mozilla.org/en-US/docs/Web/HTML/Element/base
This is a placeholder for base href that will be replaced by the value of
the `--base-href` argument provided to `flutter build`.
-->
<base href="$FLUTTER_BASE_HREF">
<meta charset="UTF-8">
<meta content="IE=Edge" http-equiv="X-UA-Compatible">
<meta name="description" content="A new Flutter project.">
<!-- iOS meta tags & icons -->
<meta name="mobile-web-app-capable" content="yes">
<meta name="apple-mobile-web-app-status-bar-style" content="black">
<meta name="apple-mobile-web-app-title" content="fotodocumentation">
<link rel="apple-touch-icon" href="icons/Icon-192.png">
<!-- Favicon -->
<link rel="icon" type="image/png" href="favicon.png"/>
<title>fotodocumentation</title>
<link rel="manifest" href="manifest.json">
</head>
<body>
<script src="flutter_bootstrap.js" async></script>
</body>
</html>

View File

@@ -0,0 +1,35 @@
{
"name": "fotodocumentation",
"short_name": "fotodocumentation",
"start_url": ".",
"display": "standalone",
"background_color": "#0175C2",
"theme_color": "#0175C2",
"description": "A new Flutter project.",
"orientation": "portrait-primary",
"prefer_related_applications": false,
"icons": [
{
"src": "icons/Icon-192.png",
"sizes": "192x192",
"type": "image/png"
},
{
"src": "icons/Icon-512.png",
"sizes": "512x512",
"type": "image/png"
},
{
"src": "icons/Icon-maskable-192.png",
"sizes": "192x192",
"type": "image/png",
"purpose": "maskable"
},
{
"src": "icons/Icon-maskable-512.png",
"sizes": "512x512",
"type": "image/png",
"purpose": "maskable"
}
]
}