Files
1356/lifetimer/lib/data/services/image_search_service.dart
T
Tomas Dvorak 37ffb93923 feat: Complete Phase 1 - Full Flutter app implementation with comprehensive features
Version: 1.1.0

Major changes:
- Implemented complete Flutter app structure with all core features
- Added comprehensive UI screens for auth, countdown, goals, profile, settings, and social features
- Integrated Supabase backend with authentication and data repositories
- Added offline support with Hive caching and local storage
- Implemented comprehensive routing with go_router
- Added location services with Google Maps integration
- Implemented notifications and home widget support
- Added voice recording capabilities and AI chat features
- Created comprehensive test suite and documentation
- Added Android and iOS platform configurations
- Implemented achievements system and social features
- Added calendar integration and bucket list functionality

This represents a complete Phase 1 milestone with 3,775 additions across 31 files.
2026-01-04 14:33:54 +01:00

114 lines
3.0 KiB
Dart

import 'dart:convert';
import 'package:http/http.dart' as http;
class UnsplashImage {
final String id;
final String url;
final String fullUrl;
final String? description;
final String? photographer;
final String? photographerUrl;
UnsplashImage({
required this.id,
required this.url,
required this.fullUrl,
this.description,
this.photographer,
this.photographerUrl,
});
factory UnsplashImage.fromJson(Map<String, dynamic> json) {
final urls = json['urls'] as Map<String, dynamic>;
final user = json['user'] as Map<String, dynamic>?;
return UnsplashImage(
id: json['id'] as String,
url: urls['regular'] as String? ?? urls['small'] as String,
fullUrl: urls['full'] as String? ?? urls['regular'] as String,
description: json['description'] as String?,
photographer: user?['name'] as String?,
photographerUrl: user?['links']?['html'] as String?,
);
}
}
class ImageSearchService {
final String _accessKey;
final http.Client _client;
ImageSearchService({
required String accessKey,
http.Client? client,
}) : _accessKey = accessKey,
_client = client ?? http.Client();
Future<List<UnsplashImage>> searchImages({
required String query,
int perPage = 10,
String orientation = 'landscape',
}) async {
try {
final uri = Uri.https('api.unsplash.com', '/search/photos', {
'query': query,
'per_page': perPage.toString(),
'orientation': orientation,
});
final response = await _client.get(
uri,
headers: {
'Authorization': 'Client-ID $_accessKey',
},
);
if (response.statusCode == 200) {
final data = json.decode(response.body) as Map<String, dynamic>;
final results = data['results'] as List;
return results
.map((json) => UnsplashImage.fromJson(json as Map<String, dynamic>))
.toList();
} else {
throw Exception('Failed to search images: ${response.statusCode}');
}
} catch (e) {
throw Exception('Error searching images: $e');
}
}
Future<UnsplashImage?> getRandomImage({
String? query,
String orientation = 'landscape',
}) async {
try {
final params = <String, String>{
'orientation': orientation,
};
if (query != null) {
params['query'] = query;
}
final uri = Uri.https('api.unsplash.com', '/photos/random', params);
final response = await _client.get(
uri,
headers: {
'Authorization': 'Client-ID $_accessKey',
},
);
if (response.statusCode == 200) {
final json = jsonDecode(response.body) as Map<String, dynamic>;
return UnsplashImage.fromJson(json);
} else {
throw Exception('Failed to get random image: ${response.statusCode}');
}
} catch (e) {
throw Exception('Error getting random image: $e');
}
}
void dispose() {
_client.close();
}
}