import 'package:flutter/material.dart'; import 'package:http/http.dart' as http; import 'package:flutter_dotenv/flutter_dotenv.dart'; // Import dotenv import 'dart:convert'; import 'dart:io'; import 'ItemMenu.dart'; import 'Camera.dart'; import '../classes/events.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:intl/intl.dart'; import 'package:intl/date_symbol_data_local.dart'; import 'package:camera/camera.dart'; import 'package:mapbox_gl/mapbox_gl.dart'; // Add this import import '../variable/globals.dart' as globals; void main() { initializeDateFormatting("fr_FR", null).then((_) => runApp(const MyApp())); } class MyApp extends StatelessWidget { const MyApp({super.key}); @override Widget build(BuildContext context) { return MaterialApp( home: const ListItemMenu(), debugShowCheckedModeBanner: false, ); } } class ListItemMenu extends StatefulWidget { const ListItemMenu({super.key}); @override State createState() => _MyHomePageState(); } class _MyHomePageState extends State { Future> postsFuture = getPosts(); List filteredPosts = []; String geographicalZone = ''; String query = ''; List suggestions = []; // Store suggestions TextEditingController inputGeo = TextEditingController(); // Fetching events from API static Future> getPosts() async { SharedPreferences prefs = await SharedPreferences.getInstance(); var accessToken = prefs.getString("access_token") ?? ""; final List body = []; if (accessToken.isNotEmpty) { var url = Uri.parse("${globals.api}/events"); final response = await http.get(url, headers: { "Content-Type": "application/json", HttpHeaders.cookieHeader: "access_token=${accessToken}" }); final List body = json.decode(utf8.decode(response.bodyBytes)); return body.map((e) => Events.fromJson(e)).toList(); } return body; } Future searchSuggestions(String input) async { await dotenv.load(fileName: ".env"); // Load .env file final mapboxAccessToken = dotenv.env['MAPBOX_ACCESS_TOKEN'] ?? ''; final url = 'https://api.mapbox.com/geocoding/v5/mapbox.places/${input}.json?access_token=${mapboxAccessToken}&proximity=ip'; // Replace with your Mapbox token final response = await http.get(Uri.parse(url)); if (response.statusCode == 200) { final data = json.decode(response.body); setState(() { suggestions = (data['features'] as List) .map((feature) => feature['place_name'] as String) // Cast to String explicitly .toList(); }); } else { throw Exception('Failed to load suggestions'); } } Padding _buildGeographicalZoneSearchField() { return Padding( padding: const EdgeInsets.all(8.0), child: Column( children: [ TextField( controller: inputGeo, decoration: InputDecoration( labelText: 'Search by geographical zone', border: OutlineInputBorder(), ), onChanged: (value) { setState(() { geographicalZone = value; searchSuggestions(value); }); }, ), if (suggestions.isNotEmpty) Container( height: 200, decoration: BoxDecoration( border: Border.all(color: Colors.blue), // Add a border color borderRadius: BorderRadius.circular(8), // Optional: rounded corners ), child: ListView.builder( shrinkWrap: true, // Ensure the list takes only the required space itemCount: suggestions.length, itemBuilder: (context, index) { return ListTile( title: Text(suggestions[index]), onTap: () { setState(() { geographicalZone = suggestions[index]; inputGeo.text = suggestions[index]; suggestions.clear(); }); }, ); }, ), ), ], ), ); } @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: const Text("Item list menu"), backgroundColor: Colors.blue, foregroundColor: Colors.white, actions: [ IconButton( icon: const Icon(Icons.search), onPressed: () { showSearch( context: context, delegate: SearchDelegateExample(), ); }, ), ], ), body: Column( children: [ _buildGeographicalZoneSearchField(), Expanded( child: FutureBuilder>( future: postsFuture, builder: (context, snapshot) { if (snapshot.connectionState == ConnectionState.waiting) { return const Center(child: CircularProgressIndicator()); } else if (snapshot.hasData) { final posts = snapshot.data!; final displayedPosts = filteredPosts.isEmpty ? posts : filteredPosts; return buildPosts(displayedPosts); } else { return const Text("No data available"); } }, ), ), ], ), ); } // Function to display fetched data on screen Widget buildPosts(List posts) { return ListView.separated( itemCount: posts.length, itemBuilder: (context, index) { final post = posts[index]; final startDate = DateTime.parse(post.startDate!); final date = DateFormat.yMd().format(startDate); final time = DateFormat.Hm().format(startDate); return ListTile( title: Text('${post.name!}'), subtitle: Text('${post.place!}\n${date} ${time}'), onTap: () { Navigator.push(context, MaterialPageRoute(builder: (_) => ItemMenu(title: post.id!))); }); }, separatorBuilder: (context, index) { return Divider(); }, ); } } class SearchDelegateExample extends SearchDelegate { @override List buildActions(BuildContext context) { return [ IconButton( icon: const Icon(Icons.clear), onPressed: () { query = ''; }, ), ]; } @override Widget buildLeading(BuildContext context) { return IconButton( icon: const Icon(Icons.arrow_back), onPressed: () { close(context, null); }, ); } @override Widget buildResults(BuildContext context) { // Perform the search and return the results return FutureBuilder>( future: searchPosts(query), builder: (context, snapshot) { if (snapshot.connectionState == ConnectionState.waiting) { return const Center(child: CircularProgressIndicator()); } else if (snapshot.hasData) { final posts = snapshot.data!; return ListView.builder( itemCount: posts.length, itemBuilder: (context, index) { final post = posts[index]; return ListTile( title: Text(post.name!), subtitle: Text(post.place!), onTap: () { Navigator.push( context, MaterialPageRoute( builder: (_) => ItemMenu(title: post.id!), ), ); }, ); }, ); } else { return const Center(child: Text("No results found")); } }, ); } @override Widget buildSuggestions(BuildContext context) { return Container(); // Implement suggestions if needed } Future> searchPosts(String query) async { SharedPreferences prefs = await SharedPreferences.getInstance(); var accessToken = prefs.getString("access_token") ?? ""; final List body = []; if (accessToken.isNotEmpty) { var url = Uri.parse("${globals.api}/events/search?item=$query"); final response = await http.get(url, headers: { "Content-Type": "application/json", HttpHeaders.cookieHeader: "access_token=${accessToken}" }); final List body = json.decode(utf8.decode(response.bodyBytes)); return body.map((e) => Events.fromJson(e)).toList(); } return body; } }