Apply dart format to all files

This commit is contained in:
dmit.b
2026-05-15 19:09:25 +03:00
parent 6497c1eebf
commit 69f3d09e62
11 changed files with 191 additions and 171 deletions
+1 -3
View File
@@ -35,9 +35,7 @@ class MyApp extends StatelessWidget {
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
),
supportedLocales: const [
Locale('en', ''),
],
supportedLocales: const [Locale('en', '')],
initialRoute: '/',
routes: {
'/': (_) => const LoginScreen(),
+124 -113
View File
@@ -67,141 +67,152 @@ class _LoginScreenState extends State<LoginScreen>
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
const Text(
'Family Safety',
textAlign: TextAlign.center,
style: TextStyle(fontSize: 32, fontWeight: FontWeight.bold),
),
const SizedBox(height: 24),
if (!kIsWeb) ...[
TextField(
controller: _serverUrlController,
decoration: const InputDecoration(
labelText: 'Server URL',
hintText: 'https://example.com/api',
border: OutlineInputBorder(),
prefixIcon: Icon(Icons.cloud_outlined),
const Text(
'Family Safety',
textAlign: TextAlign.center,
style: TextStyle(
fontSize: 32,
fontWeight: FontWeight.bold,
),
keyboardType: TextInputType.url,
),
const SizedBox(height: 16),
],
TabBar(
controller: _tabController,
tabs: const [
Tab(text: 'Login'),
Tab(text: 'Register'),
const SizedBox(height: 24),
if (!kIsWeb) ...[
TextField(
controller: _serverUrlController,
decoration: const InputDecoration(
labelText: 'Server URL',
hintText: 'https://example.com/api',
border: OutlineInputBorder(),
prefixIcon: Icon(Icons.cloud_outlined),
),
keyboardType: TextInputType.url,
),
const SizedBox(height: 16),
],
),
const SizedBox(height: 24),
TextField(
controller: _loginController,
decoration: const InputDecoration(
labelText: 'Login',
border: OutlineInputBorder(),
TabBar(
controller: _tabController,
tabs: const [
Tab(text: 'Login'),
Tab(text: 'Register'),
],
),
),
const SizedBox(height: 16),
TextField(
controller: _passwordController,
decoration: const InputDecoration(
labelText: 'Password',
border: OutlineInputBorder(),
const SizedBox(height: 24),
TextField(
controller: _loginController,
decoration: const InputDecoration(
labelText: 'Login',
border: OutlineInputBorder(),
),
),
obscureText: true,
),
if (_tabController.index == 1) ...[
const SizedBox(height: 16),
TextField(
controller: _secretKeyController,
controller: _passwordController,
decoration: const InputDecoration(
labelText: 'Secret Key',
labelText: 'Password',
border: OutlineInputBorder(),
),
obscureText: true,
),
],
const SizedBox(height: 16),
if (authProvider.error.isNotEmpty)
Text(
authProvider.error,
style: const TextStyle(color: Colors.red),
),
const SizedBox(height: 16),
ElevatedButton(
onPressed: authProvider.isLoading
? null
: () async {
if (!kIsWeb) {
final status = await Geolocator.checkPermission();
if (status == LocationPermission.denied) {
final requested = await Geolocator.requestPermission();
if (requested == LocationPermission.denied) {
authProvider.setError('Location permission denied');
if (_tabController.index == 1) ...[
const SizedBox(height: 16),
TextField(
controller: _secretKeyController,
decoration: const InputDecoration(
labelText: 'Secret Key',
border: OutlineInputBorder(),
),
obscureText: true,
),
],
const SizedBox(height: 16),
if (authProvider.error.isNotEmpty)
Text(
authProvider.error,
style: const TextStyle(color: Colors.red),
),
const SizedBox(height: 16),
ElevatedButton(
onPressed: authProvider.isLoading
? null
: () async {
if (!kIsWeb) {
final status =
await Geolocator.checkPermission();
if (status == LocationPermission.denied) {
final requested =
await Geolocator.requestPermission();
if (requested == LocationPermission.denied) {
authProvider.setError(
'Location permission denied',
);
return;
}
}
if (status ==
LocationPermission.deniedForever) {
authProvider.setError(
'Location permission permanently denied. Please enable it in settings.',
);
return;
}
}
if (status == LocationPermission.deniedForever) {
authProvider.setError('Location permission permanently denied. Please enable it in settings.');
return;
if (!kIsWeb) {
final newUrl = _serverUrlController.text.trim();
if (newUrl.isNotEmpty &&
newUrl != settings.baseUrl) {
ApiConfig.setBaseUrl(newUrl);
await settings.setBaseUrl(newUrl);
}
}
}
if (!kIsWeb) {
final newUrl = _serverUrlController.text.trim();
if (newUrl.isNotEmpty && newUrl != settings.baseUrl) {
ApiConfig.setBaseUrl(newUrl);
await settings.setBaseUrl(newUrl);
}
}
if (_tabController.index == 0) {
try {
await authProvider.login(
_loginController.text,
_passwordController.text,
if (_tabController.index == 0) {
try {
await authProvider.login(
_loginController.text,
_passwordController.text,
);
if (!context.mounted) return;
Navigator.of(
context,
).pushReplacementNamed('/map');
} catch (e) {
// Error is handled by provider
}
} else {
if (_secretKeyController.text != _secretKey) {
authProvider.setError('Invalid secret key');
return;
}
Digest digest = md5.convert(
utf8.encode(_secretKeyController.text),
);
if (!context.mounted) return;
Navigator.of(
context,
).pushReplacementNamed('/map');
} catch (e) {
// Error is handled by provider
String secretKeyHash = hex.encode(digest.bytes);
try {
await authProvider.register(
_loginController.text,
_passwordController.text,
secretKeyHash,
);
} catch (e) {
// Error is handled by provider
}
}
} else {
if (_secretKeyController.text != _secretKey) {
authProvider.setError('Invalid secret key');
return;
}
Digest digest = md5.convert(
utf8.encode(_secretKeyController.text),
);
String secretKeyHash = hex.encode(digest.bytes);
try {
await authProvider.register(
_loginController.text,
_passwordController.text,
secretKeyHash,
);
} catch (e) {
// Error is handled by provider
}
}
},
child: authProvider.isLoading
? const SizedBox(
height: 20,
width: 20,
child: CircularProgressIndicator(strokeWidth: 2),
)
: Text(
_tabController.index == 0 ? 'Login' : 'Register',
),
),
],
},
child: authProvider.isLoading
? const SizedBox(
height: 20,
width: 20,
child: CircularProgressIndicator(strokeWidth: 2),
)
: Text(
_tabController.index == 0 ? 'Login' : 'Register',
),
),
],
),
),
),
),
),
),
),
);
}
+6 -1
View File
@@ -7,7 +7,12 @@ class GeoService {
GeoService({http.Client? client}) : _client = client ?? http.Client();
Future<void> updatePosition(String token, String id, double x, double y) async {
Future<void> updatePosition(
String token,
String id,
double x,
double y,
) async {
final response = await _client.put(
Uri.parse('${ApiConfig.geoUrl}?id=$id'),
headers: {
+13 -1
View File
@@ -4,5 +4,17 @@ import 'package:flutter/foundation.dart' show kIsWeb;
class PlatformService {
bool get isWeb => kIsWeb;
bool get isNative => !kIsWeb;
String get platformName => isWeb ? 'web' : Platform.isAndroid ? 'android' : Platform.isIOS ? 'ios' : Platform.isWindows ? 'windows' : Platform.isMacOS ? 'macos' : Platform.isLinux ? 'linux' : 'unknown';
String get platformName => isWeb
? 'web'
: Platform.isAndroid
? 'android'
: Platform.isIOS
? 'ios'
: Platform.isWindows
? 'windows'
: Platform.isMacOS
? 'macos'
: Platform.isLinux
? 'linux'
: 'unknown';
}
+2 -1
View File
@@ -2,7 +2,8 @@ import 'package:shared_preferences/shared_preferences.dart';
class SettingsService {
static const String _baseUrlKey = 'server_base_url';
static const String _defaultBaseUrl = 'https://family-safety.onrender.com/api';
static const String _defaultBaseUrl =
'https://family-safety.onrender.com/api';
String _baseUrl = _defaultBaseUrl;
bool _initialized = false;
+19 -8
View File
@@ -7,7 +7,11 @@ class ShareService {
ShareService({http.Client? client}) : _client = client ?? http.Client();
Future<Map<String, dynamic>> createShare(String token, double x, double y) async {
Future<Map<String, dynamic>> createShare(
String token,
double x,
double y,
) async {
final response = await _client.post(
Uri.parse(ApiConfig.shareUrl),
headers: {
@@ -25,20 +29,27 @@ class ShareService {
'share_id': data['share_id']?.toString() ?? '',
};
} catch (e) {
throw Exception('Failed to parse response: $e | Body: ${response.body.substring(0, response.body.length > 200 ? 200 : response.body.length)}');
throw Exception(
'Failed to parse response: $e | Body: ${response.body.substring(0, response.body.length > 200 ? 200 : response.body.length)}',
);
}
} else {
final errorBody = response.body.length > 200 ? response.body.substring(0, 200) : response.body;
throw Exception('Failed to create share link: ${response.statusCode} - $errorBody');
final errorBody = response.body.length > 200
? response.body.substring(0, 200)
: response.body;
throw Exception(
'Failed to create share link: ${response.statusCode} - $errorBody',
);
}
}
Future<Map<String, dynamic>> getPositionByShareId(String token, String shareId) async {
Future<Map<String, dynamic>> getPositionByShareId(
String token,
String shareId,
) async {
final response = await _client.get(
Uri.parse('${ApiConfig.watchUrl}?share_id=$shareId'),
headers: {
'Authorization': 'Bearer $token',
},
headers: {'Authorization': 'Bearer $token'},
);
if (response.statusCode == 200) {
+1 -2
View File
@@ -1,4 +1,3 @@
class StringUtil {
static bool isValidEmail(String email) {
final regExp = RegExp(r'^[\w-]+(\.[\w-]+)*@[\w-]+(\.[\w-]+)*\.(\w+)$');
@@ -14,7 +13,7 @@ class StringUtil {
try {
final date = DateTime.parse(dateTime);
return '${date.day}/${date.month}/${date.year} '
'${date.hour}:${date.minute.toString().padLeft(2, '0')}';
'${date.hour}:${date.minute.toString().padLeft(2, '0')}';
} catch (e) {
return dateTime;
}
+3 -14
View File
@@ -4,11 +4,7 @@ class ErrorDisplay extends StatelessWidget {
final String message;
final VoidCallback? onRetry;
const ErrorDisplay({
super.key,
required this.message,
this.onRetry,
});
const ErrorDisplay({super.key, required this.message, this.onRetry});
@override
Widget build(BuildContext context) {
@@ -18,11 +14,7 @@ class ErrorDisplay extends StatelessWidget {
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
const Icon(
Icons.error_outline,
size: 48,
color: Colors.red,
),
const Icon(Icons.error_outline, size: 48, color: Colors.red),
const SizedBox(height: 16),
Text(
message,
@@ -31,10 +23,7 @@ class ErrorDisplay extends StatelessWidget {
),
if (onRetry != null) ...[
const SizedBox(height: 16),
ElevatedButton(
onPressed: onRetry,
child: const Text('Retry'),
),
ElevatedButton(onPressed: onRetry, child: const Text('Retry')),
],
],
),
+2 -8
View File
@@ -3,10 +3,7 @@ import 'package:flutter/material.dart';
class LoadingIndicator extends StatelessWidget {
final String message;
const LoadingIndicator({
super.key,
this.message = 'Loading...',
});
const LoadingIndicator({super.key, this.message = 'Loading...'});
@override
Widget build(BuildContext context) {
@@ -16,10 +13,7 @@ class LoadingIndicator extends StatelessWidget {
children: [
const CircularProgressIndicator(),
const SizedBox(height: 16),
Text(
message,
style: const TextStyle(fontSize: 16),
),
Text(message, style: const TextStyle(fontSize: 16)),
],
),
);