AtmoSphere/lib/screens/city_management_screen.dart
2026-01-16 18:22:32 +08:00

328 lines
10 KiB
Dart

import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../constants.dart';
import '../models/city_model.dart';
import '../providers/city_provider.dart';
import '../providers.dart';
class CityManagementScreen extends ConsumerStatefulWidget {
const CityManagementScreen({super.key});
@override
ConsumerState<CityManagementScreen> createState() =>
_CityManagementScreenState();
}
class _CityManagementScreenState extends ConsumerState<CityManagementScreen> {
final TextEditingController _searchController = TextEditingController();
bool _isSearching = false;
@override
void dispose() {
_searchController.dispose();
super.dispose();
}
Future<void> _addCity(String cityQuery) async {
if (cityQuery.trim().isEmpty) {
_showSnackBar('Please enter a city name');
return;
}
try {
// 先尝试获取天气数据以验证城市是否存在
final repository = ref.read(weatherRepositoryProvider);
final weather = await repository.fetchWeather(cityQuery.trim());
// 创建城市模型
final city = CityModel(
name: weather.location.name,
region: weather.location.region,
country: weather.location.country,
query: cityQuery.trim(),
);
// 添加到列表
final success = await ref.read(cityListProvider.notifier).addCity(city);
if (success) {
_showSnackBar('City added successfully');
_searchController.clear();
setState(() {
_isSearching = false;
});
} else {
_showSnackBar('City already exists');
}
} catch (e) {
_showSnackBar('Failed to add city: ${e.toString()}');
}
}
Future<void> _removeCity(String query) async {
final cities = ref.read(cityListProvider);
if (cities.length <= 1) {
_showSnackBar('At least one city must remain');
return;
}
final confirmed = await showDialog<bool>(
context: context,
builder: (context) => AlertDialog(
title: const Text('Delete City'),
content: const Text('Are you sure you want to remove this city?'),
actions: [
TextButton(
onPressed: () => Navigator.pop(context, false),
child: const Text('Cancel'),
),
TextButton(
onPressed: () => Navigator.pop(context, true),
child: const Text('Delete', style: TextStyle(color: Colors.red)),
),
],
),
);
if (confirmed == true) {
await ref.read(cityListProvider.notifier).removeCity(query);
_showSnackBar('City removed');
}
}
Future<void> _switchCity(String query) async {
await ref.read(weatherProvider.notifier).switchCity(query);
if (mounted) {
Navigator.pop(context);
_showSnackBar('City switched');
}
}
void _showSnackBar(String message) {
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(message), duration: const Duration(seconds: 2)),
);
}
}
@override
Widget build(BuildContext context) {
final cities = ref.watch(cityListProvider);
final currentCityAsync = ref.watch(currentCityProvider);
return Scaffold(
appBar: AppBar(
title: const Text('Manage Cities'),
actions: [
IconButton(
icon: Icon(_isSearching ? Icons.close : Icons.add),
onPressed: () {
setState(() {
_isSearching = !_isSearching;
if (!_isSearching) {
_searchController.clear();
}
});
},
),
],
),
body: Column(
children: [
if (_isSearching)
Container(
padding: const EdgeInsets.all(16.0),
decoration: BoxDecoration(
color: kCardBackground,
boxShadow: [
BoxShadow(
color: Colors.black.withOpacity(0.1),
blurRadius: 4,
offset: const Offset(0, 2),
),
],
),
child: Row(
children: [
Expanded(
child: TextField(
controller: _searchController,
decoration: InputDecoration(
hintText: 'Enter city name (e.g., London, Beijing)',
hintStyle: TextStyle(
color: Colors.white.withOpacity(0.7),
),
filled: true,
fillColor: Colors.black.withOpacity(0.3),
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(12),
borderSide: BorderSide.none,
),
contentPadding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 12,
),
),
style: const TextStyle(color: Colors.white),
onSubmitted: _addCity,
),
),
const SizedBox(width: 8),
IconButton(
icon: const Icon(Icons.search),
onPressed: () => _addCity(_searchController.text),
),
],
),
),
Expanded(
child: cities.isEmpty
? Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Icons.location_city,
size: 64,
color: Colors.white.withOpacity(0.5),
),
const SizedBox(height: 16),
Text(
'No cities added',
style: TextStyle(
color: Colors.white.withOpacity(0.7),
fontSize: 18,
),
),
const SizedBox(height: 8),
Text(
'Tap + to add a city',
style: TextStyle(
color: Colors.white.withOpacity(0.5),
fontSize: 14,
),
),
],
),
)
: ListView.builder(
padding: const EdgeInsets.all(16.0),
itemCount: cities.length,
itemBuilder: (context, index) {
final city = cities[index];
return currentCityAsync.when(
data: (currentQuery) => _CityCard(
city: city,
isCurrent: city.query == currentQuery,
onTap: () => _switchCity(city.query),
onDelete: cities.length > 1
? () => _removeCity(city.query)
: null,
),
loading: () => _CityCard(
city: city,
isCurrent: city.isDefault,
onTap: () => _switchCity(city.query),
onDelete: cities.length > 1
? () => _removeCity(city.query)
: null,
),
error: (_, __) => _CityCard(
city: city,
isCurrent: city.isDefault,
onTap: () => _switchCity(city.query),
onDelete: cities.length > 1
? () => _removeCity(city.query)
: null,
),
);
},
),
),
],
),
);
}
}
class _CityCard extends StatelessWidget {
final CityModel city;
final bool isCurrent;
final VoidCallback onTap;
final VoidCallback? onDelete;
const _CityCard({
required this.city,
required this.isCurrent,
required this.onTap,
this.onDelete,
});
@override
Widget build(BuildContext context) {
return Container(
margin: const EdgeInsets.only(bottom: 12.0),
decoration: BoxDecoration(
color: isCurrent ? Colors.blue.withOpacity(0.3) : kCardBackground,
borderRadius: kCardBorderRadius,
border: isCurrent
? Border.all(color: Colors.blueAccent, width: 2)
: null,
),
child: ListTile(
leading: CircleAvatar(
backgroundColor: isCurrent
? Colors.blueAccent
: Colors.white.withOpacity(0.2),
child: Icon(
isCurrent ? Icons.location_on : Icons.location_city,
color: Colors.white,
),
),
title: Text(
city.name,
style: TextStyle(
fontWeight: isCurrent ? FontWeight.bold : FontWeight.normal,
color: Colors.white,
),
),
subtitle: Text(
city.displayName,
style: TextStyle(color: Colors.white.withOpacity(0.7), fontSize: 12),
),
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
if (isCurrent)
Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
color: Colors.blueAccent,
borderRadius: BorderRadius.circular(12),
),
child: const Text(
'Current',
style: TextStyle(
color: Colors.white,
fontSize: 12,
fontWeight: FontWeight.bold,
),
),
),
if (onDelete != null) ...[
const SizedBox(width: 8),
IconButton(
icon: const Icon(Icons.delete_outline, color: Colors.red),
onPressed: onDelete,
tooltip: 'Delete city',
),
],
],
),
onTap: onTap,
),
);
}
}