Initial commit: TimeReg Flutter app med Firebase backend

- Opprettet Flutter-prosjekt med alle nødvendige avhengigheter
- Implementert datamodeller (User, TimeRegistration, TariffProfile, Deviation, AuditLog)
- Implementert tjenester (AuthService, TimeService)
- Implementert Riverpod providers for state management
- Opprettet autentiseringsskjermer (login, signup, reset password)
- Opprettet hjemmeskjerm med timer-funksjonalitet
- Opprettet placeholder-skjermer for historikk, rapporter og profil
- Lagt til norsk dokumentasjon i README
This commit is contained in:
steinhelge
2025-11-24 20:52:27 +01:00
commit c829f78984
148 changed files with 8462 additions and 0 deletions
+82
View File
@@ -0,0 +1,82 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:firebase_auth/firebase_auth.dart';
import '../services/auth_service.dart';
import '../models/user_model.dart';
// Auth service provider
final authServiceProvider = Provider<AuthService>((ref) => AuthService());
// Auth state provider - lytter til Firebase Auth state changes
final authStateProvider = StreamProvider<User?>((ref) {
final authService = ref.watch(authServiceProvider);
return authService.authStateChanges;
});
// Current user provider
final currentUserProvider = Provider<User?>((ref) {
final authState = ref.watch(authStateProvider);
return authState.when(
data: (user) => user,
loading: () => null,
error: (_, __) => null,
);
});
// User data provider - henter brukerdata fra Firestore
final userDataProvider = StreamProvider<UserModel?>((ref) {
final user = ref.watch(currentUserProvider);
if (user == null) {
return Stream.value(null);
}
final authService = ref.watch(authServiceProvider);
return authService.userDataStream(user.uid);
});
// Login provider
final loginProvider = Provider<Future<UserCredential> Function({
required String email,
required String password,
})>((ref) {
final authService = ref.watch(authServiceProvider);
return ({required String email, required String password}) {
return authService.signInWithEmail(email: email, password: password);
};
});
// Sign up provider
final signUpProvider = Provider<Future<UserCredential> Function({
required String email,
required String password,
required String displayName,
required String organizationId,
String? departmentId,
})>((ref) {
final authService = ref.watch(authServiceProvider);
return ({
required String email,
required String password,
required String displayName,
required String organizationId,
String? departmentId,
}) {
return authService.signUpWithEmail(
email: email,
password: password,
displayName: displayName,
organizationId: organizationId,
departmentId: departmentId,
);
};
});
// Sign out provider
final signOutProvider = Provider<Future<void> Function()>((ref) {
final authService = ref.watch(authServiceProvider);
return () => authService.signOut();
});
// Reset password provider
final resetPasswordProvider = Provider<Future<void> Function(String)>((ref) {
final authService = ref.watch(authServiceProvider);
return (String email) => authService.resetPassword(email);
});
+86
View File
@@ -0,0 +1,86 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../services/time_service.dart';
import '../models/time_registration.dart';
import 'auth_provider.dart';
// Time service provider
final timeServiceProvider = Provider<TimeService>((ref) => TimeService());
// Active registration provider - henter pågående timeregistrering
final activeRegistrationProvider = FutureProvider<TimeRegistration?>((ref) async {
final user = ref.watch(currentUserProvider);
if (user == null) return null;
final timeService = ref.watch(timeServiceProvider);
return timeService.getActiveRegistration(user.uid);
});
// Today's registrations provider
final todayRegistrationsProvider = StreamProvider<List<TimeRegistration>>((ref) {
final user = ref.watch(currentUserProvider);
if (user == null) {
return Stream.value([]);
}
final timeService = ref.watch(timeServiceProvider);
final today = DateTime.now();
final startOfDay = DateTime(today.year, today.month, today.day);
final endOfDay = DateTime(today.year, today.month, today.day, 23, 59, 59);
return timeService.registrationsStream(
userId: user.uid,
startDate: startOfDay,
endDate: endOfDay,
);
});
// Registrations for date range provider
final registrationsProvider = FutureProvider.family<List<TimeRegistration>, DateRange>(
(ref, dateRange) async {
final user = ref.watch(currentUserProvider);
if (user == null) return [];
final timeService = ref.watch(timeServiceProvider);
return timeService.getRegistrations(
userId: user.uid,
startDate: dateRange.start,
endDate: dateRange.end,
);
},
);
// Total hours for period provider
final totalHoursProvider = FutureProvider.family<Map<String, int>, DateRange>(
(ref, dateRange) async {
final user = ref.watch(currentUserProvider);
if (user == null) {
return {'total': 0, 'ordinary': 0, 'overtime': 0};
}
final timeService = ref.watch(timeServiceProvider);
return timeService.calculateTotalHours(
userId: user.uid,
startDate: dateRange.start,
endDate: dateRange.end,
);
},
);
// Helper class for date ranges
class DateRange {
final DateTime start;
final DateTime end;
DateRange({required this.start, required this.end});
@override
bool operator ==(Object other) =>
identical(this, other) ||
other is DateRange &&
runtimeType == other.runtimeType &&
start == other.start &&
end == other.end;
@override
int get hashCode => start.hashCode ^ end.hashCode;
}