Compare commits
6 Commits
vibe-main-
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 5f75c53866 | |||
| e18863cf1c | |||
| 5bd3b1f18f | |||
| 5b1ffaede8 | |||
| c7be5241a2 | |||
| f11e3f7b63 |
@ -1,192 +0,0 @@
|
|||||||
# Architecture du Projet - Pokéguess
|
|
||||||
|
|
||||||
## Vue d'ensemble
|
|
||||||
|
|
||||||
Application Flutter affichant les 151 premiers Pokémon avec noms en français, via l'API Tyradex et cache local SQLite.
|
|
||||||
|
|
||||||
### Informations
|
|
||||||
|
|
||||||
- **Version** : 1.0.0+1
|
|
||||||
- **SDK** : Flutter >=3.1.0 <4.0.0
|
|
||||||
- **Plateformes** : Android, iOS, Web, macOS, Linux, Windows
|
|
||||||
|
|
||||||
## Architecture Générale
|
|
||||||
|
|
||||||
L'application suit une architecture en couches (layered architecture) avec une séparation claire des responsabilités :
|
|
||||||
|
|
||||||
```txt
|
|
||||||
┌─────────────────────────────────────────┐
|
|
||||||
│ Couche Présentation │
|
|
||||||
│ (Pages & Components/Widgets) │
|
|
||||||
├─────────────────────────────────────────┤
|
|
||||||
│ Couche Métier │
|
|
||||||
│ (Models) │
|
|
||||||
├─────────────────────────────────────────┤
|
|
||||||
│ Couche Services │
|
|
||||||
│ (API & Database & Utils) │
|
|
||||||
└─────────────────────────────────────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
## Structure du Projet
|
|
||||||
|
|
||||||
```md
|
|
||||||
lib/
|
|
||||||
├── main.dart # Point d'entrée de l'application
|
|
||||||
├── api/
|
|
||||||
│ └── pokemon_api.dart # Service d'accès à l'API Tyradex
|
|
||||||
├── components/
|
|
||||||
│ ├── pokemon_tile.dart # Widget de tuile pour afficher un Pokémon
|
|
||||||
│ └── pokemon_type.dart # Widget pour afficher le type d'un Pokémon
|
|
||||||
├── database/
|
|
||||||
│ └── pokedex_database.dart # Gestion de la base de données SQLite
|
|
||||||
├── models/
|
|
||||||
│ └── pokemon.dart # Modèle de données Pokémon
|
|
||||||
├── pages/
|
|
||||||
│ ├── pokemon_detail.dart # Page de détail d'un Pokémon
|
|
||||||
│ └── pokemon_list.dart # Page de liste des Pokémon
|
|
||||||
└── utils/
|
|
||||||
└── pokemon_type.dart # Utilitaires pour les types de Pokémon
|
|
||||||
```
|
|
||||||
|
|
||||||
## Composants Principaux
|
|
||||||
|
|
||||||
### 1. Point d'Entrée
|
|
||||||
|
|
||||||
[main.dart](../lib/main.dart) - Initialisation de l'application
|
|
||||||
|
|
||||||
- Thème Material Design 3
|
|
||||||
- Routes : `/` (liste) et `/pokemon-detail` (détail)
|
|
||||||
|
|
||||||
### 2. Couche Présentation
|
|
||||||
|
|
||||||
#### Pages
|
|
||||||
|
|
||||||
##### PokemonListPage ([pokemon_list.dart](../lib/pages/pokemon_list.dart))
|
|
||||||
|
|
||||||
StatefulWidget affichant une grille de 151 Pokémon (2 colonnes mobile, 4 web) via `GridView.builder` et `FutureBuilder`.
|
|
||||||
|
|
||||||
##### PokemonDetailPage ([pokemon_detail.dart](../lib/pages/pokemon_detail.dart))
|
|
||||||
|
|
||||||
StatefulWidget affichant les détails d'un Pokémon avec toggle shiny (`_isShiny`) via `GestureDetector`.
|
|
||||||
|
|
||||||
#### Components
|
|
||||||
|
|
||||||
##### PokemonTile ([pokemon_tile.dart](../lib/components/pokemon_tile.dart))
|
|
||||||
|
|
||||||
StatefulWidget affichant une carte Pokémon (image + nom) avec navigation vers détail au tap.
|
|
||||||
|
|
||||||
##### PokemonTypeWidget ([pokemon_type.dart](../lib/components/pokemon_type.dart))
|
|
||||||
|
|
||||||
StatelessWidget affichant un badge coloré par type.
|
|
||||||
|
|
||||||
### 3. Couche Modèle
|
|
||||||
|
|
||||||
#### Pokemon ([pokemon.dart](../lib/models/pokemon.dart))
|
|
||||||
|
|
||||||
- **Propriétés** :
|
|
||||||
- `name` : String - Nom du Pokémon
|
|
||||||
- `id` : int - Numéro du Pokémon
|
|
||||||
- `type1` : PokemonType - Type principal
|
|
||||||
- `type2` : PokemonType? - Type secondaire (optionnel)
|
|
||||||
|
|
||||||
- **Propriétés calculées** : `imageUrl`, `shinyImageUrl`, `cryUrl`, `formatedName`, `type1Color`, `type2Color`, `type1Formated`, `type2Formated`
|
|
||||||
- **Méthodes** : `fromJson()`, `toJson()`, `fromID()` (cache + API)
|
|
||||||
- **Enum** : `PokemonType` (20 types)
|
|
||||||
|
|
||||||
### 4. Couche Services
|
|
||||||
|
|
||||||
#### PokemonApi ([pokemon_api.dart](../lib/api/pokemon_api.dart))
|
|
||||||
|
|
||||||
Communication avec Tyradex API (<https://tyradex.vercel.app>)
|
|
||||||
|
|
||||||
- `getPokemon(int id)` : Récupère un Pokémon depuis l'API avec nom en français
|
|
||||||
- Exception levée si code HTTP ≠ 200
|
|
||||||
|
|
||||||
#### PokedexDatabase ([pokedex_database.dart](../lib/database/pokedex_database.dart))
|
|
||||||
|
|
||||||
Gestion SQLite locale (CRUD complet)
|
|
||||||
|
|
||||||
- **Schéma** : `pokemon (id, name, type1, type2)`
|
|
||||||
- **Méthodes** : `initDatabase()`, `getDatabase()`, `insertPokemon()`, `getPokemonList()`, `getPokemon()`, `deletePokemon()`, `deleteAllPokemon()`, `updatePokemon()`
|
|
||||||
- Non disponible sur Web (`kIsWeb`)
|
|
||||||
|
|
||||||
#### Utils ([pokemon_type.dart](../lib/utils/pokemon_type.dart))
|
|
||||||
|
|
||||||
- `frenchTypeToEnum()` : Conversion types français (API) vers enum
|
|
||||||
- `typeToColor()` : Mapping type → couleur
|
|
||||||
- `formatedTypeName()` : Formatage nom de type
|
|
||||||
|
|
||||||
## Flux de Données
|
|
||||||
|
|
||||||
### Récupération d'un Pokémon
|
|
||||||
|
|
||||||
```md
|
|
||||||
1. PokemonListPage demande Pokemon.fromID(id)
|
|
||||||
↓
|
|
||||||
2. Vérification dans PokedexDatabase (si non-web)
|
|
||||||
↓
|
|
||||||
3a. Si trouvé → Retour du Pokémon en cache
|
|
||||||
↓
|
|
||||||
3b. Si non trouvé → Appel à PokemonApi.getPokemon(id)
|
|
||||||
↓
|
|
||||||
4. Parsing JSON et création de l'objet Pokemon
|
|
||||||
↓
|
|
||||||
5. Sauvegarde dans PokedexDatabase (si non-web)
|
|
||||||
↓
|
|
||||||
6. Retour du Pokemon à la vue
|
|
||||||
```
|
|
||||||
|
|
||||||
### Navigation
|
|
||||||
|
|
||||||
```md
|
|
||||||
PokemonListPage
|
|
||||||
↓ (tap sur PokemonTile)
|
|
||||||
Navigator.pushNamed('/pokemon-detail', arguments: pokemon)
|
|
||||||
↓
|
|
||||||
PokemonDetailPage (récupère le pokemon via ModalRoute)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Patterns et Bonnes Pratiques
|
|
||||||
|
|
||||||
**Patterns** : Repository (classe `Pokemon`), Factory (`fromJson()`), FutureBuilder (async)
|
|
||||||
|
|
||||||
**Plateformes** : Détection Web via `kIsWeb`, adaptation colonnes grille
|
|
||||||
|
|
||||||
**Erreurs** : Try/catch + logs, vérification HTTP, messages dans FutureBuilder
|
|
||||||
|
|
||||||
## Dépendances
|
|
||||||
|
|
||||||
**Production** : `flutter`, `cupertino_icons` (^1.0.2), `sqflite` (^2.3.0), `http` (^1.1.0)
|
|
||||||
|
|
||||||
**Développement** : `flutter_test`, `flutter_lints` (^2.0.0)
|
|
||||||
|
|
||||||
## Points d'Extension
|
|
||||||
|
|
||||||
1. Recherche et filtres (type, génération)
|
|
||||||
2. Mode jeu (deviner à partir de silhouette, score)
|
|
||||||
3. Système de favoris
|
|
||||||
4. Lecture audio des cris
|
|
||||||
5. Détails enrichis (stats, évolutions, capacités)
|
|
||||||
6. Préchargement offline des 151 Pokémon
|
|
||||||
7. Tests unitaires et d'intégration
|
|
||||||
|
|
||||||
## Limitations
|
|
||||||
|
|
||||||
1. Pas de persistance locale sur Web
|
|
||||||
2. Limité à 151 Pokémon (Gen 1)
|
|
||||||
3. Gestion d'erreur minimale
|
|
||||||
4. Pas d'internationalisation
|
|
||||||
5. Pas de tests automatisés
|
|
||||||
|
|
||||||
## Notes Techniques
|
|
||||||
|
|
||||||
- Material Design 3
|
|
||||||
- Images : repository Yarkis01/TyraDex (GitHub)
|
|
||||||
- API : Tyradex API (https://tyradex.vercel.app)
|
|
||||||
- Noms de Pokémon en français
|
|
||||||
- Routage : routes nommées Flutter
|
|
||||||
- Pas de state management externe
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Dernière mise à jour** : 2 février 2026
|
|
||||||
@ -1,32 +1,56 @@
|
|||||||
PODS:
|
PODS:
|
||||||
- Flutter (1.0.0)
|
- Flutter (1.0.0)
|
||||||
- FMDB (2.7.12):
|
- sqflite_darwin (0.0.4):
|
||||||
- FMDB/standard (= 2.7.12)
|
|
||||||
- FMDB/Core (2.7.12)
|
|
||||||
- FMDB/standard (2.7.12):
|
|
||||||
- FMDB/Core
|
|
||||||
- sqflite (0.0.3):
|
|
||||||
- Flutter
|
- Flutter
|
||||||
- FMDB (>= 2.7.5)
|
- FlutterMacOS
|
||||||
|
- sqlite3 (3.51.1):
|
||||||
|
- sqlite3/common (= 3.51.1)
|
||||||
|
- sqlite3/common (3.51.1)
|
||||||
|
- sqlite3/dbstatvtab (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3/fts5 (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3/math (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3/perf-threadsafe (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3/rtree (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3/session (3.51.1):
|
||||||
|
- sqlite3/common
|
||||||
|
- sqlite3_flutter_libs (0.0.1):
|
||||||
|
- Flutter
|
||||||
|
- FlutterMacOS
|
||||||
|
- sqlite3 (~> 3.51.1)
|
||||||
|
- sqlite3/dbstatvtab
|
||||||
|
- sqlite3/fts5
|
||||||
|
- sqlite3/math
|
||||||
|
- sqlite3/perf-threadsafe
|
||||||
|
- sqlite3/rtree
|
||||||
|
- sqlite3/session
|
||||||
|
|
||||||
DEPENDENCIES:
|
DEPENDENCIES:
|
||||||
- Flutter (from `Flutter`)
|
- Flutter (from `Flutter`)
|
||||||
- sqflite (from `.symlinks/plugins/sqflite/ios`)
|
- sqflite_darwin (from `.symlinks/plugins/sqflite_darwin/darwin`)
|
||||||
|
- sqlite3_flutter_libs (from `.symlinks/plugins/sqlite3_flutter_libs/darwin`)
|
||||||
|
|
||||||
SPEC REPOS:
|
SPEC REPOS:
|
||||||
trunk:
|
trunk:
|
||||||
- FMDB
|
- sqlite3
|
||||||
|
|
||||||
EXTERNAL SOURCES:
|
EXTERNAL SOURCES:
|
||||||
Flutter:
|
Flutter:
|
||||||
:path: Flutter
|
:path: Flutter
|
||||||
sqflite:
|
sqflite_darwin:
|
||||||
:path: ".symlinks/plugins/sqflite/ios"
|
:path: ".symlinks/plugins/sqflite_darwin/darwin"
|
||||||
|
sqlite3_flutter_libs:
|
||||||
|
:path: ".symlinks/plugins/sqlite3_flutter_libs/darwin"
|
||||||
|
|
||||||
SPEC CHECKSUMS:
|
SPEC CHECKSUMS:
|
||||||
Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7
|
Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7
|
||||||
FMDB: 728731dd336af3936ce00f91d9d8495f5718a0e6
|
sqflite_darwin: 20b2a3a3b70e43edae938624ce550a3cbf66a3d0
|
||||||
sqflite: 5b24d06a453c198c13b305ceea4b4286cc07cfe4
|
sqlite3: 8d708bc63e9f4ce48f0ad9d6269e478c5ced1d9b
|
||||||
|
sqlite3_flutter_libs: d13b8b3003f18f596e542bcb9482d105577eff41
|
||||||
|
|
||||||
PODFILE CHECKSUM: 4305caec6b40dde0ae97be1573c53de1882a07e5
|
PODFILE CHECKSUM: 4305caec6b40dde0ae97be1573c53de1882a07e5
|
||||||
|
|
||||||
|
|||||||
@ -114,6 +114,7 @@
|
|||||||
379611D51312FF940696FCAB /* Pods-RunnerTests.release.xcconfig */,
|
379611D51312FF940696FCAB /* Pods-RunnerTests.release.xcconfig */,
|
||||||
0572F0648954163DB68882FD /* Pods-RunnerTests.profile.xcconfig */,
|
0572F0648954163DB68882FD /* Pods-RunnerTests.profile.xcconfig */,
|
||||||
);
|
);
|
||||||
|
name = Pods;
|
||||||
path = Pods;
|
path = Pods;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
};
|
};
|
||||||
@ -468,14 +469,13 @@
|
|||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
DEVELOPMENT_TEAM = 45K9ZX66MZ;
|
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
INFOPLIST_FILE = Runner/Info.plist;
|
INFOPLIST_FILE = Runner/Info.plist;
|
||||||
LD_RUNPATH_SEARCH_PATHS = (
|
LD_RUNPATH_SEARCH_PATHS = (
|
||||||
"$(inherited)",
|
"$(inherited)",
|
||||||
"@executable_path/Frameworks",
|
"@executable_path/Frameworks",
|
||||||
);
|
);
|
||||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokeguess;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokedex;
|
||||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||||
SWIFT_VERSION = 5.0;
|
SWIFT_VERSION = 5.0;
|
||||||
@ -647,14 +647,13 @@
|
|||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
DEVELOPMENT_TEAM = 45K9ZX66MZ;
|
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
INFOPLIST_FILE = Runner/Info.plist;
|
INFOPLIST_FILE = Runner/Info.plist;
|
||||||
LD_RUNPATH_SEARCH_PATHS = (
|
LD_RUNPATH_SEARCH_PATHS = (
|
||||||
"$(inherited)",
|
"$(inherited)",
|
||||||
"@executable_path/Frameworks",
|
"@executable_path/Frameworks",
|
||||||
);
|
);
|
||||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokeguess;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokedex;
|
||||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||||
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||||
@ -670,14 +669,13 @@
|
|||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
DEVELOPMENT_TEAM = 45K9ZX66MZ;
|
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
INFOPLIST_FILE = Runner/Info.plist;
|
INFOPLIST_FILE = Runner/Info.plist;
|
||||||
LD_RUNPATH_SEARCH_PATHS = (
|
LD_RUNPATH_SEARCH_PATHS = (
|
||||||
"$(inherited)",
|
"$(inherited)",
|
||||||
"@executable_path/Frameworks",
|
"@executable_path/Frameworks",
|
||||||
);
|
);
|
||||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokeguess;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.pokedex;
|
||||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||||
SWIFT_VERSION = 5.0;
|
SWIFT_VERSION = 5.0;
|
||||||
|
|||||||
@ -2,17 +2,6 @@
|
|||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
<plist version="1.0">
|
<plist version="1.0">
|
||||||
<dict>
|
<dict>
|
||||||
<key>NSLocalNetworkUsageDescription</key>
|
|
||||||
<string>This app uses the local network to discover and connect to nearby devices on your network.</string>
|
|
||||||
|
|
||||||
<key>NSBonjourServices</key>
|
|
||||||
<array>
|
|
||||||
<!-- Replace with the actual service types your app uses -->
|
|
||||||
<string>_http._tcp.</string>
|
|
||||||
<!-- Example: <string>_yourservice._tcp.</string> -->
|
|
||||||
</array>
|
|
||||||
<key>CADisableMinimumFrameDurationOnPhone</key>
|
|
||||||
<true/>
|
|
||||||
<key>CFBundleDevelopmentRegion</key>
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||||
<key>CFBundleDisplayName</key>
|
<key>CFBundleDisplayName</key>
|
||||||
@ -35,8 +24,6 @@
|
|||||||
<string>$(FLUTTER_BUILD_NUMBER)</string>
|
<string>$(FLUTTER_BUILD_NUMBER)</string>
|
||||||
<key>LSRequiresIPhoneOS</key>
|
<key>LSRequiresIPhoneOS</key>
|
||||||
<true/>
|
<true/>
|
||||||
<key>UIApplicationSupportsIndirectInputEvents</key>
|
|
||||||
<true/>
|
|
||||||
<key>UILaunchStoryboardName</key>
|
<key>UILaunchStoryboardName</key>
|
||||||
<string>LaunchScreen</string>
|
<string>LaunchScreen</string>
|
||||||
<key>UIMainStoryboardFile</key>
|
<key>UIMainStoryboardFile</key>
|
||||||
@ -54,5 +41,9 @@
|
|||||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||||
</array>
|
</array>
|
||||||
|
<key>CADisableMinimumFrameDurationOnPhone</key>
|
||||||
|
<true/>
|
||||||
|
<key>UIApplicationSupportsIndirectInputEvents</key>
|
||||||
|
<true/>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|||||||
@ -1,10 +1,7 @@
|
|||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'pages/home_page.dart';
|
|
||||||
import 'pages/game_page.dart';
|
|
||||||
import 'pages/pokemon_list.dart';
|
import 'pages/pokemon_list.dart';
|
||||||
import 'pages/pokemon_detail.dart';
|
import 'pages/pokemon_detail.dart';
|
||||||
import 'package:sqflite_common/sqflite.dart';
|
|
||||||
import 'package:sqflite_common_ffi/sqflite_ffi.dart';
|
import 'package:sqflite_common_ffi/sqflite_ffi.dart';
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
@ -27,14 +24,13 @@ class MyApp extends StatelessWidget {
|
|||||||
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
|
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
|
||||||
useMaterial3: true,
|
useMaterial3: true,
|
||||||
),
|
),
|
||||||
debugShowCheckedModeBanner:
|
debugShowCheckedModeBanner: false, // Permet de masquer la bannière "Debug"
|
||||||
false, // Permet de masquer la bannière "Debug"
|
// home a été enlevé pour être remplacé par la route "/"
|
||||||
routes: {
|
routes: {
|
||||||
'/': (context) => const HomePage(), // Menu principal
|
'/': (context) => const PokemonListPage(), // La route "/" est la page d'accueil
|
||||||
'/game': (context) => const GamePage(), // Page de jeu
|
'/pokemon-detail':(context) => const PokemonDetailPage(),
|
||||||
'/pokedex': (context) => const PokemonListPage(), // Liste des Pokémon
|
}
|
||||||
'/pokemon-detail': (context) => const PokemonDetailPage(),
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -1,519 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
import 'dart:math';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import '../models/pokemon.dart';
|
|
||||||
|
|
||||||
/// GamePage - Main quiz game where users guess Pokémon from silhouettes
|
|
||||||
class GamePage extends StatefulWidget {
|
|
||||||
const GamePage({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<GamePage> createState() => _GamePageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _GamePageState extends State<GamePage> {
|
|
||||||
// Game state
|
|
||||||
int score = 0;
|
|
||||||
int lives = 3;
|
|
||||||
Pokemon? currentPokemon;
|
|
||||||
bool isShiny = false;
|
|
||||||
bool isRevealed = false;
|
|
||||||
bool showHint = false;
|
|
||||||
bool isLoading = true;
|
|
||||||
|
|
||||||
final TextEditingController _controller = TextEditingController();
|
|
||||||
final Random _random = Random();
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_loadNewPokemon();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
_controller.dispose();
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Load a new random Pokémon (1-151)
|
|
||||||
Future<void> _loadNewPokemon() async {
|
|
||||||
setState(() {
|
|
||||||
isLoading = true;
|
|
||||||
isRevealed = false;
|
|
||||||
showHint = false;
|
|
||||||
_controller.clear();
|
|
||||||
});
|
|
||||||
|
|
||||||
// Random ID between 1 and 151
|
|
||||||
final int randomId = _random.nextInt(151) + 1;
|
|
||||||
|
|
||||||
// Shiny chance: 1/20 (5%)
|
|
||||||
final bool shiny = _random.nextInt(20) == 0;
|
|
||||||
|
|
||||||
// Fetch Pokémon using existing model (uses cache/API layer)
|
|
||||||
final Pokemon? pokemon = await Pokemon.fromID(randomId);
|
|
||||||
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
currentPokemon = pokemon;
|
|
||||||
isShiny = shiny;
|
|
||||||
isLoading = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Validate user's guess
|
|
||||||
void _validateGuess() {
|
|
||||||
// Hide keyboard
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
|
|
||||||
if (currentPokemon == null) return;
|
|
||||||
|
|
||||||
final String userInput = _controller.text.trim().toLowerCase();
|
|
||||||
final String correctName = currentPokemon!.name.toLowerCase();
|
|
||||||
|
|
||||||
if (userInput == correctName) {
|
|
||||||
_handleCorrectGuess();
|
|
||||||
} else {
|
|
||||||
_handleWrongGuess();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Handle correct guess
|
|
||||||
void _handleCorrectGuess() {
|
|
||||||
final int points = isShiny ? 20 : 10;
|
|
||||||
|
|
||||||
setState(() {
|
|
||||||
isRevealed = true;
|
|
||||||
score += points;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Show success message
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(
|
|
||||||
content: Text(
|
|
||||||
isShiny ? '✨ SHINY! +$points points!' : '✅ Correct! +$points points!',
|
|
||||||
style: const TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
backgroundColor: isShiny ? Colors.amber : Colors.green,
|
|
||||||
duration: const Duration(seconds: 2),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
// Wait 2 seconds, then load new Pokémon
|
|
||||||
Timer(const Duration(seconds: 2), () {
|
|
||||||
if (mounted) {
|
|
||||||
_loadNewPokemon();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Handle wrong guess
|
|
||||||
void _handleWrongGuess() {
|
|
||||||
setState(() {
|
|
||||||
lives--;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (lives <= 0) {
|
|
||||||
_showGameOverDialog();
|
|
||||||
} else {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(
|
|
||||||
content: Text(
|
|
||||||
'❌ Incorrect! Il reste $lives vie${lives > 1 ? 's' : ''}',
|
|
||||||
style: const TextStyle(fontSize: 16),
|
|
||||||
),
|
|
||||||
backgroundColor: Colors.redAccent,
|
|
||||||
duration: const Duration(seconds: 1),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Show game over dialog
|
|
||||||
void _showGameOverDialog() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (context) => AlertDialog(
|
|
||||||
backgroundColor: const Color(0xFF1A1A2E),
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
|
||||||
title: const Text(
|
|
||||||
'GAME OVER',
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 28,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
content: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
const Text(
|
|
||||||
'Le Pokémon était:',
|
|
||||||
style: TextStyle(color: Colors.white70, fontSize: 16),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
Text(
|
|
||||||
currentPokemon?.formatedName ?? '???',
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 24,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 24),
|
|
||||||
const Text(
|
|
||||||
'Score final',
|
|
||||||
style: TextStyle(color: Colors.white70, fontSize: 16),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
'$score',
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Color(0xFFE94560),
|
|
||||||
fontSize: 48,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
actions: [
|
|
||||||
Center(
|
|
||||||
child: ElevatedButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
_restartGame();
|
|
||||||
},
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: const Color(0xFFE94560),
|
|
||||||
padding:
|
|
||||||
const EdgeInsets.symmetric(horizontal: 40, vertical: 12),
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(25),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: const Text(
|
|
||||||
'REJOUER',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: Colors.white,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Restart the game
|
|
||||||
void _restartGame() {
|
|
||||||
setState(() {
|
|
||||||
score = 0;
|
|
||||||
lives = 3;
|
|
||||||
});
|
|
||||||
_loadNewPokemon();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Use hint (costs 5 points)
|
|
||||||
void _useHint() {
|
|
||||||
if (score >= 5 && !showHint) {
|
|
||||||
setState(() {
|
|
||||||
score -= 5;
|
|
||||||
showHint = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: const Color(0xFF1A1A2E),
|
|
||||||
appBar: AppBar(
|
|
||||||
backgroundColor: Colors.transparent,
|
|
||||||
elevation: 0,
|
|
||||||
leading: IconButton(
|
|
||||||
icon: const Icon(Icons.arrow_back, color: Colors.white),
|
|
||||||
onPressed: () => Navigator.pop(context),
|
|
||||||
),
|
|
||||||
title: const Text(
|
|
||||||
'POKÉGUESS',
|
|
||||||
style: TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
letterSpacing: 2,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
centerTitle: true,
|
|
||||||
),
|
|
||||||
body: SafeArea(
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(20),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
// Score and Lives row
|
|
||||||
_buildScoreAndLives(),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
|
|
||||||
// Pokémon silhouette
|
|
||||||
Expanded(
|
|
||||||
child: _buildPokemonDisplay(),
|
|
||||||
),
|
|
||||||
|
|
||||||
// Hint display
|
|
||||||
if (showHint && currentPokemon != null)
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 16),
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 20, vertical: 10),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: currentPokemon!.type1Color.withValues(alpha: 0.3),
|
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
border: Border.all(color: currentPokemon!.type1Color),
|
|
||||||
),
|
|
||||||
child: Text(
|
|
||||||
'Type: ${currentPokemon!.type1Formated}',
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// Input and buttons
|
|
||||||
_buildInputSection(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Build score and lives display
|
|
||||||
Widget _buildScoreAndLives() {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
// Score
|
|
||||||
Container(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 10),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Colors.white.withValues(alpha: 0.1),
|
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
const Icon(Icons.star, color: Colors.amber, size: 24),
|
|
||||||
const SizedBox(width: 8),
|
|
||||||
Text(
|
|
||||||
'$score',
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 22,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// Lives (hearts)
|
|
||||||
Row(
|
|
||||||
children: List.generate(3, (index) {
|
|
||||||
return Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 4),
|
|
||||||
child: Icon(
|
|
||||||
index < lives ? Icons.favorite : Icons.favorite_border,
|
|
||||||
color: const Color(0xFFE94560),
|
|
||||||
size: 32,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Build Pokémon silhouette or revealed image
|
|
||||||
Widget _buildPokemonDisplay() {
|
|
||||||
if (isLoading) {
|
|
||||||
return const Center(
|
|
||||||
child: CircularProgressIndicator(
|
|
||||||
color: Color(0xFFE94560),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentPokemon == null) {
|
|
||||||
return const Center(
|
|
||||||
child: Text(
|
|
||||||
'Erreur de chargement',
|
|
||||||
style: TextStyle(color: Colors.white),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
final String imageUrl =
|
|
||||||
isShiny ? currentPokemon!.shinyImageUrl : currentPokemon!.imageUrl;
|
|
||||||
|
|
||||||
Widget image = Image.network(
|
|
||||||
imageUrl,
|
|
||||||
width: 250,
|
|
||||||
height: 250,
|
|
||||||
fit: BoxFit.contain,
|
|
||||||
loadingBuilder: (context, child, loadingProgress) {
|
|
||||||
if (loadingProgress == null) return child;
|
|
||||||
return const SizedBox(
|
|
||||||
width: 250,
|
|
||||||
height: 250,
|
|
||||||
child: Center(
|
|
||||||
child: CircularProgressIndicator(color: Color(0xFFE94560)),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
errorBuilder: (context, error, stackTrace) {
|
|
||||||
return const SizedBox(
|
|
||||||
width: 250,
|
|
||||||
height: 250,
|
|
||||||
child: Center(
|
|
||||||
child: Icon(Icons.error, color: Colors.red, size: 50),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
// Apply silhouette filter if not revealed
|
|
||||||
if (!isRevealed) {
|
|
||||||
image = ColorFiltered(
|
|
||||||
colorFilter: const ColorFilter.mode(Colors.black, BlendMode.srcIn),
|
|
||||||
child: image,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add shiny sparkle effect if revealed and shiny
|
|
||||||
return Center(
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
children: [
|
|
||||||
// Glow effect when revealed
|
|
||||||
if (isRevealed)
|
|
||||||
Container(
|
|
||||||
width: 280,
|
|
||||||
height: 280,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
boxShadow: [
|
|
||||||
BoxShadow(
|
|
||||||
color: isShiny
|
|
||||||
? Colors.amber.withValues(alpha: 0.4)
|
|
||||||
: Colors.blueAccent.withValues(alpha: 0.3),
|
|
||||||
blurRadius: 40,
|
|
||||||
spreadRadius: 10,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
image,
|
|
||||||
// Shiny indicator
|
|
||||||
if (isShiny && isRevealed)
|
|
||||||
const Positioned(
|
|
||||||
top: 0,
|
|
||||||
right: 50,
|
|
||||||
child: Text(
|
|
||||||
'✨',
|
|
||||||
style: TextStyle(fontSize: 32),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Build input section with text field and buttons
|
|
||||||
Widget _buildInputSection() {
|
|
||||||
return Column(
|
|
||||||
children: [
|
|
||||||
// Text input
|
|
||||||
Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Colors.white.withValues(alpha: 0.1),
|
|
||||||
borderRadius: BorderRadius.circular(15),
|
|
||||||
),
|
|
||||||
child: TextField(
|
|
||||||
controller: _controller,
|
|
||||||
enabled: !isRevealed,
|
|
||||||
style: const TextStyle(color: Colors.white, fontSize: 18),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
textCapitalization: TextCapitalization.words,
|
|
||||||
decoration: const InputDecoration(
|
|
||||||
hintText: 'Nom du Pokémon...',
|
|
||||||
hintStyle: TextStyle(color: Colors.white38),
|
|
||||||
border: InputBorder.none,
|
|
||||||
contentPadding:
|
|
||||||
EdgeInsets.symmetric(horizontal: 20, vertical: 16),
|
|
||||||
),
|
|
||||||
onSubmitted: (_) => _validateGuess(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 16),
|
|
||||||
|
|
||||||
// Buttons row
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
// Hint button
|
|
||||||
Expanded(
|
|
||||||
child: ElevatedButton.icon(
|
|
||||||
onPressed:
|
|
||||||
score >= 5 && !showHint && !isRevealed ? _useHint : null,
|
|
||||||
icon: const Icon(Icons.lightbulb_outline, size: 20),
|
|
||||||
label: const Text('Indice (5 pts)'),
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: Colors.amber.withValues(alpha: 0.8),
|
|
||||||
foregroundColor: Colors.black87,
|
|
||||||
disabledBackgroundColor: Colors.grey.withValues(alpha: 0.3),
|
|
||||||
disabledForegroundColor: Colors.white38,
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 14),
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
|
|
||||||
// Validate button
|
|
||||||
Expanded(
|
|
||||||
flex: 1,
|
|
||||||
child: ElevatedButton(
|
|
||||||
onPressed: !isRevealed ? _validateGuess : null,
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: const Color(0xFFE94560),
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
disabledBackgroundColor: Colors.grey.withValues(alpha: 0.3),
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 14),
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: const Text(
|
|
||||||
'VALIDER',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
letterSpacing: 1,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,135 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
/// HomePage - Main menu for Pokéguess
|
|
||||||
/// Contains PLAY and POKEDEX navigation buttons
|
|
||||||
class HomePage extends StatelessWidget {
|
|
||||||
const HomePage({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Scaffold(
|
|
||||||
body: Container(
|
|
||||||
decoration: const BoxDecoration(
|
|
||||||
gradient: LinearGradient(
|
|
||||||
begin: Alignment.topCenter,
|
|
||||||
end: Alignment.bottomCenter,
|
|
||||||
colors: [
|
|
||||||
Color(0xFF1A1A2E),
|
|
||||||
Color(0xFF16213E),
|
|
||||||
Color(0xFF0F3460),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: SafeArea(
|
|
||||||
child: Center(
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
// Logo / Title
|
|
||||||
const Text(
|
|
||||||
'POKÉGUESS',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 48,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: Colors.white,
|
|
||||||
letterSpacing: 4,
|
|
||||||
shadows: [
|
|
||||||
Shadow(
|
|
||||||
blurRadius: 20,
|
|
||||||
color: Colors.blueAccent,
|
|
||||||
offset: Offset(0, 0),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
const Text(
|
|
||||||
'Devine le Pokémon !',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18,
|
|
||||||
color: Colors.white70,
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 80),
|
|
||||||
|
|
||||||
// PLAY Button
|
|
||||||
_MenuButton(
|
|
||||||
label: 'JOUER',
|
|
||||||
icon: Icons.play_arrow_rounded,
|
|
||||||
color: const Color(0xFFE94560),
|
|
||||||
onPressed: () => Navigator.pushNamed(context, '/game'),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 24),
|
|
||||||
|
|
||||||
// POKEDEX Button
|
|
||||||
_MenuButton(
|
|
||||||
label: 'POKÉDEX',
|
|
||||||
icon: Icons.catching_pokemon,
|
|
||||||
color: const Color(0xFF0F3460),
|
|
||||||
borderColor: Colors.white38,
|
|
||||||
onPressed: () => Navigator.pushNamed(context, '/pokedex'),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Reusable menu button with icon and gradient effect
|
|
||||||
class _MenuButton extends StatelessWidget {
|
|
||||||
final String label;
|
|
||||||
final IconData icon;
|
|
||||||
final Color color;
|
|
||||||
final Color? borderColor;
|
|
||||||
final VoidCallback onPressed;
|
|
||||||
|
|
||||||
const _MenuButton({
|
|
||||||
required this.label,
|
|
||||||
required this.icon,
|
|
||||||
required this.color,
|
|
||||||
required this.onPressed,
|
|
||||||
this.borderColor,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return SizedBox(
|
|
||||||
width: 220,
|
|
||||||
height: 60,
|
|
||||||
child: ElevatedButton(
|
|
||||||
onPressed: onPressed,
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: color,
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
elevation: 8,
|
|
||||||
shadowColor: color.withValues(alpha: 0.5),
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(30),
|
|
||||||
side: borderColor != null
|
|
||||||
? BorderSide(color: borderColor!, width: 2)
|
|
||||||
: BorderSide.none,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Icon(icon, size: 28),
|
|
||||||
const SizedBox(width: 12),
|
|
||||||
Text(
|
|
||||||
label,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
letterSpacing: 2,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,8 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import '../models/pokemon.dart';
|
import '../models/pokemon.dart';
|
||||||
import '../components/pokemon_tile.dart';
|
import '../components/pokemon_tile.dart';
|
||||||
import 'package:flutter/foundation.dart'
|
import 'package:flutter/foundation.dart' show kIsWeb; // Platform is not supported on web
|
||||||
show kIsWeb; // Platform is not supported on web
|
|
||||||
|
|
||||||
// Page de la liste des pokémons. Elle est appelée par la route "/". Elle affiche la liste des 151 premiers pokémons.
|
// Page de la liste des pokémons. Elle est appelée par la route "/". Elle affiche la liste des 151 premiers pokémons.
|
||||||
// Elle hérite de la classe StatefulWidget car elle a besoin de gérer un état (la liste des pokémons).
|
// Elle hérite de la classe StatefulWidget car elle a besoin de gérer un état (la liste des pokémons).
|
||||||
@ -14,6 +13,7 @@ class PokemonListPage extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _PokemonListPageState extends State<PokemonListPage> {
|
class _PokemonListPageState extends State<PokemonListPage> {
|
||||||
|
|
||||||
Widget _buildPokemonTile(BuildContext context, int index) {
|
Widget _buildPokemonTile(BuildContext context, int index) {
|
||||||
// On utilise un FutureBuilder pour afficher un pokémon à partir de son ID. L'index commençant à 0, on ajoute 1 pour le numéro du pokémon.
|
// On utilise un FutureBuilder pour afficher un pokémon à partir de son ID. L'index commençant à 0, on ajoute 1 pour le numéro du pokémon.
|
||||||
// Le FutureBuilder va permettre d'afficher un widget en fonction de l'état du Future
|
// Le FutureBuilder va permettre d'afficher un widget en fonction de l'état du Future
|
||||||
@ -51,21 +51,19 @@ class _PokemonListPageState extends State<PokemonListPage> {
|
|||||||
title: const Text('Liste des pokémons'),
|
title: const Text('Liste des pokémons'),
|
||||||
),
|
),
|
||||||
body: Center(
|
body: Center(
|
||||||
child: GridView.builder(
|
child: GridView.builder(
|
||||||
// Le GridView permet d'afficher une liste de widgets sous forme de grille
|
// Le GridView permet d'afficher une liste de widgets sous forme de grille
|
||||||
// On utilise le constructeur GridView.builder pour construire la grille
|
// On utilise le constructeur GridView.builder pour construire la grille
|
||||||
// Le GridView.builder prend en paramètre un itemCount qui correspond au nombre d'éléments à afficher
|
// Le GridView.builder prend en paramètre un itemCount qui correspond au nombre d'éléments à afficher
|
||||||
// Il prend aussi en paramètre un itemBuilder qui va permettre de construire chaque élément de la grille
|
// Il prend aussi en paramètre un itemBuilder qui va permettre de construire chaque élément de la grille
|
||||||
// Le GridView.builder prend aussi en paramètre un gridDelegate qui va permettre de définir le nombre de colonnes de la grille
|
// Le GridView.builder prend aussi en paramètre un gridDelegate qui va permettre de définir le nombre de colonnes de la grille
|
||||||
gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount(
|
gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount(
|
||||||
crossAxisCount: kIsWeb
|
crossAxisCount: kIsWeb ? 4 : 2, // On affiche 4 colonnes sur le web et 2 colonnes sur mobile
|
||||||
? 4
|
),
|
||||||
: 2, // On affiche 4 colonnes sur le web et 2 colonnes sur mobile
|
itemCount: 151, // On pourrait en mettre plus mais on va se limiter aux 151 premiers pokémons
|
||||||
),
|
itemBuilder: _buildPokemonTile,
|
||||||
itemCount:
|
)
|
||||||
151, // On pourrait en mettre plus mais on va se limiter aux 151 premiers pokémons
|
),
|
||||||
itemBuilder: _buildPokemonTile,
|
|
||||||
)),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,33 +0,0 @@
|
|||||||
PODS:
|
|
||||||
- FlutterMacOS (1.0.0)
|
|
||||||
- FMDB (2.7.12):
|
|
||||||
- FMDB/standard (= 2.7.12)
|
|
||||||
- FMDB/Core (2.7.12)
|
|
||||||
- FMDB/standard (2.7.12):
|
|
||||||
- FMDB/Core
|
|
||||||
- sqflite (0.0.2):
|
|
||||||
- FlutterMacOS
|
|
||||||
- FMDB (>= 2.7.5)
|
|
||||||
|
|
||||||
DEPENDENCIES:
|
|
||||||
- FlutterMacOS (from `Flutter/ephemeral`)
|
|
||||||
- sqflite (from `Flutter/ephemeral/.symlinks/plugins/sqflite/macos`)
|
|
||||||
|
|
||||||
SPEC REPOS:
|
|
||||||
trunk:
|
|
||||||
- FMDB
|
|
||||||
|
|
||||||
EXTERNAL SOURCES:
|
|
||||||
FlutterMacOS:
|
|
||||||
:path: Flutter/ephemeral
|
|
||||||
sqflite:
|
|
||||||
:path: Flutter/ephemeral/.symlinks/plugins/sqflite/macos
|
|
||||||
|
|
||||||
SPEC CHECKSUMS:
|
|
||||||
FlutterMacOS: 8f6f14fa908a6fb3fba0cd85dbd81ec4b251fb24
|
|
||||||
FMDB: 728731dd336af3936ce00f91d9d8495f5718a0e6
|
|
||||||
sqflite: c73556b2499b92f0b6e6946abe4a4084510cdf90
|
|
||||||
|
|
||||||
PODFILE CHECKSUM: 7eb978b976557c8c1cd717d8185ec483fd090a82
|
|
||||||
|
|
||||||
COCOAPODS: 1.16.2
|
|
||||||
@ -21,14 +21,12 @@
|
|||||||
/* End PBXAggregateTarget section */
|
/* End PBXAggregateTarget section */
|
||||||
|
|
||||||
/* Begin PBXBuildFile section */
|
/* Begin PBXBuildFile section */
|
||||||
2F63F588D69E6E84EF178DDE /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 38A997E4AF06585806D63A93 /* Pods_RunnerTests.framework */; };
|
|
||||||
331C80D8294CF71000263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C80D7294CF71000263BE5 /* RunnerTests.swift */; };
|
331C80D8294CF71000263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C80D7294CF71000263BE5 /* RunnerTests.swift */; };
|
||||||
335BBD1B22A9A15E00E9071D /* GeneratedPluginRegistrant.swift in Sources */ = {isa = PBXBuildFile; fileRef = 335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */; };
|
335BBD1B22A9A15E00E9071D /* GeneratedPluginRegistrant.swift in Sources */ = {isa = PBXBuildFile; fileRef = 335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */; };
|
||||||
33CC10F12044A3C60003C045 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC10F02044A3C60003C045 /* AppDelegate.swift */; };
|
33CC10F12044A3C60003C045 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC10F02044A3C60003C045 /* AppDelegate.swift */; };
|
||||||
33CC10F32044A3C60003C045 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F22044A3C60003C045 /* Assets.xcassets */; };
|
33CC10F32044A3C60003C045 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F22044A3C60003C045 /* Assets.xcassets */; };
|
||||||
33CC10F62044A3C60003C045 /* MainMenu.xib in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F42044A3C60003C045 /* MainMenu.xib */; };
|
33CC10F62044A3C60003C045 /* MainMenu.xib in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F42044A3C60003C045 /* MainMenu.xib */; };
|
||||||
33CC11132044BFA00003C045 /* MainFlutterWindow.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC11122044BFA00003C045 /* MainFlutterWindow.swift */; };
|
33CC11132044BFA00003C045 /* MainFlutterWindow.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC11122044BFA00003C045 /* MainFlutterWindow.swift */; };
|
||||||
BC0787830DD8D138B3AFA3FE /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 1ADD6793C62B3D4B8AF3C642 /* Pods_Runner.framework */; };
|
|
||||||
/* End PBXBuildFile section */
|
/* End PBXBuildFile section */
|
||||||
|
|
||||||
/* Begin PBXContainerItemProxy section */
|
/* Begin PBXContainerItemProxy section */
|
||||||
@ -62,12 +60,11 @@
|
|||||||
/* End PBXCopyFilesBuildPhase section */
|
/* End PBXCopyFilesBuildPhase section */
|
||||||
|
|
||||||
/* Begin PBXFileReference section */
|
/* Begin PBXFileReference section */
|
||||||
1ADD6793C62B3D4B8AF3C642 /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
|
||||||
331C80D5294CF71000263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
331C80D5294CF71000263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
331C80D7294CF71000263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; };
|
331C80D7294CF71000263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; };
|
||||||
333000ED22D3DE5D00554162 /* Warnings.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Warnings.xcconfig; sourceTree = "<group>"; };
|
333000ED22D3DE5D00554162 /* Warnings.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Warnings.xcconfig; sourceTree = "<group>"; };
|
||||||
335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GeneratedPluginRegistrant.swift; sourceTree = "<group>"; };
|
335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GeneratedPluginRegistrant.swift; sourceTree = "<group>"; };
|
||||||
33CC10ED2044A3C60003C045 /* pokedex.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = pokedex.app; sourceTree = BUILT_PRODUCTS_DIR; };
|
33CC10ED2044A3C60003C045 /* pokedex.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "pokedex.app"; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
33CC10F02044A3C60003C045 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; };
|
33CC10F02044A3C60003C045 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; };
|
||||||
33CC10F22044A3C60003C045 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Assets.xcassets; path = Runner/Assets.xcassets; sourceTree = "<group>"; };
|
33CC10F22044A3C60003C045 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Assets.xcassets; path = Runner/Assets.xcassets; sourceTree = "<group>"; };
|
||||||
33CC10F52044A3C60003C045 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/MainMenu.xib; sourceTree = "<group>"; };
|
33CC10F52044A3C60003C045 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/MainMenu.xib; sourceTree = "<group>"; };
|
||||||
@ -79,15 +76,8 @@
|
|||||||
33E51913231747F40026EE4D /* DebugProfile.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = DebugProfile.entitlements; sourceTree = "<group>"; };
|
33E51913231747F40026EE4D /* DebugProfile.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = DebugProfile.entitlements; sourceTree = "<group>"; };
|
||||||
33E51914231749380026EE4D /* Release.entitlements */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.entitlements; path = Release.entitlements; sourceTree = "<group>"; };
|
33E51914231749380026EE4D /* Release.entitlements */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.entitlements; path = Release.entitlements; sourceTree = "<group>"; };
|
||||||
33E5194F232828860026EE4D /* AppInfo.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = AppInfo.xcconfig; sourceTree = "<group>"; };
|
33E5194F232828860026EE4D /* AppInfo.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = AppInfo.xcconfig; sourceTree = "<group>"; };
|
||||||
38A997E4AF06585806D63A93 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
|
||||||
7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Release.xcconfig; sourceTree = "<group>"; };
|
7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Release.xcconfig; sourceTree = "<group>"; };
|
||||||
8425D271D17EE83BF167A836 /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
86FC039ADCC2CFF33032C226 /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = Debug.xcconfig; sourceTree = "<group>"; };
|
9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = Debug.xcconfig; sourceTree = "<group>"; };
|
||||||
A60151594AB2579C4697BCA2 /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
AE1BE7E94A2D9EB80387B250 /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
E791602D5E2611FF301C4F22 /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
FACE8614944708AF3373EFE5 /* Pods-RunnerTests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.release.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.release.xcconfig"; sourceTree = "<group>"; };
|
|
||||||
/* End PBXFileReference section */
|
/* End PBXFileReference section */
|
||||||
|
|
||||||
/* Begin PBXFrameworksBuildPhase section */
|
/* Begin PBXFrameworksBuildPhase section */
|
||||||
@ -95,7 +85,6 @@
|
|||||||
isa = PBXFrameworksBuildPhase;
|
isa = PBXFrameworksBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
files = (
|
files = (
|
||||||
2F63F588D69E6E84EF178DDE /* Pods_RunnerTests.framework in Frameworks */,
|
|
||||||
);
|
);
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
};
|
};
|
||||||
@ -103,7 +92,6 @@
|
|||||||
isa = PBXFrameworksBuildPhase;
|
isa = PBXFrameworksBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
files = (
|
files = (
|
||||||
BC0787830DD8D138B3AFA3FE /* Pods_Runner.framework in Frameworks */,
|
|
||||||
);
|
);
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
};
|
};
|
||||||
@ -137,7 +125,6 @@
|
|||||||
331C80D6294CF71000263BE5 /* RunnerTests */,
|
331C80D6294CF71000263BE5 /* RunnerTests */,
|
||||||
33CC10EE2044A3C60003C045 /* Products */,
|
33CC10EE2044A3C60003C045 /* Products */,
|
||||||
D73912EC22F37F3D000D13A0 /* Frameworks */,
|
D73912EC22F37F3D000D13A0 /* Frameworks */,
|
||||||
B9231BE3DD2D9C2F0FE7683B /* Pods */,
|
|
||||||
);
|
);
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
};
|
};
|
||||||
@ -185,25 +172,9 @@
|
|||||||
path = Runner;
|
path = Runner;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
};
|
};
|
||||||
B9231BE3DD2D9C2F0FE7683B /* Pods */ = {
|
|
||||||
isa = PBXGroup;
|
|
||||||
children = (
|
|
||||||
A60151594AB2579C4697BCA2 /* Pods-Runner.debug.xcconfig */,
|
|
||||||
8425D271D17EE83BF167A836 /* Pods-Runner.release.xcconfig */,
|
|
||||||
86FC039ADCC2CFF33032C226 /* Pods-Runner.profile.xcconfig */,
|
|
||||||
E791602D5E2611FF301C4F22 /* Pods-RunnerTests.debug.xcconfig */,
|
|
||||||
FACE8614944708AF3373EFE5 /* Pods-RunnerTests.release.xcconfig */,
|
|
||||||
AE1BE7E94A2D9EB80387B250 /* Pods-RunnerTests.profile.xcconfig */,
|
|
||||||
);
|
|
||||||
name = Pods;
|
|
||||||
path = Pods;
|
|
||||||
sourceTree = "<group>";
|
|
||||||
};
|
|
||||||
D73912EC22F37F3D000D13A0 /* Frameworks */ = {
|
D73912EC22F37F3D000D13A0 /* Frameworks */ = {
|
||||||
isa = PBXGroup;
|
isa = PBXGroup;
|
||||||
children = (
|
children = (
|
||||||
1ADD6793C62B3D4B8AF3C642 /* Pods_Runner.framework */,
|
|
||||||
38A997E4AF06585806D63A93 /* Pods_RunnerTests.framework */,
|
|
||||||
);
|
);
|
||||||
name = Frameworks;
|
name = Frameworks;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@ -215,7 +186,6 @@
|
|||||||
isa = PBXNativeTarget;
|
isa = PBXNativeTarget;
|
||||||
buildConfigurationList = 331C80DE294CF71000263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */;
|
buildConfigurationList = 331C80DE294CF71000263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */;
|
||||||
buildPhases = (
|
buildPhases = (
|
||||||
6E388177BD91FCED76707214 /* [CP] Check Pods Manifest.lock */,
|
|
||||||
331C80D1294CF70F00263BE5 /* Sources */,
|
331C80D1294CF70F00263BE5 /* Sources */,
|
||||||
331C80D2294CF70F00263BE5 /* Frameworks */,
|
331C80D2294CF70F00263BE5 /* Frameworks */,
|
||||||
331C80D3294CF70F00263BE5 /* Resources */,
|
331C80D3294CF70F00263BE5 /* Resources */,
|
||||||
@ -234,13 +204,11 @@
|
|||||||
isa = PBXNativeTarget;
|
isa = PBXNativeTarget;
|
||||||
buildConfigurationList = 33CC10FB2044A3C60003C045 /* Build configuration list for PBXNativeTarget "Runner" */;
|
buildConfigurationList = 33CC10FB2044A3C60003C045 /* Build configuration list for PBXNativeTarget "Runner" */;
|
||||||
buildPhases = (
|
buildPhases = (
|
||||||
29B2F3029551E8A809B61379 /* [CP] Check Pods Manifest.lock */,
|
|
||||||
33CC10E92044A3C60003C045 /* Sources */,
|
33CC10E92044A3C60003C045 /* Sources */,
|
||||||
33CC10EA2044A3C60003C045 /* Frameworks */,
|
33CC10EA2044A3C60003C045 /* Frameworks */,
|
||||||
33CC10EB2044A3C60003C045 /* Resources */,
|
33CC10EB2044A3C60003C045 /* Resources */,
|
||||||
33CC110E2044A8840003C045 /* Bundle Framework */,
|
33CC110E2044A8840003C045 /* Bundle Framework */,
|
||||||
3399D490228B24CF009A79C7 /* ShellScript */,
|
3399D490228B24CF009A79C7 /* ShellScript */,
|
||||||
C7326370B9BF34B0BE156E06 /* [CP] Embed Pods Frameworks */,
|
|
||||||
);
|
);
|
||||||
buildRules = (
|
buildRules = (
|
||||||
);
|
);
|
||||||
@ -259,7 +227,7 @@
|
|||||||
isa = PBXProject;
|
isa = PBXProject;
|
||||||
attributes = {
|
attributes = {
|
||||||
LastSwiftUpdateCheck = 0920;
|
LastSwiftUpdateCheck = 0920;
|
||||||
LastUpgradeCheck = 1510;
|
LastUpgradeCheck = 1430;
|
||||||
ORGANIZATIONNAME = "";
|
ORGANIZATIONNAME = "";
|
||||||
TargetAttributes = {
|
TargetAttributes = {
|
||||||
331C80D4294CF70F00263BE5 = {
|
331C80D4294CF70F00263BE5 = {
|
||||||
@ -322,28 +290,6 @@
|
|||||||
/* End PBXResourcesBuildPhase section */
|
/* End PBXResourcesBuildPhase section */
|
||||||
|
|
||||||
/* Begin PBXShellScriptBuildPhase section */
|
/* Begin PBXShellScriptBuildPhase section */
|
||||||
29B2F3029551E8A809B61379 /* [CP] Check Pods Manifest.lock */ = {
|
|
||||||
isa = PBXShellScriptBuildPhase;
|
|
||||||
buildActionMask = 2147483647;
|
|
||||||
files = (
|
|
||||||
);
|
|
||||||
inputFileListPaths = (
|
|
||||||
);
|
|
||||||
inputPaths = (
|
|
||||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
|
||||||
"${PODS_ROOT}/Manifest.lock",
|
|
||||||
);
|
|
||||||
name = "[CP] Check Pods Manifest.lock";
|
|
||||||
outputFileListPaths = (
|
|
||||||
);
|
|
||||||
outputPaths = (
|
|
||||||
"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt",
|
|
||||||
);
|
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
|
||||||
shellPath = /bin/sh;
|
|
||||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
|
||||||
showEnvVarsInLog = 0;
|
|
||||||
};
|
|
||||||
3399D490228B24CF009A79C7 /* ShellScript */ = {
|
3399D490228B24CF009A79C7 /* ShellScript */ = {
|
||||||
isa = PBXShellScriptBuildPhase;
|
isa = PBXShellScriptBuildPhase;
|
||||||
alwaysOutOfDate = 1;
|
alwaysOutOfDate = 1;
|
||||||
@ -382,45 +328,6 @@
|
|||||||
shellPath = /bin/sh;
|
shellPath = /bin/sh;
|
||||||
shellScript = "\"$FLUTTER_ROOT\"/packages/flutter_tools/bin/macos_assemble.sh && touch Flutter/ephemeral/tripwire";
|
shellScript = "\"$FLUTTER_ROOT\"/packages/flutter_tools/bin/macos_assemble.sh && touch Flutter/ephemeral/tripwire";
|
||||||
};
|
};
|
||||||
6E388177BD91FCED76707214 /* [CP] Check Pods Manifest.lock */ = {
|
|
||||||
isa = PBXShellScriptBuildPhase;
|
|
||||||
buildActionMask = 2147483647;
|
|
||||||
files = (
|
|
||||||
);
|
|
||||||
inputFileListPaths = (
|
|
||||||
);
|
|
||||||
inputPaths = (
|
|
||||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
|
||||||
"${PODS_ROOT}/Manifest.lock",
|
|
||||||
);
|
|
||||||
name = "[CP] Check Pods Manifest.lock";
|
|
||||||
outputFileListPaths = (
|
|
||||||
);
|
|
||||||
outputPaths = (
|
|
||||||
"$(DERIVED_FILE_DIR)/Pods-RunnerTests-checkManifestLockResult.txt",
|
|
||||||
);
|
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
|
||||||
shellPath = /bin/sh;
|
|
||||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
|
||||||
showEnvVarsInLog = 0;
|
|
||||||
};
|
|
||||||
C7326370B9BF34B0BE156E06 /* [CP] Embed Pods Frameworks */ = {
|
|
||||||
isa = PBXShellScriptBuildPhase;
|
|
||||||
buildActionMask = 2147483647;
|
|
||||||
files = (
|
|
||||||
);
|
|
||||||
inputFileListPaths = (
|
|
||||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-input-files.xcfilelist",
|
|
||||||
);
|
|
||||||
name = "[CP] Embed Pods Frameworks";
|
|
||||||
outputFileListPaths = (
|
|
||||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-output-files.xcfilelist",
|
|
||||||
);
|
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
|
||||||
shellPath = /bin/sh;
|
|
||||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n";
|
|
||||||
showEnvVarsInLog = 0;
|
|
||||||
};
|
|
||||||
/* End PBXShellScriptBuildPhase section */
|
/* End PBXShellScriptBuildPhase section */
|
||||||
|
|
||||||
/* Begin PBXSourcesBuildPhase section */
|
/* Begin PBXSourcesBuildPhase section */
|
||||||
@ -472,7 +379,6 @@
|
|||||||
/* Begin XCBuildConfiguration section */
|
/* Begin XCBuildConfiguration section */
|
||||||
331C80DB294CF71000263BE5 /* Debug */ = {
|
331C80DB294CF71000263BE5 /* Debug */ = {
|
||||||
isa = XCBuildConfiguration;
|
isa = XCBuildConfiguration;
|
||||||
baseConfigurationReference = E791602D5E2611FF301C4F22 /* Pods-RunnerTests.debug.xcconfig */;
|
|
||||||
buildSettings = {
|
buildSettings = {
|
||||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||||
CURRENT_PROJECT_VERSION = 1;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
@ -487,7 +393,6 @@
|
|||||||
};
|
};
|
||||||
331C80DC294CF71000263BE5 /* Release */ = {
|
331C80DC294CF71000263BE5 /* Release */ = {
|
||||||
isa = XCBuildConfiguration;
|
isa = XCBuildConfiguration;
|
||||||
baseConfigurationReference = FACE8614944708AF3373EFE5 /* Pods-RunnerTests.release.xcconfig */;
|
|
||||||
buildSettings = {
|
buildSettings = {
|
||||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||||
CURRENT_PROJECT_VERSION = 1;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
@ -502,7 +407,6 @@
|
|||||||
};
|
};
|
||||||
331C80DD294CF71000263BE5 /* Profile */ = {
|
331C80DD294CF71000263BE5 /* Profile */ = {
|
||||||
isa = XCBuildConfiguration;
|
isa = XCBuildConfiguration;
|
||||||
baseConfigurationReference = AE1BE7E94A2D9EB80387B250 /* Pods-RunnerTests.profile.xcconfig */;
|
|
||||||
buildSettings = {
|
buildSettings = {
|
||||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||||
CURRENT_PROJECT_VERSION = 1;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<Scheme
|
<Scheme
|
||||||
LastUpgradeVersion = "1510"
|
LastUpgradeVersion = "1430"
|
||||||
version = "1.3">
|
version = "1.3">
|
||||||
<BuildAction
|
<BuildAction
|
||||||
parallelizeBuildables = "YES"
|
parallelizeBuildables = "YES"
|
||||||
@ -59,7 +59,6 @@
|
|||||||
ignoresPersistentStateOnLaunch = "NO"
|
ignoresPersistentStateOnLaunch = "NO"
|
||||||
debugDocumentVersioning = "YES"
|
debugDocumentVersioning = "YES"
|
||||||
debugServiceExtension = "internal"
|
debugServiceExtension = "internal"
|
||||||
enableGPUValidationMode = "1"
|
|
||||||
allowLocationSimulation = "YES">
|
allowLocationSimulation = "YES">
|
||||||
<BuildableProductRunnable
|
<BuildableProductRunnable
|
||||||
runnableDebuggingMode = "0">
|
runnableDebuggingMode = "0">
|
||||||
|
|||||||
@ -4,7 +4,4 @@
|
|||||||
<FileRef
|
<FileRef
|
||||||
location = "group:Runner.xcodeproj">
|
location = "group:Runner.xcodeproj">
|
||||||
</FileRef>
|
</FileRef>
|
||||||
<FileRef
|
|
||||||
location = "group:Pods/Pods.xcodeproj">
|
|
||||||
</FileRef>
|
|
||||||
</Workspace>
|
</Workspace>
|
||||||
|
|||||||
@ -1,13 +1,9 @@
|
|||||||
import Cocoa
|
import Cocoa
|
||||||
import FlutterMacOS
|
import FlutterMacOS
|
||||||
|
|
||||||
@main
|
@NSApplicationMain
|
||||||
class AppDelegate: FlutterAppDelegate {
|
class AppDelegate: FlutterAppDelegate {
|
||||||
override func applicationShouldTerminateAfterLastWindowClosed(_ sender: NSApplication) -> Bool {
|
override func applicationShouldTerminateAfterLastWindowClosed(_ sender: NSApplication) -> Bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
override func applicationSupportsSecureRestorableState(_ app: NSApplication) -> Bool {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -7,8 +7,7 @@
|
|||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
import 'package:flutter_test/flutter_test.dart';
|
||||||
|
import 'package:pokeguess/main.dart';
|
||||||
import 'package:pokedex/main.dart';
|
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
testWidgets('Counter increments smoke test', (WidgetTester tester) async {
|
testWidgets('Counter increments smoke test', (WidgetTester tester) async {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user