Ei kuvausta

main.dart 7.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. import 'dart:convert';
  2. import 'package:flutter/foundation.dart';
  3. import 'package:flutter/material.dart';
  4. import 'package:http/http.dart' as http;
  5. import 'package:shared_preferences/shared_preferences.dart';
  6. import 'custom_info_card.dart'; // Ensure this path is correct
  7. void main() => runApp(const MyApp());
  8. class MyApp extends StatelessWidget {
  9. const MyApp({super.key});
  10. static const Color primaryColor = Color.fromARGB(255, 254, 100, 91);
  11. static const Color secondaryColor = Colors.orange;
  12. @override
  13. Widget build(BuildContext context) {
  14. return MaterialApp(
  15. title: 'Custom Info Cards',
  16. theme: ThemeData(
  17. primaryColor: primaryColor,
  18. appBarTheme: const AppBarTheme(
  19. backgroundColor: primaryColor,
  20. elevation: 0,
  21. ),
  22. ),
  23. home: const HomePage(),
  24. debugShowCheckedModeBanner: false,
  25. );
  26. }
  27. }
  28. class HomePage extends StatefulWidget {
  29. const HomePage({super.key});
  30. @override
  31. State<HomePage> createState() => _HomePageState();
  32. }
  33. class _HomePageState extends State<HomePage> {
  34. List<Map<String, dynamic>> infoCards = [];
  35. static const String googleSheetUrl = 'https://script.google.com/macros/s/AKfycbw1htmk7rlwLMOkOrpZ9ED-7ErWgMlYNtLKxoQ9QO-FopqTAhJuQkR7Gs1LxTJakbMT/exec';
  36. @override
  37. void initState() {
  38. super.initState();
  39. _loadInfoCards();
  40. }
  41. Future<void> _loadInfoCards() async {
  42. final prefs = await SharedPreferences.getInstance();
  43. // Load default data if no saved data
  44. String? savedData = prefs.getString('infoCardsData');
  45. setState(() {
  46. if (savedData != null) {
  47. try {
  48. final decoded = json.decode(savedData);
  49. infoCards = List<Map<String, dynamic>>.from(decoded);
  50. } catch (e) {
  51. if (kDebugMode) {
  52. print('Error decoding saved data: $e');
  53. }
  54. infoCards = _defaultInfoCards();
  55. }
  56. } else {
  57. infoCards = _defaultInfoCards();
  58. }
  59. });
  60. await _fetchUpdatedData();
  61. }
  62. List<Map<String, dynamic>> _defaultInfoCards() {
  63. return [
  64. {
  65. 'image': 'assets/icons/icono1.jpg',
  66. 'title': 'Tutorias de programación',
  67. 'subtitle': '8:00am a 11:30am En biblioteca Lázaro',
  68. },
  69. ];
  70. }
  71. Future<void> _fetchUpdatedData() async {
  72. try {
  73. final response = await http.get(Uri.parse(googleSheetUrl));
  74. if (response.statusCode == 200 && response.body.isNotEmpty) {
  75. try {
  76. final dynamic decodedData = json.decode(response.body);
  77. List<Map<String, dynamic>> newData = [];
  78. if (decodedData is Map && decodedData.containsKey('feed')) {
  79. final entries = decodedData['feed']['entry'] as List?;
  80. if (entries != null) {
  81. for (var entry in entries) {
  82. final Map<String, dynamic> card = {
  83. 'image': entry['gsx\$image']?['\$t'] ?? '',
  84. 'title': entry['gsx\$title']?['\$t'] ?? '',
  85. 'subtitle': entry['gsx\$subtitle']?['\$t'] ?? '',
  86. 'texto': entry['gsx\$texto']?['\$t'] ?? '',
  87. };
  88. newData.add(card);
  89. }
  90. }
  91. } else if (decodedData is List) {
  92. newData = List<Map<String, dynamic>>.from(decodedData);
  93. } else {
  94. throw Exception('Unexpected JSON format');
  95. }
  96. if (json.encode(newData) != json.encode(infoCards)) {
  97. setState(() {
  98. infoCards = newData;
  99. });
  100. final prefs = await SharedPreferences.getInstance();
  101. prefs.setString('infoCardsData', json.encode(newData));
  102. if (kDebugMode) {
  103. print('Updated with new data from Google Sheet');
  104. }
  105. }
  106. } catch (e) {
  107. if (kDebugMode) {
  108. print('Error parsing response data: $e');
  109. }
  110. }
  111. } else {
  112. if (kDebugMode) {
  113. print('Failed to fetch updated data: ${response.statusCode}');
  114. }
  115. }
  116. } catch (e) {
  117. if (kDebugMode) {
  118. print('Error fetching data: $e');
  119. }
  120. }
  121. }
  122. @override
  123. Widget build(BuildContext context) {
  124. return Scaffold(
  125. appBar: AppBar(
  126. flexibleSpace: Padding(
  127. padding: EdgeInsets.only(top: MediaQuery.of(context).padding.top),
  128. child: Image.asset(
  129. 'assets/header_image.png',
  130. fit: BoxFit.cover,
  131. ),
  132. ),
  133. toolbarHeight: MediaQuery.of(context).size.height * 0.19,
  134. ),
  135. body: Container(
  136. decoration: const BoxDecoration(
  137. gradient: LinearGradient(
  138. colors: [MyApp.primaryColor, MyApp.secondaryColor],
  139. begin: Alignment.topCenter,
  140. end: Alignment.bottomCenter,
  141. ),
  142. ),
  143. child: SafeArea(
  144. child: Column(
  145. children: [
  146. Expanded(
  147. child: RefreshIndicator(
  148. onRefresh: _fetchUpdatedData,
  149. color: MyApp.primaryColor,
  150. child: ListView.builder(
  151. padding: const EdgeInsets.all(0),
  152. itemCount: infoCards.isEmpty ? 1 : infoCards.length,
  153. itemBuilder: (context, index) {
  154. // Show error widget if no data is available
  155. if (infoCards.isEmpty) {
  156. return Card(
  157. margin: const EdgeInsets.all(16),
  158. color: Colors.white,
  159. child: Padding(
  160. padding: const EdgeInsets.all(16),
  161. child: Column(
  162. children: [
  163. const Icon(Icons.warning_amber_rounded,
  164. size: 48, color: Colors.orange),
  165. const SizedBox(height: 16),
  166. const Text(
  167. 'No se pudo cargar la información',
  168. style: TextStyle(
  169. fontSize: 18,
  170. fontWeight: FontWeight.bold,
  171. ),
  172. ),
  173. const SizedBox(height: 8),
  174. const Text(
  175. 'Desliza hacia abajo para intentar de nuevo',
  176. textAlign: TextAlign.center,
  177. ),
  178. const SizedBox(height: 16),
  179. ElevatedButton(
  180. onPressed: () => _fetchUpdatedData(),
  181. style: ElevatedButton.styleFrom(
  182. backgroundColor: MyApp.primaryColor,
  183. ),
  184. child: const Text('Reintentar'),
  185. ),
  186. ],
  187. ),
  188. ),
  189. );
  190. }
  191. // Show data card
  192. final card = infoCards[index];
  193. return CustomInfoCard(
  194. imagePath: "assets/icons/${card['image']}",
  195. title: card['title'],
  196. subtitle: card['subtitle'],
  197. texto: card['texto'],
  198. );
  199. },
  200. ),
  201. ),
  202. ),
  203. ],
  204. ),
  205. ),
  206. ),
  207. );
  208. }
  209. }