Flutter - Guide rapide
En général, le développement d'une application mobile est une tâche complexe et difficile. Il existe de nombreux frameworks disponibles pour développer une application mobile. Android fournit un framework natif basé sur le langage Java et iOS fournit un framework natif basé sur le langage Objective-C / Swift.
Cependant, pour développer une application prenant en charge les deux systèmes d'exploitation, nous devons coder dans deux langues différentes en utilisant deux frameworks différents. Pour aider à surmonter cette complexité, il existe des frameworks mobiles prenant en charge les deux OS. Ces frameworks vont du simple framework d'application mobile hybride basé sur HTML (qui utilise HTML pour l'interface utilisateur et JavaScript pour la logique d'application) au framework spécifique à un langage complexe (qui fait le gros du travail de conversion du code en code natif). Indépendamment de leur simplicité ou de leur complexité, ces cadres présentent toujours de nombreux inconvénients, l'un des principaux inconvénients étant leur faible performance.
Dans ce scénario, Flutter - un framework simple et haute performance basé sur le langage Dart, fournit des performances élevées en rendant l'interface utilisateur directement dans le canevas du système d'exploitation plutôt que via un framework natif.
Flutter propose également de nombreux widgets (UI) prêts à l'emploi pour créer une application moderne. Ces widgets sont optimisés pour l'environnement mobile et la conception de l'application à l'aide de widgets est aussi simple que la conception HTML.
Pour être précis, l'application Flutter est elle-même un widget. Les widgets Flutter prennent également en charge les animations et les gestes. La logique d'application est basée sur la programmation réactive. Le widget peut éventuellement avoir un état. En modifiant l'état du widget, Flutter comparera automatiquement (programmation réactive) l'état du widget (ancien et nouveau) et restituera le widget avec uniquement les modifications nécessaires au lieu de restituer le widget entier.
Nous discuterons de l'architecture complète dans les prochains chapitres.
Caractéristiques de Flutter
Le framework Flutter offre les fonctionnalités suivantes aux développeurs -
Cadre moderne et réactif.
Utilise le langage de programmation Dart et il est très facile à apprendre.
Développement rapide.
Interfaces utilisateur belles et fluides.
Énorme catalogue de widgets.
Exécute la même interface utilisateur pour plusieurs plates-formes.
Application haute performance.
Avantages de Flutter
Flutter est livré avec des widgets magnifiques et personnalisables pour des performances élevées et une application mobile exceptionnelle. Il répond à tous les besoins et exigences personnalisés. Outre ceux-ci, Flutter offre de nombreux autres avantages comme mentionné ci-dessous -
Dart dispose d'un vaste référentiel de progiciels qui vous permet d'étendre les capacités de votre application.
Les développeurs doivent écrire une seule base de code pour les deux applications (plates-formes Android et iOS). Flutter pourrait également être étendu à d'autres plates-formes à l'avenir.
Flutter nécessite moins de tests. En raison de sa base de code unique, il suffit d'écrire des tests automatisés une fois pour les deux plates-formes.
La simplicité de Flutter en fait un bon candidat pour un développement rapide. Sa capacité de personnalisation et son extensibilité le rendent encore plus puissant.
Avec Flutter, les développeurs ont un contrôle total sur les widgets et leur mise en page.
Flutter propose d'excellents outils de développement, avec une recharge à chaud incroyable.
Inconvénients de Flutter
Malgré ses nombreux avantages, le flutter présente les inconvénients suivants -
Comme il est codé en langage Dart, un développeur doit apprendre un nouveau langage (bien qu'il soit facile à apprendre).
Le cadre moderne tente de séparer autant que possible la logique et l'interface utilisateur mais, dans Flutter, l'interface utilisateur et la logique sont mélangées. Nous pouvons surmonter cela en utilisant un codage intelligent et en utilisant un module de haut niveau pour séparer l'interface utilisateur et la logique.
Flutter est encore un autre cadre pour créer une application mobile. Les développeurs ont du mal à choisir les bons outils de développement dans un segment extrêmement peuplé.
Ce chapitre vous guidera à travers l'installation de Flutter sur votre ordinateur local en détail.
Installation sous Windows
Dans cette section, voyons comment installer le SDK Flutter et ses exigences dans un système Windows.
Step 1 - Aller à l'URL,https://flutter.dev/docs/get-started/install/windowset téléchargez le dernier SDK Flutter. Depuis avril 2019, la version est 1.2.1 et le fichier est flutter_windows_v1.2.1-stable.zip.
Step 2 - Décompressez l'archive zip dans un dossier, dites C: \ flutter \
Step 3 - Mettez à jour le chemin du système pour inclure le répertoire flutter bin.
Step 4 - Flutter fournit un outil, Flutter Doctor pour vérifier que toutes les exigences du développement de flutter sont satisfaites.
flutter doctor
Step 5 - L'exécution de la commande ci-dessus analysera le système et affichera son rapport comme indiqué ci-dessous -
Doctor summary (to see all details, run flutter doctor -v):
[√] Flutter (Channel stable, v1.2.1, on Microsoft Windows [Version
10.0.17134.706], locale en-US)
[√] Android toolchain - develop for Android devices (Android SDK version
28.0.3)
[√] Android Studio (version 3.2)
[√] VS Code, 64-bit edition (version 1.29.1)
[!] Connected device
! No devices available
! Doctor found issues in 1 category.
Le rapport indique que tous les outils de développement sont disponibles mais que l'appareil n'est pas connecté. Nous pouvons résoudre ce problème en connectant un appareil Android via USB ou en démarrant un émulateur Android.
Step 6 - Installez le dernier SDK Android, s'il est signalé par Flutter Doctor
Step 7 - Installez le dernier Android Studio, s'il est signalé par Flutter Doctor
Step 8 - Démarrez un émulateur Android ou connectez un véritable appareil Android au système.
Step 9- Installez le plugin Flutter et Dart pour Android Studio. Il fournit un modèle de démarrage pour créer une nouvelle application Flutter, une option pour exécuter et déboguer l'application Flutter dans le studio Android lui-même, etc.,
Ouvrez Android Studio.
Cliquez sur Fichier → Paramètres → Plugins.
Sélectionnez le plugin Flutter et cliquez sur Installer.
Cliquez sur Oui lorsque vous êtes invité à installer le plugin Dart.
Redémarrez le studio Android.
Installation sous MacOS
Pour installer Flutter sur MacOS, vous devrez suivre les étapes suivantes -
Step 1 - Aller à l'URL,https://flutter.dev/docs/get-started/install/macoset téléchargez le dernier SDK Flutter. Depuis avril 2019, la version est 1.2.1 et le fichier est flutter_macos_v1.2.1- stable.zip.
Step 2 - Décompressez l'archive zip dans un dossier, dites / chemin / vers / flutter
Step 3 - Mettez à jour le chemin du système pour inclure le répertoire flutter bin (dans le fichier ~ / .bashrc).
> export PATH = "$PATH:/path/to/flutter/bin"
Step 4 - Activez le chemin mis à jour dans la session en cours à l'aide de la commande ci-dessous, puis vérifiez-le également.
source ~/.bashrc
source $HOME/.bash_profile
echo $PATH
Flutter fournit un outil, flutter doctor, pour vérifier que toutes les exigences du développement de flutter sont satisfaites. Il est similaire à l'homologue Windows.
Step 5 - Installez le dernier XCode, s'il est signalé par le médecin de flutter
Step 6 - Installez le dernier SDK Android, si signalé par Flutter Doctor
Step 7 - Installez le dernier Android Studio, si signalé par Flutter Doctor
Step 8 - Démarrez un émulateur Android ou connectez un véritable appareil Android au système pour développer une application Android.
Step 9 - Ouvrez le simulateur iOS ou connectez un vrai appareil iPhone au système pour développer une application iOS.
Step 10- Installez le plugin Flutter et Dart pour Android Studio. Il fournit le modèle de démarrage pour créer une nouvelle application Flutter, une option pour exécuter et déboguer l'application Flutter dans le studio Android lui-même, etc.,
Ouvrez Android Studio
Cliquez sur Preferences → Plugins
Sélectionnez le plugin Flutter et cliquez sur Installer
Cliquez sur Oui lorsque vous êtes invité à installer le plugin Dart.
Redémarrez le studio Android.
Dans ce chapitre, créons une application Flutter simple pour comprendre les bases de la création d'une application Flutter dans Android Studio.
Step 1 - Ouvrez Android Studio
Step 2- Créer un projet Flutter. Pour cela, cliquez surFile → New → New Flutter Project
Step 3- Sélectionnez Application Flutter. Pour cela, sélectionnezFlutter Application et cliquez Next.
Step 4 - Configurez l'application comme ci-dessous et cliquez sur Next.
Nom du projet: hello_app
Chemin du SDK Flutter: <path_to_flutter_sdk>
Emplacement du projet: <path_to_project_folder>
La description: Flutter based hello world application
Step 5 - Configurer le projet.
Définissez le domaine de l'entreprise comme flutterapp.tutorialspoint.com et cliquez Finish.
Step 6 - Entrez le domaine de l'entreprise.
Android Studio crée une application de flutter entièrement fonctionnelle avec un minimum de fonctionnalités. Vérifions la structure de l'application puis modifions le code pour faire notre tâche.
La structure de l'application et son objectif sont les suivants -
Différents composants de la structure de l'application sont expliqués ici -
android - Code source généré automatiquement pour créer une application Android
ios - Code source généré automatiquement pour créer une application iOS
lib - Dossier principal contenant le code Dart écrit à l'aide du framework Flutter
ib/main.dart - Point d'entrée de l'application Flutter
test - Dossier contenant le code Dart pour tester l'application Flutter
test/widget_test.dart - Exemple de code
.gitignore - Fichier de contrôle de version Git
.metadata - généré automatiquement par les outils de flutter
.packages - généré automatiquement pour suivre les paquets de flutter
.iml - fichier de projet utilisé par le studio Android
pubspec.yaml - Utilisé par Pub, Gestionnaire de paquets Flutter
pubspec.lock - Généré automatiquement par le gestionnaire de paquets Flutter, Pub
README.md - Fichier de description du projet écrit au format Markdown
Step 7- Remplacez le code de fléchettes dans le fichier lib / main.dart par le code ci-dessous -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Hello World Demo Application',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Home page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.title),
),
body: Center(
child:
Text(
'Hello World',
)
),
);
}
}
Comprenons le code de la fléchette ligne par ligne.
Line 1- importe le paquet flutter, matériel . Le matériau est un package flottant pour créer une interface utilisateur conformément aux directives de conception de matériau spécifiées par Android.
Line 3- C'est le point d'entrée de l'application Flutter. Appelle la fonction runApp et lui transmet un objet de la classe MyApp . Le but de la fonction runApp est d'attacher le widget donné à l'écran.
Line 5-17- Widget est utilisé pour créer une interface utilisateur dans le cadre de flutter. StatelessWidget est un widget, qui ne maintient aucun état du widget. MyApp étend StatelessWidget et remplace sa méthode de génération . Le but de la méthode de génération est de créer une partie de l'interface utilisateur de l'application. Ici, la méthode de construction utilise MaterialApp , un widget pour créer l'interface utilisateur de niveau racine de l'application. Il a trois propriétés: le titre, le thème et la maison .
title est le titre de l'application
theme est le thème du widget. Ici, nous définissons le bleu comme couleur globale de l'application à l'aide de la classe ThemeData et de sa propriété, primarySwatch .
home est l'interface utilisateur interne de l'application, que nous définissons un autre widget, MyHomePage
Line 19 - 38- MyHomePage est identique à MyApp sauf qu'il renvoie le widget Scaffold . Scaffold est un widget de niveau supérieur à côté du widget MaterialApp utilisé pour créer une conception matérielle conforme à l'interface utilisateur. Il a deux propriétés importantes, appBar pour afficher l'en-tête de l'application et le corps pour afficher le contenu réel de l'application. AppBar est un autre widget pour rendre l'en-tête de l'application et nous l'avons utilisé dans la propriété appBar . Dans la propriété du corps , nous avons utilisé le widget Centre , qui le centre du widget enfant. Le texte est le dernier widget le plus interne pour afficher le texte et il est affiché au centre de l'écran.
Step 8 - Maintenant, lancez l'application en utilisant, Run → Run main.dart
Step 9 - Enfin, la sortie de l'application est la suivante -
Dans ce chapitre, parlons de l'architecture du framework Flutter.
Widgets
Le concept de base du framework Flutter est In Flutter, Everything is a widget. Les widgets sont essentiellement des composants d'interface utilisateur utilisés pour créer l'interface utilisateur de l'application.
Dans Flutter , l'application est elle-même un widget. L'application est le widget de niveau supérieur et son interface utilisateur est construite à l'aide d'un ou plusieurs enfants (widgets), qui à nouveau construits à l'aide de ses widgets enfants. Cecomposability fonctionnalité nous aide à créer une interface utilisateur de toute complexité.
Par exemple, la hiérarchie des widgets de l'application hello world (créée dans le chapitre précédent) est telle que spécifiée dans le diagramme suivant -
Ici, les points suivants méritent d'être notés -
MyApp est le widget créé par l'utilisateur et il est construit à l'aide du widget natif Flutter, MaterialApp .
MaterialApp a une propriété home pour spécifier l'interface utilisateur de la page d'accueil, qui est à nouveau un widget créé par l'utilisateur, MyHomePage .
MyHomePage est construit à l'aide d'un autre widget natif de flottement, Scaffold
Scaffold a deux propriétés - body et appBar
body est utilisé pour spécifier son interface utilisateur principale et appBar est utilisé pour spécifier son interface utilisateur d'en-tête
L'interface utilisateur d'en-tête est construite à l'aide d'un widget natif de flutter, l' AppBar et l' interface utilisateur du corps sont créées à l' aide du widget Center .
Le widget Centre a une propriété, Enfant , qui fait référence au contenu réel et il est construit à l'aide du widget Texte
Gestes
Les widgets Flutter prennent en charge l'interaction via un widget spécial, GestureDetector . GestureDetector est un widget invisible ayant la capacité de capturer les interactions de l'utilisateur telles que le tapotement, le glissement, etc., de son widget enfant. De nombreux widgets natifs de Flutter prennent en charge l'interaction via l'utilisation de GestureDetector . Nous pouvons également incorporer une fonctionnalité interactive dans le widget existant en le composant avec le widget GestureDetector . Nous apprendrons les gestes séparément dans les prochains chapitres.
Concept d'État
Les widgets Flutter prennent en charge la maintenance de l'état en fournissant un widget spécial, StatefulWidget . Le widget doit être dérivé du widget StatefulWidget pour prendre en charge la maintenance de l'état et tous les autres widgets doivent être dérivés de StatefulWidget . Les widgets Flutter sontreactiveen natif. Ceci est similaire à reactjs et StatefulWidget sera automatiquement restitué chaque fois que son état interne est modifié. Le re-rendu est optimisé en trouvant la différence entre l'ancienne et la nouvelle interface utilisateur du widget et en ne rendant que les changements nécessaires
Couches
Le concept le plus important du cadre Flutter est que le cadre est regroupé en plusieurs catégories en termes de complexité et clairement organisé en couches de complexité décroissante. Une couche est créée à l'aide de sa couche de niveau suivant immédiate. La couche la plus haute est un widget spécifique à Android et iOS . La couche suivante contient tous les widgets natifs flottants. La couche suivante est la couche de rendu , qui est un composant de rendu de bas niveau et rend tout dans l'application de flutter. Les couches descendent vers le code spécifique à la plate-forme principale
L'aperçu général d'un calque dans Flutter est spécifié dans le diagramme ci-dessous -
Les points suivants résument l'architecture de Flutter -
Dans Flutter, tout est un widget et un widget complexe est composé de widgets déjà existants.
Des fonctionnalités interactives peuvent être intégrées chaque fois que nécessaire à l'aide du widget GestureDetector .
L'état d'un widget peut être maintenu chaque fois que nécessaire à l'aide du widget StatefulWidget .
Flutter propose une conception en couches afin que n'importe quelle couche puisse être programmée en fonction de la complexité de la tâche.
Nous discuterons de tous ces concepts en détail dans les prochains chapitres.
Dart est un langage de programmation open-source polyvalent. Il est à l'origine développé par Google. Dart est un langage orienté objet avec une syntaxe de style C. Il prend en charge les concepts de programmation tels que les interfaces, les classes, contrairement à d'autres langages de programmation, Dart ne prend pas en charge les tableaux. Les collections Dart peuvent être utilisées pour répliquer des structures de données telles que des tableaux, des génériques et un typage facultatif.
Le code suivant montre un programme Dart simple -
void main() {
print("Dart language is easy to learn");
}
Variables et types de données
La variable est nommée emplacement de stockage et les types de données se réfèrent simplement au type et à la taille des données associées aux variables et aux fonctions.
Dart utilise le mot-clé var pour déclarer la variable. La syntaxe de var est définie ci-dessous,
var name = 'Dart';
Les mots clés final et const sont utilisés pour déclarer des constantes. Ils sont définis comme ci-dessous -
void main() {
final a = 12;
const pi = 3.14;
print(a);
print(pi);
}
Le langage Dart prend en charge les types de données suivants -
Numbers - Il est utilisé pour représenter des littéraux numériques - Integer et Double.
Strings- Il représente une séquence de caractères. Les valeurs de chaîne sont spécifiées entre guillemets simples ou doubles.
Booleans- Dart utilise le mot-clé bool pour représenter les valeurs booléennes - vrai et faux.
Lists and Maps- Il est utilisé pour représenter une collection d'objets. Une liste simple peut être définie comme ci-dessous -.
void main() {
var list = [1,2,3,4,5];
print(list);
}
La liste ci-dessus produit la liste [1,2,3,4,5].
La carte peut être définie comme indiqué ici -
void main() {
var mapping = {'id': 1,'name':'Dart'};
print(mapping);
}
Dynamic- Si le type de variable n'est pas défini, son type par défaut est dynamique. L'exemple suivant illustre la variable de type dynamique -
void main() {
dynamic name = "Dart";
print(name);
}
Prise de décision et boucles
Un bloc de prise de décision évalue une condition avant que les instructions ne soient exécutées. Dart prend en charge les instructions If, If..else et switch.
Les boucles sont utilisées pour répéter un bloc de code jusqu'à ce qu'une condition spécifique soit remplie. Dart prend en charge les boucles for, for..in, while et do.. while.
Comprenons un exemple simple sur l'utilisation des instructions de contrôle et des boucles -
void main() {
for( var i = 1 ; i <= 10; i++ ) {
if(i%2==0) {
print(i);
}
}
}
Le code ci-dessus imprime les nombres pairs de 1 à 10.
Les fonctions
Une fonction est un groupe d'instructions qui, ensemble, exécute une tâche spécifique. Examinons une fonction simple de Dart comme indiqué ici -
void main() {
add(3,4);
}
void add(int a,int b) {
int c;
c = a+b;
print(c);
}
La fonction ci-dessus ajoute deux valeurs et produit 7 comme sortie.
Programmation orientée objet
Dart est un langage orienté objet. Il prend en charge les fonctionnalités de programmation orientées objet telles que les classes, les interfaces, etc.
Une classe est un modèle pour créer des objets. Une définition de classe comprend les éléments suivants:
- Fields
- Getters et setters
- Constructors
- Functions
Maintenant, créons une classe simple en utilisant les définitions ci-dessus -
class Employee {
String name;
//getter method
String get emp_name {
return name;
}
//setter method
void set emp_name(String name) {
this.name = name;
}
//function definition
void result() {
print(name);
}
}
void main() {
//object creation
Employee emp = new Employee();
emp.name = "employee1";
emp.result(); //function call
}
Comme nous l'avons appris dans le chapitre précédent, les widgets sont tout dans le framework Flutter. Nous avons déjà appris à créer de nouveaux widgets dans les chapitres précédents.
Dans ce chapitre, laissez-nous comprendre le concept réel derrière la création des widgets et les différents types de widgets disponibles dans le framework Flutter .
Laissez - nous vérifier le Bonjour tout le monde de l' application MyHomePage widget. Le code à cet effet est comme indiqué ci-dessous -
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text(this.title), ),
body: Center(child: Text( 'Hello World',)),
);
}
}
Ici, nous avons créé un nouveau widget en étendant StatelessWidget .
Notez que StatelessWidget ne nécessite qu'une seule génération de méthode pour être implémentée dans sa classe dérivée. La méthode de construction obtient l'environnement de contexte nécessaire pour créer les widgets via le paramètre BuildContext et renvoie le widget qu'elle construit.
Dans le code, nous avons utilisé title comme l'un des arguments du constructeur et également utilisé Key comme autre argument. Le titre est utilisé pour afficher le titre et Key est utilisé pour identifier le widget dans l'environnement de construction.
Ici, la méthode de construction appelle la méthode de construction de Scaffold , qui à son tour appelle la méthode de construction d' AppBar et Center pour créer son interface utilisateur.
Enfin, la méthode de génération Center appelle la méthode de génération Text .
Pour une meilleure compréhension, la représentation visuelle de la même chose est donnée ci-dessous -
Visualisation de la construction du widget
Dans Flutter , les widgets peuvent être regroupés en plusieurs catégories en fonction de leurs fonctionnalités, comme indiqué ci-dessous -
- Widgets spécifiques à la plateforme
- Widgets de mise en page
- Widgets de maintenance d'état
- Widgets de base / indépendants de la plateforme
Parlons maintenant de chacun d'eux en détail.
Widgets spécifiques à la plateforme
Flutter a des widgets spécifiques à une plate-forme particulière - Android ou iOS.
Les widgets spécifiques à Android sont conçus conformément aux directives de conception matérielle d'Android OS. Les widgets spécifiques à Android sont appelés widgets Material .
Les widgets spécifiques à iOS sont conçus conformément aux directives de l'interface humaine d'Apple et sont appelés widgets Cupertino .
Certains des widgets matériels les plus utilisés sont les suivants:
- Scaffold
- AppBar
- BottomNavigationBar
- TabBar
- TabBarView
- ListTile
- RaisedButton
- FloatingActionButton
- FlatButton
- IconButton
- DropdownButton
- PopupMenuButton
- ButtonBar
- TextField
- Checkbox
- Radio
- Switch
- Slider
- Sélecteurs de date et d'heure
- SimpleDialog
- AlertDialog
Certains des widgets Cupertino les plus utilisés sont les suivants -
- CupertinoButton
- CupertinoPicker
- CupertinoDatePicker
- CupertinoTimerPicker
- CupertinoNavigationBar
- CupertinoTabBar
- CupertinoTabScaffold
- CupertinoTabView
- CupertinoTextField
- CupertinoDialog
- CupertinoDialogAction
- CupertinoFullscreenDialogTransition
- CupertinoPageScaffold
- CupertinoPageTransition
- CupertinoActionSheet
- CupertinoActivityIndicator
- CupertinoAlertDialog
- CupertinoPopupSurface
- CupertinoSlider
Widgets de mise en page
Dans Flutter, un widget peut être créé en composant un ou plusieurs widgets. Pour composer plusieurs widgets dans un seul widget, Flutter fournit un grand nombre de widgets avec une fonction de mise en page. Par exemple, le widget enfant peut être centré à l'aide du widget Centre .
Certains des widgets de mise en page populaires sont les suivants:
Container- Une boîte rectangulaire décorée à l'aide des widgets BoxDecoration avec fond, bordure et ombre.
Center - Centrez son widget enfant.
Row - Disposez ses enfants dans le sens horizontal.
Column - Disposez ses enfants dans le sens vertical.
Stack - Disposez les uns au-dessus des autres.
Nous vérifierons les widgets de mise en page en détail dans le prochain chapitre Introduction aux widgets de mise en page .
Widgets de maintenance d'état
Dans Flutter, tous les widgets sont dérivés de StatelessWidget ou StatefulWidget .
Le widget dérivé de StatelessWidget n'a aucune information d'état mais il peut contenir un widget dérivé de StatefulWidget . La nature dynamique de l'application est due au comportement interactif des widgets et aux changements d'état pendant l'interaction. Par exemple, appuyer sur un bouton de compteur augmentera / diminuera l'état interne du compteur d'une unité et la nature réactive du widget Flutter rendra automatiquement le widget en utilisant de nouvelles informations d'état.
Nous apprendrons en détail le concept des widgets StatefulWidget dans le prochain chapitre sur la gestion des états .
Widgets de base / indépendants de la plateforme
Flutter fournit un grand nombre de widgets de base pour créer une interface utilisateur simple et complexe d'une manière indépendante de la plate-forme. Voyons quelques widgets de base dans ce chapitre.
Text
Le widget de texte est utilisé pour afficher un morceau de chaîne. Le style de la chaîne peut être défini à l'aide de la propriété style et de la classe TextStyle . L'exemple de code à cet effet est le suivant:
Text('Hello World!', style: TextStyle(fontWeight: FontWeight.bold))
Le widget texte a un constructeur spécial, Text.rich , qui accepte l'enfant de type TextSpan pour spécifier la chaîne avec un style différent. Le widget TextSpan est de nature récursive et accepte TextSpan comme ses enfants. L'exemple de code à cet effet est le suivant:
Text.rich(
TextSpan(
children: <TextSpan>[
TextSpan(text: "Hello ", style:
TextStyle(fontStyle: FontStyle.italic)),
TextSpan(text: "World", style:
TextStyle(fontWeight: FontWeight.bold)),
],
),
)
Les propriétés les plus importantes du widget Texte sont les suivantes -
maxLines, int - Nombre maximum de lignes à afficher
overflow, TextOverFlow- Spécifiez comment le débordement visuel est géré à l'aide de la classe TextOverFlow
style, TextStyle- Spécifiez le style de la chaîne à l'aide de la classe TextStyle
textAlign, TextAlign- Alignement du texte comme à droite, à gauche, justifier, etc., en utilisant la classe TextAlign
textDirection, TextDirection - Sens de circulation du texte, de gauche à droite ou de droite à gauche
Image
Le widget Image est utilisé pour afficher une image dans l'application. Le widget Image fournit différents constructeurs pour charger des images à partir de plusieurs sources et ils sont les suivants -
Image- Chargeur d'image générique utilisant ImageProvider
Image.asset - Charger l'image à partir des actifs du projet Flutter
Image.file - Charger l'image à partir du dossier système
Image.memory - Charger l'image de la mémoire
Image.Network - Charger l'image du réseau
L'option la plus simple pour charger et afficher une image dans Flutter consiste à inclure l'image en tant qu'actif de l'application et à la charger dans le widget à la demande.
Créez un dossier, des actifs dans le dossier du projet et placez les images nécessaires.
Spécifiez les actifs dans pubspec.yaml comme indiqué ci-dessous -
flutter:
assets:
- assets/smiley.png
Maintenant, chargez et affichez l'image dans l'application.
Image.asset('assets/smiley.png')
Le code source complet du widget MyHomePage de l'application hello world et le résultat est comme indiqué ci-dessous -.
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar( title: Text(this.title), ),
body: Center( child: Image.asset("assets/smiley.png")),
);
}
}
L'image chargée est comme indiqué ci-dessous -
Les propriétés les plus importantes du widget Image sont les suivantes -
image, ImageProvider - Image réelle à charger
width, double - Largeur de l'image
height, double - Hauteur de l'image
alignment, AlignmentGeometry - Comment aligner l'image dans ses limites
Icon
Le widget Icon est utilisé pour afficher un glyphe à partir d'une police décrite dans la classe IconData . Le code pour charger une icône de courrier électronique simple est le suivant -
Icon(Icons.email)
Le code source complet pour l'appliquer dans l'application Hello World est le suivant -
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text(this.title),),
body: Center( child: Icon(Icons.email)),
);
}
}
L'icône chargée est comme indiqué ci-dessous -
Puisque le concept de base de Flutter est que tout est widget , Flutter intègre une fonctionnalité de mise en page d'interface utilisateur dans les widgets eux-mêmes. Flutter fournit un grand nombre de widgets spécialement conçus comme Container, Center, Align , etc., uniquement dans le but de mettre en page l'interface utilisateur. Les widgets créés en composant d'autres widgets utilisent normalement des widgets de mise en page. Laissez l'utilisation apprendre le concept de disposition Flutter dans ce chapitre.
Type de widgets de mise en page
Les widgets de mise en page peuvent être regroupés en deux catégories distinctes en fonction de leur enfant -
- Widget supportant un seul enfant
- Widget prenant en charge plusieurs enfants
Apprenons à la fois le type de widgets et ses fonctionnalités dans les sections à venir.
Widgets enfant unique
Dans cette catégorie, les widgets n'auront qu'un seul widget comme enfant et chaque widget aura une fonctionnalité de mise en page spéciale.
Par exemple, le widget Centre centre simplement son widget enfant par rapport à son widget parent et le widget Conteneur offre une flexibilité complète pour le placer enfant à n'importe quel endroit donné à l'intérieur en utilisant différentes options telles que le remplissage, la décoration, etc.,
Les widgets enfants uniques sont d'excellentes options pour créer des widgets de haute qualité ayant une seule fonctionnalité telle que bouton, étiquette, etc.,
Le code pour créer un bouton simple à l'aide du widget Container est le suivant -
class MyButton extends StatelessWidget {
MyButton({Key key}) : super(key: key);
@override
Widget build(BuildContext context) {
return Container(
decoration: const BoxDecoration(
border: Border(
top: BorderSide(width: 1.0, color: Color(0xFFFFFFFFFF)),
left: BorderSide(width: 1.0, color: Color(0xFFFFFFFFFF)),
right: BorderSide(width: 1.0, color: Color(0xFFFF000000)),
bottom: BorderSide(width: 1.0, color: Color(0xFFFF000000)),
),
),
child: Container(
padding: const
EdgeInsets.symmetric(horizontal: 20.0, vertical: 2.0),
decoration: const BoxDecoration(
border: Border(
top: BorderSide(width: 1.0, color: Color(0xFFFFDFDFDF)),
left: BorderSide(width: 1.0, color: Color(0xFFFFDFDFDF)),
right: BorderSide(width: 1.0, color: Color(0xFFFF7F7F7F)),
bottom: BorderSide(width: 1.0, color: Color(0xFFFF7F7F7F)),
),
color: Colors.grey,
),
child: const Text(
'OK',textAlign: TextAlign.center, style: TextStyle(color: Colors.black)
),
),
);
}
}
Ici, nous avons utilisé deux widgets - un widget Container et un widget Text . Le résultat du widget est un bouton personnalisé comme indiqué ci-dessous -
Voyons quelques-uns des widgets de mise en page pour enfant unique les plus importants fournis par Flutter -
Padding- Utilisé pour organiser son widget enfant par le remplissage donné. Ici, le remplissage peut être fourni par la classe EdgeInsets .
Align- Alignez son widget enfant sur lui-même en utilisant la valeur de la propriété alignement . La valeur de la propriété d' alignement peut être fournie par la classe FractionalOffset . La classe FractionalOffset spécifie les décalages en termes de distance à partir du coin supérieur gauche.
Certaines des valeurs possibles des décalages sont les suivantes -
FractionalOffset (1.0, 0.0) représente le coin supérieur droit.
FractionalOffset (0.0, 1.0) représente le coin inférieur gauche.
Un exemple de code sur les décalages est présenté ci-dessous -
Center(
child: Container(
height: 100.0,
width: 100.0,
color: Colors.yellow, child: Align(
alignment: FractionalOffset(0.2, 0.6),
child: Container( height: 40.0, width:
40.0, color: Colors.red,
),
),
),
)
FittedBox - Il met à l'échelle le widget enfant, puis le positionne en fonction de l'ajustement spécifié.
AspectRatio - Il tente de dimensionner le widget enfant au rapport hauteur / largeur spécifié
ConstrainedBox
Baseline
FractinallySizedBox
IntrinsicHeight
IntrinsicWidth
LiimitedBox
OffStage
OverflowBox
SizedBox
SizedOverflowBox
Transform
CustomSingleChildLayout
Notre application hello world utilise des widgets de mise en page basés sur des matériaux pour concevoir la page d'accueil. Modifions notre application hello world pour créer la page d'accueil en utilisant des widgets de mise en page de base comme spécifié ci-dessous -
Container - Widget conteneur générique, enfant unique, basé sur une boîte avec alignement, remplissage, bordure et marge ainsi que de riches fonctionnalités de style.
Center - Widget conteneur enfant simple et unique, qui centre son widget enfant.
Le code modifié du widget MyHomePage et MyApp est comme ci-dessous -
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MyHomePage(title: "Hello World demo app");
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Container(
decoration: BoxDecoration(color: Colors.white,),
padding: EdgeInsets.all(25), child: Center(
child:Text(
'Hello World', style: TextStyle(
color: Colors.black, letterSpacing: 0.5, fontSize: 20,
),
textDirection: TextDirection.ltr,
),
)
);
}
}
Ici,
Le widget de conteneur est le widget de niveau supérieur ou racine. Le conteneur est configuré à l'aide de la propriété decoration et padding pour mettre en page son contenu.
BoxDecoration a de nombreuses propriétés comme la couleur, la bordure, etc., pour décorer le widget Conteneur et ici, la couleur est utilisée pour définir la couleur du conteneur.
le remplissage du widget Container est défini à l'aide de la classe dgeInsets , qui offre la possibilité de spécifier la valeur de remplissage.
Center est le widget enfant du widget Container . Encore une fois, Text est l'enfant du widget Centre . Le texte est utilisé pour afficher le message et le centre est utilisé pour centrer le message texte par rapport au widget parent, Container .
Le résultat final du code donné ci-dessus est un exemple de mise en page comme indiqué ci-dessous -
Widgets enfants multiples
Dans cette catégorie, un widget donné aura plus d'un widget enfant et la disposition de chaque widget est unique.
Par exemple, le widget Row permet la disposition de ses enfants dans le sens horizontal, tandis que le widget Column permet de disposer de ses enfants dans le sens vertical. En composant Row et Column , un widget avec n'importe quel niveau de complexité peut être construit.
Apprenons quelques-uns des widgets fréquemment utilisés dans cette section.
Row - Permet de disposer ses enfants de manière horizontale.
Column - Permet de disposer ses enfants de manière verticale.
ListView - Permet d'organiser ses enfants sous forme de liste.
GridView - Permet d'agencer ses enfants en galerie.
Expanded - Utilisé pour que les enfants du widget Ligne et Colonne occupent la zone maximale possible.
Table - Widget basé sur une table.
Flow - Widget basé sur le flux.
Stack - Widget basé sur la pile.
Application de mise en page avancée
Dans cette section, apprenons à créer une interface utilisateur complexe de liste de produits avec une conception personnalisée à l'aide de widgets de mise en page enfants simples et multiples.
Pour cela, suivez la séquence ci-dessous -
Créez une nouvelle application Flutter dans le studio Android, product_layout_app .
Remplacez le code main.dart par le code suivant -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo', theme: ThemeData(
primarySwatch: Colors.blue,),
home: MyHomePage(title: 'Product layout demo home page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text(this.title),),
body: Center(child: Text( 'Hello World', )),
);
}
}
Here,
Nous avons créé le widget MyHomePage en étendant StatelessWidget au lieu de StatefulWidget par défaut , puis avons supprimé le code correspondant.
Maintenant, créez un nouveau widget, ProductBox selon la conception spécifiée comme indiqué ci-dessous -
Le code de la ProductBox est le suivant.
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.name, this.description, this.price, this.image})
: super(key: key);
final String name;
final String description;
final int price;
final String image;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2), height: 120, child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: <Widget>[
Image.asset("assets/appimages/" +image), Expanded(
child: Container(
padding: EdgeInsets.all(5), child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(this.name, style: TextStyle(fontWeight:
FontWeight.bold)), Text(this.description),
Text("Price: " + this.price.toString()),
],
)
)
)
]
)
)
);
}
}
Veuillez observer ce qui suit dans le code -
ProductBox a utilisé quatre arguments comme spécifié ci-dessous -
name - Nom du produit
description - Description du produit
prix - Prix du produit
image - Image du produit
ProductBox utilise sept widgets intégrés comme spécifié ci-dessous -
- Container
- Expanded
- Row
- Column
- Card
- Text
- Image
ProductBox est conçu en utilisant le widget mentionné ci-dessus. La disposition ou la hiérarchie du widget est spécifiée dans le diagramme ci-dessous -
Maintenant, placez une image factice (voir ci-dessous) pour les informations sur le produit dans le dossier assets de l'application et configurez le dossier assets dans le fichier pubspec.yaml comme indiqué ci-dessous -
assets:
- assets/appimages/floppy.png
- assets/appimages/iphone.png
- assets/appimages/laptop.png
- assets/appimages/pendrive.png
- assets/appimages/pixel.png
- assets/appimages/tablet.png
iPhone.png
Pixel.png
Laptop.png
Tablet.png
Pendrive.png
Floppy.png
Enfin, utilisez le widget ProductBox dans le widget MyHomePage comme spécifié ci-dessous -
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title:Text("Product Listing")),
body: ListView(
shrinkWrap: true, padding: const EdgeInsets.fromLTRB(2.0, 10.0, 2.0, 10.0),
children: <Widget> [
ProductBox(
name: "iPhone",
description: "iPhone is the stylist phone ever",
price: 1000,
image: "iphone.png"
),
ProductBox(
name: "Pixel",
description: "Pixel is the most featureful phone ever",
price: 800,
image: "pixel.png"
),
ProductBox(
name: "Laptop",
description: "Laptop is most productive development tool",
price: 2000,
image: "laptop.png"
),
ProductBox(
name: "Tablet",
description: "Tablet is the most useful device ever for meeting",
price: 1500,
image: "tablet.png"
),
ProductBox(
name: "Pendrive",
description: "Pendrive is useful storage medium",
price: 100,
image: "pendrive.png"
),
ProductBox(
name: "Floppy Drive",
description: "Floppy drive is useful rescue storage medium",
price: 20,
image: "floppy.png"
),
],
)
);
}
}
Ici, nous avons utilisé ProductBox comme enfants du widget ListView .
Le code complet (main.dart) de l'application de mise en page du produit (product_layout_app) est le suivant -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo', theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Product layout demo home page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text("Product Listing")),
body: ListView(
shrinkWrap: true,
padding: const EdgeInsets.fromLTRB(2.0, 10.0, 2.0, 10.0),
children: <Widget>[
ProductBox(
name: "iPhone",
description: "iPhone is the stylist phone ever",
price: 1000,
image: "iphone.png"
),
ProductBox(
name: "Pixel",
description: "Pixel is the most featureful phone ever",
price: 800,
image: "pixel.png"
),
ProductBox(
name: "Laptop",
description: "Laptop is most productive development tool",
price: 2000,
image: "laptop.png"
),
ProductBox(
name: "Tablet",
description: "Tablet is the most useful device ever for meeting",
price: 1500,
image: "tablet.png"
),
ProductBox(
name: "Pendrive",
description: "Pendrive is useful storage medium",
price: 100,
image: "pendrive.png"
),
ProductBox(
name: "Floppy Drive",
description: "Floppy drive is useful rescue storage medium",
price: 20,
image: "floppy.png"
),
],
)
);
}
}
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.name, this.description, this.price, this.image}) :
super(key: key);
final String name;
final String description;
final int price;
final String image;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2),
height: 120,
child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Image.asset("assets/appimages/" + image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(
this.name, style: TextStyle(
fontWeight: FontWeight.bold
)
),
Text(this.description), Text(
"Price: " + this.price.toString()
),
],
)
)
)
]
)
)
);
}
}
La sortie finale de l'application est la suivante -
Les gestes sont principalement un moyen pour un utilisateur d'interagir avec une application mobile (ou tout autre appareil tactile). Les gestes sont généralement définis comme toute action / mouvement physique d'un utilisateur dans l'intention d'activer un contrôle spécifique de l'appareil mobile. Les gestes sont aussi simples que d'appuyer sur l'écran de l'appareil mobile pour des actions plus complexes utilisées dans les applications de jeu.
Certains des gestes largement utilisés sont mentionnés ici -
Tap - Toucher la surface de l'appareil avec le bout du doigt pendant une courte période puis relâcher le bout du doigt.
Double Tap - Tapotez deux fois en peu de temps.
Drag - Toucher la surface de l'appareil avec le bout du doigt, puis déplacer le bout du doigt de manière régulière et enfin relâcher le bout du doigt.
Flick - Similaire au glisser, mais en le faisant de manière plus rapide.
Pinch - Pincer la surface de l'appareil à l'aide de deux doigts.
Spread/Zoom - Face au pincement.
Panning - Toucher la surface de l'appareil du bout du doigt et le déplacer dans n'importe quelle direction sans relâcher le bout du doigt.
Flutter fournit un excellent support pour tout type de gestes grâce à son widget exclusif, GestureDetector. GestureDetector est un widget non visuel principalement utilisé pour détecter le geste de l'utilisateur. Pour identifier un geste ciblé sur un widget, le widget peut être placé à l'intérieur du widget GestureDetector. GestureDetector capturera le geste et enverra plusieurs événements en fonction du geste.
Certains gestes et événements correspondants sont donnés ci-dessous -
- Tap
- onTapDown
- onTapUp
- onTap
- onTapCancel
- Tapez deux fois
- onDoubleTap
- Appui long
- onLongPress
- Glissement vertical
- onVerticalDragStart
- onVerticalDragUpdate
- onVerticalDragEnd
- Traînée horizontale
- onHorizontalDragStart
- onHorizontalDragUpdate
- onHorizontalDragEnd
- Pan
- onPanStart
- onPanUpdate
- onPanEnd
Maintenant, modifions l'application hello world pour inclure la fonction de détection des gestes et essayons de comprendre le concept.
Modifiez le contenu du corps du widget MyHomePage comme indiqué ci-dessous -
body: Center(
child: GestureDetector(
onTap: () {
_showDialog(context);
},
child: Text( 'Hello World', )
)
),
Observez qu'ici nous avons placé le widget GestureDetector au-dessus du widget Texte dans la hiérarchie des widgets, capturé l'événement onTap et enfin affiché une fenêtre de dialogue.
Implémentez la fonction * _showDialog * pour présenter une boîte de dialogue lorsque l'utilisateur tabule le message hello world . Il utilise les widgets génériques showDialog et AlertDialog pour créer un nouveau widget de dialogue. Le code est indiqué ci-dessous -
// user defined function void _showDialog(BuildContext context) {
// flutter defined function
showDialog(
context: context, builder: (BuildContext context) {
// return object of type Dialog
return AlertDialog(
title: new Text("Message"),
content: new Text("Hello World"),
actions: <Widget>[
new FlatButton(
child: new Text("Close"),
onPressed: () {
Navigator.of(context).pop();
},
),
],
);
},
);
}
L'application se rechargera dans l'appareil à l'aide de la fonction de rechargement à chaud. Maintenant, cliquez simplement sur le message, Hello World et il affichera la boîte de dialogue comme ci-dessous -
Maintenant, fermez la boîte de dialogue en cliquant sur l' option de fermeture dans la boîte de dialogue.
Le code complet (main.dart) est le suivant -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Hello World Demo Application',
theme: ThemeData( primarySwatch: Colors.blue,),
home: MyHomePage(title: 'Home page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
// user defined function
void _showDialog(BuildContext context) {
// flutter defined function showDialog(
context: context, builder: (BuildContext context) {
// return object of type Dialog return AlertDialog(
title: new Text("Message"),
content: new Text("Hello World"),
actions: <Widget>[
new FlatButton(
child: new Text("Close"),
onPressed: () {
Navigator.of(context).pop();
},
),
],
);
},
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text(this.title),),
body: Center(
child: GestureDetector(
onTap: () {
_showDialog(context);
},
child: Text( 'Hello World', )
)
),
);
}
}
Enfin, Flutter fournit également un mécanisme de détection des gestes de bas niveau via le widget Listener . Il détectera toutes les interactions des utilisateurs, puis diffusera les événements suivants -
- PointerDownEvent
- PointerMoveEvent
- PointerUpEvent
- PointerCancelEvent
Flutter fournit également un petit ensemble de widgets pour effectuer des gestes spécifiques et avancés. Les widgets sont listés ci-dessous -
Dismissible - Prend en charge le geste flick pour rejeter le widget.
Draggable - Prend en charge le geste de glisser pour déplacer le widget.
LongPressDraggable - Prend en charge le geste de glisser pour déplacer un widget, lorsque son widget parent est également déplaçable.
DragTarget- Accepte tous les widgets déplaçables
IgnorePointer - Masque le widget et ses enfants du processus de détection des gestes.
AbsorbPointer - Arrête le processus de détection de geste lui-même et ainsi tout widget qui se chevauche ne peut pas non plus participer au processus de détection de geste et, par conséquent, aucun événement n'est déclenché.
Scrollable - Supporte le défilement du contenu disponible à l'intérieur du widget.
La gestion de l'état dans une application est l'un des processus les plus importants et nécessaires du cycle de vie d'une application.
Considérons une simple application de panier.
L'utilisateur se connectera en utilisant ses informations d'identification dans l'application.
Une fois que l'utilisateur est connecté, l'application doit conserver les détails de l'utilisateur connecté dans tout l'écran.
Encore une fois, lorsque l'utilisateur sélectionne un produit et l'enregistre dans un panier, les informations du panier doivent persister entre les pages jusqu'à ce que l'utilisateur extrait le panier.
L'utilisateur et les informations de son panier à n'importe quelle instance sont appelés l'état de l'application à cette instance.
Une gestion d'état peut être divisée en deux catégories en fonction de la durée d'un état particulier dans une application.
Ephemeral- Durée de quelques secondes comme l'état actuel d'une animation ou une seule page comme l'évaluation actuelle d'un produit. Flutter prend en charge son via StatefulWidget.
app state- Dernier pour toute l'application comme les détails de l'utilisateur connecté, les informations du panier, etc., Flutter prend en charge son via scoped_model.
Navigation et routage
Dans n'importe quelle application, la navigation d'une page / écran à l'autre définit le flux de travail de l'application. La façon dont la navigation d'une application est gérée est appelée Routage. Flutter fournit une classe de routage de base - MaterialPageRoute et deux méthodes - Navigator.push et Navigator.pop, pour définir le flux de travail d'une application.
MaterialPageRoute
MaterialPageRoute est un widget utilisé pour rendre son interface utilisateur en remplaçant tout l'écran par une animation spécifique à la plate-forme.
MaterialPageRoute(builder: (context) => Widget())
Ici, le constructeur acceptera une fonction pour construire son contenu en ajoutant le contexte actuel de l'application.
Navigation.push
Navigation.push est utilisé pour naviguer vers un nouvel écran à l'aide du widget MaterialPageRoute.
Navigator.push( context, MaterialPageRoute(builder: (context) => Widget()), );
Navigation.pop
Navigation.pop est utilisé pour naviguer vers l'écran précédent.
Navigator.pop(context);
Créons une nouvelle application pour mieux comprendre le concept de navigation.
Créez une nouvelle application Flutter dans le studio Android, product_nav_app
Copiez le dossier des actifs de product_nav_app vers product_state_app et ajoutez des actifs dans le fichier pubspec.yaml.
flutter:
assets:
- assets/appimages/floppy.png
- assets/appimages/iphone.png
- assets/appimages/laptop.png
- assets/appimages/pendrive.png
- assets/appimages/pixel.png
- assets/appimages/tablet.png
Remplacez le code de démarrage par défaut (main.dart) par notre code de démarrage.
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(
title: 'Product state demo home page'
),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.title),
),
body: Center(
child: Text('Hello World',)
),
);
}
}
Créons une classe Product pour organiser les informations produit.
class Product {
final String name;
final String description;
final int price;
final String image;
Product(this.name, this.description, this.price, this.image);
}
Écrivons une méthode getProducts dans la classe Product pour générer nos fiches produits factices.
static List<Product> getProducts() {
List<Product> items = <Product>[];
items.add(
Product(
"Pixel",
"Pixel is the most feature-full phone ever", 800,
"pixel.png"
)
);
items.add(
Product(
"Laptop",
"Laptop is most productive development tool",
2000, "
laptop.png"
)
);
items.add(
Product(
"Tablet",
"Tablet is the most useful device ever for meeting",
1500,
"tablet.png"
)
);
items.add(
Product(
"Pendrive",
"Pendrive is useful storage medium",
100,
"pendrive.png"
)
);
items.add(
Product(
"Floppy Drive",
"Floppy drive is useful rescue storage medium",
20,
"floppy.png"
)
);
return items;
}
import product.dart in main.dart
import 'Product.dart';
Incluons notre nouveau widget, RatingBox.
class RatingBox extends StatefulWidget {
@override
_RatingBoxState createState() =>_RatingBoxState();
}
class _RatingBoxState extends State<RatingBox> {
int _rating = 0;
void _setRatingAsOne() {
setState(() {
_rating = 1;
});
}
void _setRatingAsTwo() {
setState(() {
_rating = 2;
});
}
void _setRatingAsThree() {
setState(() {
_rating = 3;
});
}
Widget build(BuildContext context) {
double _size = 20;
print(_rating);
return Row(
mainAxisAlignment: MainAxisAlignment.end,
crossAxisAlignment: CrossAxisAlignment.end,
mainAxisSize: MainAxisSize.max,
children: <Widget>[
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 1?
Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsOne,
iconSize: _size,
),
),
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 2?
Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsTwo,
iconSize: _size,
),
),
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 3 ?
Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsThree,
iconSize: _size,
),
),
],
);
}
}
Modifions notre widget ProductBox pour qu'il fonctionne avec notre nouvelle classe Product.
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.item}) : super(key: key);
final Product item;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2),
height: 140,
child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Image.asset("assets/appimages/" + this.item.image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(this.item.name,
style: TextStyle(fontWeight: FontWeight.bold)),
Text(this.item.description),
Text("Price: " + this.item.price.toString()),
RatingBox(),
],
)
)
)
]
),
)
);
}
}
Réécrivons notre widget MyHomePage pour qu'il fonctionne avec le modèle de produit et pour lister tous les produits en utilisant ListView.
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
final items = Product.getProducts();
@override
Widget build(BuildContext context) {
return Scaffold( appBar: AppBar(title: Text("Product Navigation")),
body: ListView.builder(
itemCount: items.length,
itemBuilder: (context, index) {
return GestureDetector(
child: ProductBox(item: items[index]),
onTap: () {
Navigator.push(
context, MaterialPageRoute(
builder: (context) => ProductPage(item: items[index]),
),
);
},
);
},
));
}
}
Ici, nous avons utilisé MaterialPageRoute pour accéder à la page de détails du produit.
Maintenant, ajoutons ProductPage pour afficher les détails du produit.
class ProductPage extends StatelessWidget {
ProductPage({Key key, this.item}) : super(key: key);
final Product item;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.item.name),
),
body: Center(
child: Container(
padding: EdgeInsets.all(0),
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Image.asset("assets/appimages/" + this.item.image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(
this.item.name, style: TextStyle(
fontWeight: FontWeight.bold
)
),
Text(this.item.description),
Text("Price: " + this.item.price.toString()),
RatingBox(),
],
)
)
)
]
),
),
),
);
}
}
Le code complet de l'application est le suivant -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class Product {
final String name;
final String description;
final int price;
final String image;
Product(this.name, this.description, this.price, this.image);
static List<Product> getProducts() {
List<Product> items = <Product>[];
items.add(
Product(
"Pixel",
"Pixel is the most featureful phone ever",
800,
"pixel.png"
)
);
items.add(
Product(
"Laptop",
"Laptop is most productive development tool",
2000,
"laptop.png"
)
);
items.add(
Product(
"Tablet",
"Tablet is the most useful device ever for meeting",
1500,
"tablet.png"
)
);
items.add(
Product(
"Pendrive",
"iPhone is the stylist phone ever",
100,
"pendrive.png"
)
);
items.add(
Product(
"Floppy Drive",
"iPhone is the stylist phone ever",
20,
"floppy.png"
)
);
items.add(
Product(
"iPhone",
"iPhone is the stylist phone ever",
1000,
"iphone.png"
)
);
return items;
}
}
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Product Navigation demo home page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
final items = Product.getProducts();
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text("Product Navigation")),
body: ListView.builder(
itemCount: items.length,
itemBuilder: (context, index) {
return GestureDetector(
child: ProductBox(item: items[index]),
onTap: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => ProductPage(item: items[index]),
),
);
},
);
},
)
);
}
}
class ProductPage extends StatelessWidget {
ProductPage({Key key, this.item}) : super(key: key);
final Product item;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.item.name),
),
body: Center(
child: Container(
padding: EdgeInsets.all(0),
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Image.asset("assets/appimages/" + this.item.image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(this.item.name, style: TextStyle(fontWeight: FontWeight.bold)),
Text(this.item.description),
Text("Price: " + this.item.price.toString()),
RatingBox(),
],
)
)
)
]
),
),
),
);
}
}
class RatingBox extends StatefulWidget {
@override
_RatingBoxState createState() => _RatingBoxState();
}
class _RatingBoxState extends State<RatingBox> {
int _rating = 0;
void _setRatingAsOne() {
setState(() {
_rating = 1;
});
}
void _setRatingAsTwo() {
setState(() {
_rating = 2;
});
}
void _setRatingAsThree() {
setState(() {
_rating = 3;
});
}
Widget build(BuildContext context) {
double _size = 20;
print(_rating);
return Row(
mainAxisAlignment: MainAxisAlignment.end,
crossAxisAlignment: CrossAxisAlignment.end,
mainAxisSize: MainAxisSize.max,
children: <Widget>[
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 1 ? Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsOne,
iconSize: _size,
),
),
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 2 ?
Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsTwo,
iconSize: _size,
),
),
Container(
padding: EdgeInsets.all(0),
child: IconButton(
icon: (
_rating >= 3 ?
Icon(
Icons.star,
size: _size,
)
: Icon(
Icons.star_border,
size: _size,
)
),
color: Colors.red[500],
onPressed: _setRatingAsThree,
iconSize: _size,
),
),
],
);
}
}
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.item}) : super(key: key);
final Product item;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2),
height: 140,
child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Image.asset("assets/appimages/" + this.item.image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(this.item.name, style: TextStyle(fontWeight: FontWeight.bold)), Text(this.item.description),
Text("Price: " + this.item.price.toString()),
RatingBox(),
],
)
)
)
]
),
)
);
}
}
Exécutez l'application et cliquez sur l'un des éléments du produit. Il affichera la page des détails pertinents. Nous pouvons passer à la page d'accueil en cliquant sur le bouton retour. La page de liste de produits et la page de détails de produit de l'application sont affichées comme suit -
L'animation est une procédure complexe dans n'importe quelle application mobile. En dépit de sa complexité, Animation améliore l'expérience utilisateur à un nouveau niveau et offre une interaction utilisateur riche. En raison de sa richesse, l'animation devient partie intégrante de l'application mobile moderne. Le framework Flutter reconnaît l'importance de l'animation et fournit un cadre simple et intuitif pour développer tous les types d'animations.
introduction
L'animation est un processus qui consiste à montrer une série d'images / d'images dans un ordre particulier pendant une durée spécifique pour donner une illusion de mouvement. Les aspects les plus importants de l'animation sont les suivants -
L'animation a deux valeurs distinctes: la valeur de début et la valeur de fin. L'animation commence à partir de la valeur de début et passe par une série de valeurs intermédiaires et se termine enfin aux valeurs de fin. Par exemple, pour animer un widget pour qu'il disparaisse, la valeur initiale sera l'opacité totale et la valeur finale sera l'opacité zéro.
Les valeurs intermédiaires peuvent être de nature linéaire ou non linéaire (courbe) et elles peuvent être configurées. Comprenez que l'animation fonctionne telle qu'elle est configurée. Chaque configuration donne une sensation différente à l'animation. Par exemple, la décoloration d'un widget sera de nature linéaire tandis que le rebond d'une balle sera de nature non linéaire.
La durée du processus d'animation affecte la vitesse (lenteur ou rapidité) de l'animation.
La possibilité de contrôler le processus d'animation comme le démarrage de l'animation, l'arrêt de l'animation, la répétition de l'animation pour définir un nombre de fois, l'inversion du processus d'animation, etc.,
Dans Flutter, le système d'animation ne fait aucune véritable animation. Au lieu de cela, il ne fournit que les valeurs requises à chaque image pour rendre les images.
Classes basées sur l'animation
Le système d'animation Flutter est basé sur des objets d'animation. Les classes d'animation de base et son utilisation sont les suivantes -
Animation
Génère des valeurs interpolées entre deux nombres sur une certaine durée. Les classes d'animation les plus courantes sont -
Animation<double> - interpoler les valeurs entre deux nombres décimaux
Animation<Color> - interpoler les couleurs entre deux couleurs
Animation<Size> - interpoler les tailles entre deux tailles
AnimationController- Objet spécial Animation pour contrôler l'animation elle-même. Il génère de nouvelles valeurs chaque fois que l'application est prête pour une nouvelle image. Il prend en charge l'animation linéaire et la valeur commence de 0,0 à 1,0
controller = AnimationController(duration: const Duration(seconds: 2), vsync: this);
Ici, le contrôleur contrôle l'animation et l'option de durée contrôle la durée du processus d'animation. vsync est une option spéciale utilisée pour optimiser la ressource utilisée dans l'animation.
CourbéAnimation
Similaire à AnimationController mais prend en charge l'animation non linéaire. CurvedAnimation peut être utilisé avec l'objet Animation comme ci-dessous -
controller = AnimationController(duration: const Duration(seconds: 2), vsync: this);
animation = CurvedAnimation(parent: controller, curve: Curves.easeIn)
Tween <T>
Dérivé de Animatable <T> et utilisé pour générer des nombres entre deux nombres autres que 0 et 1. Il peut être utilisé avec l'objet Animation en utilisant la méthode animate et en passant l'objet Animation réel.
AnimationController controller = AnimationController(
duration: const Duration(milliseconds: 1000),
vsync: this); Animation<int> customTween = IntTween(
begin: 0, end: 255).animate(controller);
Tween peut également être utilisé avec CurvedAnimation comme ci-dessous -
AnimationController controller = AnimationController(
duration: const Duration(milliseconds: 500), vsync: this);
final Animation curve = CurvedAnimation(parent: controller, curve: Curves.easeOut);
Animation<int> customTween = IntTween(begin: 0, end: 255).animate(curve);
Ici, le contrôleur est le véritable contrôleur d'animation. La courbe fournit le type de non-linéarité et le customTween fournit une plage personnalisée de 0 à 255.
Flux de travail de l'animation Flutter
Le flux de travail de l'animation est le suivant -
Définissez et démarrez le contrôleur d'animation dans l'initState du StatefulWidget.
AnimationController(duration: const Duration(seconds: 2), vsync: this);
animation = Tween<double>(begin: 0, end: 300).animate(controller);
controller.forward();
Ajoutez un écouteur basé sur une animation, addListener pour changer l'état du widget.
animation = Tween<double>(begin: 0, end: 300).animate(controller) ..addListener(() {
setState(() {
// The state that has changed here is the animation object’s value.
});
});
Les widgets intégrés, AnimatedWidget et AnimatedBuilder peuvent être utilisés pour ignorer ce processus. Les deux widgets acceptent l'objet Animation et obtiennent les valeurs actuelles requises pour l'animation.
Obtenez les valeurs d'animation pendant le processus de construction du widget, puis appliquez-les pour la largeur, la hauteur ou toute propriété pertinente au lieu de la valeur d'origine.
child: Container(
height: animation.value,
width: animation.value,
child: <Widget>,
)
Application de travail
Écrivons une application simple basée sur l'animation pour comprendre le concept d'animation dans le cadre Flutter.
Créez une nouvelle application Flutter dans le studio Android, product_animation_app.
Copiez le dossier des actifs de product_nav_app vers product_animation_app et ajoutez des actifs dans le fichier pubspec.yaml.
flutter:
assets:
- assets/appimages/floppy.png
- assets/appimages/iphone.png
- assets/appimages/laptop.png
- assets/appimages/pendrive.png
- assets/appimages/pixel.png
- assets/appimages/tablet.png
Supprimez le code de démarrage par défaut (main.dart).
Ajoutez l'importation et la fonction principale de base.
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
Créez le widget MyApp dérivé de StatefulWidgtet.
class MyApp extends StatefulWidget {
_MyAppState createState() => _MyAppState();
}
Créez un widget _MyAppState et implémentez initState et supprimez-le en plus de la méthode de construction par défaut.
class _MyAppState extends State<MyApp> with SingleTickerProviderStateMixin {
Animation<double> animation;
AnimationController controller;
@override void initState() {
super.initState();
controller = AnimationController(
duration: const Duration(seconds: 10), vsync: this
);
animation = Tween<double>(begin: 0.0, end: 1.0).animate(controller);
controller.forward();
}
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
controller.forward();
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(primarySwatch: Colors.blue,),
home: MyHomePage(title: 'Product layout demo home page', animation: animation,)
);
}
@override
void dispose() {
controller.dispose();
super.dispose();
}
}
Ici,
Dans la méthode initState, nous avons créé un objet contrôleur d'animation (contrôleur), un objet animation (animation) et démarré l'animation à l'aide de controller.forward.
Dans la méthode dispose, nous avons supprimé l'objet contrôleur d'animation (contrôleur).
Dans la méthode de construction, envoyez une animation au widget MyHomePage via le constructeur. Désormais, le widget MyHomePage peut utiliser l'objet d'animation pour animer son contenu.
Maintenant, ajoutez le widget ProductBox
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.name, this.description, this.price, this.image})
: super(key: key);
final String name;
final String description;
final int price;
final String image;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2),
height: 140,
child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Image.asset("assets/appimages/" + image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(this.name, style:
TextStyle(fontWeight: FontWeight.bold)),
Text(this.description),
Text("Price: " + this.price.toString()),
],
)
)
)
]
)
)
);
}
}
Créez un nouveau widget, MyAnimatedWidget pour faire une animation de fondu simple en utilisant l'opacité.
class MyAnimatedWidget extends StatelessWidget {
MyAnimatedWidget({this.child, this.animation});
final Widget child;
final Animation<double> animation;
Widget build(BuildContext context) => Center(
child: AnimatedBuilder(
animation: animation,
builder: (context, child) => Container(
child: Opacity(opacity: animation.value, child: child),
),
child: child),
);
}
Ici, nous avons utilisé AniatedBuilder pour faire notre animation. AnimatedBuilder est un widget qui construit son contenu tout en faisant l'animation en même temps. Il accepte un objet d'animation pour obtenir la valeur d'animation actuelle. Nous avons utilisé la valeur d'animation, animation.value pour définir l'opacité du widget enfant. En effet, le widget animera le widget enfant en utilisant le concept d'opacité.
Enfin, créez le widget MyHomePage et utilisez l'objet animation pour animer l'un de ses contenus.
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title, this.animation}) : super(key: key);
final String title;
final Animation<double>
animation;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text("Product Listing")),body: ListView(
shrinkWrap: true,
padding: const EdgeInsets.fromLTRB(2.0, 10.0, 2.0, 10.0),
children: <Widget>[
FadeTransition(
child: ProductBox(
name: "iPhone",
description: "iPhone is the stylist phone ever",
price: 1000,
image: "iphone.png"
), opacity: animation
),
MyAnimatedWidget(child: ProductBox(
name: "Pixel",
description: "Pixel is the most featureful phone ever",
price: 800,
image: "pixel.png"
), animation: animation),
ProductBox(
name: "Laptop",
description: "Laptop is most productive development tool",
price: 2000,
image: "laptop.png"
),
ProductBox(
name: "Tablet",
description: "Tablet is the most useful device ever for meeting",
price: 1500,
image: "tablet.png"
),
ProductBox(
name: "Pendrive",
description: "Pendrive is useful storage medium",
price: 100,
image: "pendrive.png"
),
ProductBox(
name: "Floppy Drive",
description: "Floppy drive is useful rescue storage medium",
price: 20,
image: "floppy.png"
),
],
)
);
}
}
Ici, nous avons utilisé FadeAnimation et MyAnimationWidget pour animer les deux premiers éléments de la liste. FadeAnimation est une classe d'animation intégrée, que nous avons utilisée pour animer son enfant en utilisant le concept d'opacité.
Le code complet est le suivant -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatefulWidget {
_MyAppState createState() => _MyAppState();
}
class _MyAppState extends State<MyApp> with SingleTickerProviderStateMixin {
Animation<double> animation;
AnimationController controller;
@override
void initState() {
super.initState();
controller = AnimationController(
duration: const Duration(seconds: 10), vsync: this);
animation = Tween<double>(begin: 0.0, end: 1.0).animate(controller);
controller.forward();
}
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
controller.forward();
return MaterialApp(
title: 'Flutter Demo', theme: ThemeData(primarySwatch: Colors.blue,),
home: MyHomePage(title: 'Product layout demo home page', animation: animation,)
);
}
@override
void dispose() {
controller.dispose();
super.dispose();
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title, this.animation}): super(key: key);
final String title;
final Animation<double> animation;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text("Product Listing")),
body: ListView(
shrinkWrap: true,
padding: const EdgeInsets.fromLTRB(2.0, 10.0, 2.0, 10.0),
children: <Widget>[
FadeTransition(
child: ProductBox(
name: "iPhone",
description: "iPhone is the stylist phone ever",
price: 1000,
image: "iphone.png"
),
opacity: animation
),
MyAnimatedWidget(
child: ProductBox(
name: "Pixel",
description: "Pixel is the most featureful phone ever",
price: 800,
image: "pixel.png"
),
animation: animation
),
ProductBox(
name: "Laptop",
description: "Laptop is most productive development tool",
price: 2000,
image: "laptop.png"
),
ProductBox(
name: "Tablet",
description: "Tablet is the most useful device ever for meeting",
price: 1500,
image: "tablet.png"
),
ProductBox(
name: "Pendrive",
description: "Pendrive is useful storage medium",
price: 100,
image: "pendrive.png"
),
ProductBox(
name: "Floppy Drive",
description: "Floppy drive is useful rescue storage medium",
price: 20,
image: "floppy.png"
),
],
)
);
}
}
class ProductBox extends StatelessWidget {
ProductBox({Key key, this.name, this.description, this.price, this.image}) :
super(key: key);
final String name;
final String description;
final int price;
final String image;
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.all(2),
height: 140,
child: Card(
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Image.asset("assets/appimages/" + image),
Expanded(
child: Container(
padding: EdgeInsets.all(5),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: <Widget>[
Text(
this.name, style: TextStyle(
fontWeight: FontWeight.bold
)
),
Text(this.description), Text(
"Price: " + this.price.toString()
),
],
)
)
)
]
)
)
);
}
}
class MyAnimatedWidget extends StatelessWidget {
MyAnimatedWidget({this.child, this.animation});
final Widget child;
final Animation<double> animation;
Widget build(BuildContext context) => Center(
child: AnimatedBuilder(
animation: animation,
builder: (context, child) => Container(
child: Opacity(opacity: animation.value, child: child),
),
child: child
),
);
}
Compilez et exécutez l'application pour voir les résultats. La version initiale et finale de l'application est la suivante -
Flutter fournit un cadre général pour accéder aux fonctionnalités spécifiques de la plate-forme. Cela permet au développeur d'étendre les fonctionnalités du framework Flutter à l' aide d'un code spécifique à la plateforme. Les fonctionnalités spécifiques à la plate-forme comme la caméra, le niveau de la batterie, le navigateur, etc., sont facilement accessibles via le cadre.
L'idée générale d'accéder au code spécifique à la plate-forme est via un protocole de messagerie simple. Le code Flutter, le client et le code de la plateforme et l'hôte se lient à un canal de message commun. Le client envoie un message à l'hôte via le canal de messages. L'hôte écoute sur le canal de message, reçoit le message et effectue les fonctionnalités nécessaires et, enfin, renvoie le résultat au client via le canal de message.
L'architecture de code spécifique à la plate-forme est illustrée dans le schéma fonctionnel ci-dessous -
Le protocole de messagerie utilise un codec de message standard (classe StandardMessageCodec) qui prend en charge la sérialisation binaire des valeurs de type JSON telles que les nombres, les chaînes, les valeurs booléennes, etc., La sérialisation et la désérialisation fonctionnent de manière transparente entre le client et l'hôte.
Écrivons une application simple pour ouvrir un navigateur à l'aide du SDK Android et comprendre comment
Créez une nouvelle application Flutter dans le studio Android, flutter_browser_app
Remplacez le code main.dart par le code ci-dessous -
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Flutter Demo Home Page'),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.title),
),
body: Center(
child: RaisedButton(
child: Text('Open Browser'),
onPressed: null,
),
),
);
}
}
Ici, nous avons créé un nouveau bouton pour ouvrir le navigateur et définir sa méthode onPressed sur null.
Maintenant, importez les packages suivants -
import 'dart:async';
import 'package:flutter/services.dart';
Ici, services.dart inclut la fonctionnalité pour appeler du code spécifique à la plate-forme.
Créez un nouveau canal de message dans le widget MyHomePage.
static const platform = const
MethodChannel('flutterapp.tutorialspoint.com/browser');
Écrivez une méthode, _openBrowser pour appeler une méthode spécifique à la plate-forme, une méthode openBrowser via le canal de message.
Future<void> _openBrowser() async {
try {
final int result = await platform.invokeMethod(
'openBrowser', <String, String>{
'url': "https://flutter.dev"
}
);
}
on PlatformException catch (e) {
// Unable to open the browser
print(e);
}
}
Ici, nous avons utilisé platform.invokeMethod pour appeler openBrowser (expliqué dans les étapes à venir). openBrowser a un argument, url pour ouvrir une URL spécifique.
Modifiez la valeur de la propriété onPressed du RaisedButton de null à _openBrowser.
onPressed: _openBrowser,
Ouvrez MainActivity.java (dans le dossier Android) et importez la bibliothèque requise -
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import io.flutter.app.FlutterActivity;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel;
import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
import io.flutter.plugin.common.MethodChannel.Result;
import io.flutter.plugins.GeneratedPluginRegistrant;
Ecrire une méthode, openBrowser pour ouvrir un navigateur
private void openBrowser(MethodCall call, Result result, String url) {
Activity activity = this;
if (activity == null) {
result.error("ACTIVITY_NOT_AVAILABLE",
"Browser cannot be opened without foreground
activity", null);
return;
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse(url));
activity.startActivity(intent);
result.success((Object) true);
}
Maintenant, définissez le nom du canal dans la classe MainActivity -
private static final String CHANNEL = "flutterapp.tutorialspoint.com/browser";
Écrivez un code spécifique à Android pour définir la gestion des messages dans la méthode onCreate -
new MethodChannel(getFlutterView(), CHANNEL).setMethodCallHandler(
new MethodCallHandler() {
@Override
public void onMethodCall(MethodCall call, Result result) {
String url = call.argument("url");
if (call.method.equals("openBrowser")) {
openBrowser(call, result, url);
} else {
result.notImplemented();
}
}
});
Ici, nous avons créé un canal de message à l'aide de la classe MethodChannel et utilisé la classe MethodCallHandler pour gérer le message. onMethodCall est la méthode réelle chargée d'appeler le bon code spécifique à la plate-forme en vérifiant le message. La méthode onMethodCall extrait l'url du message, puis appelle l'openBrowser uniquement lorsque l'appel de la méthode est openBrowser. Sinon, il renvoie la méthode notImplemented.
Le code source complet de l'application est le suivant -
main.dart
MainActivity.java
package com.tutorialspoint.flutterapp.flutter_browser_app;
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import io.flutter.app.FlutterActivity;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel.Result;
import io.flutter.plugins.GeneratedPluginRegistrant;
public class MainActivity extends FlutterActivity {
private static final String CHANNEL = "flutterapp.tutorialspoint.com/browser";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
GeneratedPluginRegistrant.registerWith(this);
new MethodChannel(getFlutterView(), CHANNEL).setMethodCallHandler(
new MethodCallHandler() {
@Override
public void onMethodCall(MethodCall call, Result result) {
String url = call.argument("url");
if (call.method.equals("openBrowser")) {
openBrowser(call, result, url);
} else {
result.notImplemented();
}
}
}
);
}
private void openBrowser(MethodCall call, Result result, String url) {
Activity activity = this; if (activity == null) {
result.error(
"ACTIVITY_NOT_AVAILABLE", "Browser cannot be opened without foreground activity", null
);
return;
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse(url));
activity.startActivity(intent);
result.success((Object) true);
}
}
main.dart
import 'package:flutter/material.dart';
import 'dart:async';
import 'package:flutter/services.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(
title: 'Flutter Demo Home Page'
),
);
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
static const platform = const MethodChannel('flutterapp.tutorialspoint.com/browser');
Future<void> _openBrowser() async {
try {
final int result = await platform.invokeMethod('openBrowser', <String, String>{
'url': "https://flutter.dev"
});
}
on PlatformException catch (e) {
// Unable to open the browser print(e);
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.title),
),
body: Center(
child: RaisedButton(
child: Text('Open Browser'),
onPressed: _openBrowser,
),
),
);
}
}
Exécutez l'application et cliquez sur le bouton Ouvrir le navigateur et vous pouvez voir que le navigateur est lancé. L'application Navigateur - La page d'accueil est comme indiqué dans la capture d'écran ici -
L'accès au code spécifique à iOS est similaire à celui de la plate-forme Android, sauf qu'il utilise des langages spécifiques à iOS - Objective-C ou Swift et iOS SDK. Sinon, le concept est le même que celui de la plateforme Android.
Écrivons également la même application que dans le chapitre précédent pour la plateforme iOS.
Créons une nouvelle application dans Android Studio (macOS), flutter_browser_ios_app
Suivez les étapes 2 à 6 comme dans le chapitre précédent.
Démarrez XCode et cliquez sur File → Open
Choisissez le projet xcode dans le répertoire ios de notre projet Flutter.
Ouvrez AppDelegate.m sous Runner → Runner path. Il contient le code suivant -
#include "AppDelegate.h"
#include "GeneratedPluginRegistrant.h"
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// [GeneratedPluginRegistrant registerWithRegistry:self];
// Override point for customization after application launch.
return [super application:application didFinishLaunchingWithOptions:launchOptions];
}
@end
Nous avons ajouté une méthode, openBrowser pour ouvrir le navigateur avec l'URL spécifiée. Il accepte un seul argument, url.
- (void)openBrowser:(NSString *)urlString {
NSURL *url = [NSURL URLWithString:urlString];
UIApplication *application = [UIApplication sharedApplication];
[application openURL:url];
}
Dans la méthode didFinishLaunchingWithOptions, recherchez le contrôleur et définissez-le dans la variable du contrôleur.
FlutterViewController* controller = (FlutterViewController*)self.window.rootViewController;
Dans la méthode didFinishLaunchingWithOptions, définissez le canal du navigateur sur flutterapp.tutorialspoint.com/browse -
FlutterMethodChannel* browserChannel = [
FlutterMethodChannel methodChannelWithName:
@"flutterapp.tutorialspoint.com/browser" binaryMessenger:controller];
Créer une variable, lowSelf et définir la classe courante -
__weak typeof(self) weakSelf = self;
Maintenant, implémentez setMethodCallHandler. Appelez openBrowser en faisant correspondre call.method. Obtenez l'URL en appelant call.arguments et transmettez-la en appelant openBrowser.
[browserChannel setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
if ([@"openBrowser" isEqualToString:call.method]) {
NSString *url = call.arguments[@"url"];
[weakSelf openBrowser:url];
} else { result(FlutterMethodNotImplemented); }
}];
Le code complet est le suivant -
#include "AppDelegate.h"
#include "GeneratedPluginRegistrant.h"
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// custom code starts
FlutterViewController* controller = (FlutterViewController*)self.window.rootViewController;
FlutterMethodChannel* browserChannel = [
FlutterMethodChannel methodChannelWithName:
@"flutterapp.tutorialspoint.com /browser" binaryMessenger:controller];
__weak typeof(self) weakSelf = self;
[browserChannel setMethodCallHandler:^(
FlutterMethodCall* call, FlutterResult result) {
if ([@"openBrowser" isEqualToString:call.method]) {
NSString *url = call.arguments[@"url"];
[weakSelf openBrowser:url];
} else { result(FlutterMethodNotImplemented); }
}];
// custom code ends
[GeneratedPluginRegistrant registerWithRegistry:self];
// Override point for customization after application launch.
return [super application:application didFinishLaunchingWithOptions:launchOptions];
}
- (void)openBrowser:(NSString *)urlString {
NSURL *url = [NSURL URLWithString:urlString];
UIApplication *application = [UIApplication sharedApplication];
[application openURL:url];
}
@end
Ouvrez les paramètres du projet.
Aller à Capabilities et activer Background Modes.
Ajouter *Background fetch et Remote Notification**.
Maintenant, exécutez l'application. Cela fonctionne de la même manière que la version Android, mais le navigateur Safari sera ouvert à la place de Chrome.
La façon dont Dart organise et partage un ensemble de fonctionnalités est via Package. Dart Package est simplement des bibliothèques ou des modules partageables. En général, le package Dart est le même que celui de l'application Dart, sauf que le package Dart n'a pas de point d'entrée d'application, main.
La structure générale de Package (considérez un package de démonstration, my_demo_package) est comme ci-dessous -
lib/src/* - Fichiers de code Dart privés.
lib/my_demo_package.dart- Fichier de code principal de Dart. Il peut être importé dans une application en tant que -
import 'package:my_demo_package/my_demo_package.dart'
Un autre fichier de code privé peut être exporté dans le fichier de code principal (my_demo_package.dart), si nécessaire comme indiqué ci-dessous -
export src/my_private_code.dart
lib/*- N'importe quel nombre de fichiers de code Dart organisés dans n'importe quelle structure de dossier personnalisée. Le code est accessible comme suit:
import 'package:my_demo_package/custom_folder/custom_file.dart'
pubspec.yaml - Spécification du projet, identique à celle de l'application,
Tous les fichiers de code Dart dans le package sont simplement des classes Dart et il n'y a aucune exigence particulière pour un code Dart pour l'inclure dans un package.
Types de forfaits
Étant donné que les packages Dart sont essentiellement un petit ensemble de fonctionnalités similaires, ils peuvent être classés en fonction de leurs fonctionnalités.
Paquet de fléchettes
Code Dart générique, qui peut être utilisé à la fois dans un environnement Web et mobile. Par exemple, english_words est un tel paquet qui contient environ 5000 mots et a des fonctions utilitaires de base comme les noms (lister les noms en anglais), les syllabes (spécifier le nombre de syllabes dans un mot.
Forfait Flutter
Code Dart générique, qui dépend du framework Flutter et ne peut être utilisé que dans un environnement mobile. Par exemple, fluro est un routeur personnalisé pour le flutter. Cela dépend du framework Flutter.
Plugin Flutter
Code Dart générique, qui dépend du framework Flutter ainsi que du code de plate-forme sous-jacent (SDK Android ou SDK iOS). Par exemple, la caméra est un plugin pour interagir avec la caméra de l'appareil. Cela dépend du framework Flutter ainsi que du framework sous-jacent pour accéder à la caméra.
Utilisation d'un package Dart
Les packages Dart sont hébergés et publiés sur le serveur en direct, https://pub.dartlang.org.En outre, Flutter fournit un outil simple, pub pour gérer les packages Dart dans l'application. Les étapes nécessaires pour utiliser comme package sont les suivantes -
Incluez le nom du package et la version nécessaire dans le pubspec.yaml comme indiqué ci-dessous -
dependencies: english_words: ^3.1.5
Le dernier numéro de version peut être trouvé en vérifiant le serveur en ligne.
Installez le package dans l'application en utilisant la commande suivante -
flutter packages get
Lors du développement dans le studio Android, Android Studio détecte tout changement dans pubspec.yaml et affiche une alerte de package de studio Android au développeur comme indiqué ci-dessous -
Les packages Dart peuvent être installés ou mis à jour dans Android Studio à l'aide des options de menu.
Importez le fichier nécessaire à l'aide de la commande ci-dessous et commencez à travailler -
import 'package:english_words/english_words.dart';
Utilisez n'importe quelle méthode disponible dans le package,
nouns.take(50).forEach(print);
Ici, nous avons utilisé la fonction des noms pour obtenir et imprimer les 50 premiers mots.
Développer un package de plugins Flutter
Le développement d'un plugin Flutter est similaire au développement d'une application Dart ou d'un package Dart. La seule exception est que le plugin utilisera l'API système (Android ou iOS) pour obtenir les fonctionnalités spécifiques à la plate-forme requises.
Comme nous l'avons déjà appris à accéder au code de la plateforme dans les chapitres précédents, développons un simple plugin, my_browser pour comprendre le processus de développement du plugin. La fonctionnalité du plugin my_browser est de permettre à l'application d'ouvrir le site Web donné dans le navigateur spécifique à la plate-forme.
Démarrez Android Studio.
Cliquez sur File → New Flutter Project et sélectionnez l'option Flutter Plugin.
Vous pouvez voir une fenêtre de sélection du plugin Flutter comme indiqué ici -
Entrez my_browser comme nom de projet et cliquez sur Suivant.
Entrez le nom du plugin et d'autres détails dans la fenêtre comme indiqué ici -
Entrez le domaine de l'entreprise, flutterplugins.tutorialspoint.com dans la fenêtre ci-dessous, puis cliquez sur Finish. Il générera un code de démarrage pour développer notre nouveau plugin.
Ouvrez le fichier my_browser.dart et écrivez une méthode, openBrowser pour appeler la méthode openBrowser spécifique à la plate-forme.
Future<void> openBrowser(String urlString) async {
try {
final int result = await _channel.invokeMethod(
'openBrowser', <String, String>{ 'url': urlString }
);
}
on PlatformException catch (e) {
// Unable to open the browser print(e);
}
}
Ouvrez le fichier MyBrowserPlugin.java et importez les classes suivantes -
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
Ici, nous devons importer la bibliothèque requise pour ouvrir un navigateur depuis Android.
Ajoutez une nouvelle variable privée mRegistrar de type Registrar dans la classe MyBrowserPlugin.
private final Registrar mRegistrar;
Ici, Registrar est utilisé pour obtenir des informations de contexte sur le code appelant.
Ajoutez un constructeur pour définir Registrar dans la classe MyBrowserPlugin.
private MyBrowserPlugin(Registrar registrar) {
this.mRegistrar = registrar;
}
Modifiez registerWith pour inclure notre nouveau constructeur dans la classe MyBrowserPlugin.
public static void registerWith(Registrar registrar) {
final MethodChannel channel = new MethodChannel(registrar.messenger(), "my_browser");
MyBrowserPlugin instance = new MyBrowserPlugin(registrar);
channel.setMethodCallHandler(instance);
}
Modifiez onMethodCall pour inclure la méthode openBrowser dans la classe MyBrowserPlugin.
@Override
public void onMethodCall(MethodCall call, Result result) {
String url = call.argument("url");
if (call.method.equals("getPlatformVersion")) {
result.success("Android " + android.os.Build.VERSION.RELEASE);
}
else if (call.method.equals("openBrowser")) {
openBrowser(call, result, url);
} else {
result.notImplemented();
}
}
Écrivez la méthode openBrowser spécifique à la plate-forme pour accéder au navigateur dans la classe MyBrowserPlugin.
private void openBrowser(MethodCall call, Result result, String url) {
Activity activity = mRegistrar.activity();
if (activity == null) {
result.error("ACTIVITY_NOT_AVAILABLE",
"Browser cannot be opened without foreground activity", null);
return;
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse(url));
activity.startActivity(intent);
result.success((Object) true);
}
Le code source complet du plugin my_browser est le suivant -
my_browser.dart
import 'dart:async';
import 'package:flutter/services.dart';
class MyBrowser {
static const MethodChannel _channel = const MethodChannel('my_browser');
static Future<String> get platformVersion async {
final String version = await _channel.invokeMethod('getPlatformVersion'); return version;
}
Future<void> openBrowser(String urlString) async {
try {
final int result = await _channel.invokeMethod(
'openBrowser', <String, String>{'url': urlString});
}
on PlatformException catch (e) {
// Unable to open the browser print(e);
}
}
}
MyBrowserPlugin.java
package com.tutorialspoint.flutterplugins.my_browser;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel;
import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
import io.flutter.plugin.common.MethodChannel.Result;
import io.flutter.plugin.common.PluginRegistry.Registrar;
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
/** MyBrowserPlugin */
public class MyBrowserPlugin implements MethodCallHandler {
private final Registrar mRegistrar;
private MyBrowserPlugin(Registrar registrar) {
this.mRegistrar = registrar;
}
/** Plugin registration. */
public static void registerWith(Registrar registrar) {
final MethodChannel channel = new MethodChannel(
registrar.messenger(), "my_browser");
MyBrowserPlugin instance = new MyBrowserPlugin(registrar);
channel.setMethodCallHandler(instance);
}
@Override
public void onMethodCall(MethodCall call, Result result) {
String url = call.argument("url");
if (call.method.equals("getPlatformVersion")) {
result.success("Android " + android.os.Build.VERSION.RELEASE);
}
else if (call.method.equals("openBrowser")) {
openBrowser(call, result, url);
} else {
result.notImplemented();
}
}
private void openBrowser(MethodCall call, Result result, String url) {
Activity activity = mRegistrar.activity();
if (activity == null) {
result.error("ACTIVITY_NOT_AVAILABLE",
"Browser cannot be opened without foreground activity", null);
return;
}
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse(url));
activity.startActivity(intent);
result.success((Object) true);
}
}
Créez un nouveau projet, my_browser_plugin_test pour tester notre plugin nouvellement créé.
Ouvrez pubspec.yaml et définissez my_browser comme dépendance de plugin.
dependencies:
flutter:
sdk: flutter
my_browser:
path: ../my_browser
Le studio Android alertera que pubspec.yaml est mis à jour comme indiqué dans l'alerte de package de studio Android donnée ci-dessous -
Cliquez sur l'option Obtenir les dépendances. Le studio Android obtiendra le package sur Internet et le configurera correctement pour l'application.
Ouvrez main.dart et incluez le plugin my_browser comme ci-dessous -
import 'package:my_browser/my_browser.dart';
Appelez la fonction openBrowser à partir du plugin my_browser comme indiqué ci-dessous -
onPressed: () => MyBrowser().openBrowser("https://flutter.dev"),
Le code complet du main.dart est le suivant -
import 'package:flutter/material.dart';
import 'package:my_browser/my_browser.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(
title: 'Flutter Demo Home Page'
),
);,
}
}
class MyHomePage extends StatelessWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(this.title),
),
body: Center(
child: RaisedButton(
child: Text('Open Browser'),
onPressed: () => MyBrowser().openBrowser("https://flutter.dev"),
),
),
);
}
}
Exécutez l'application et cliquez sur le bouton Ouvrir le navigateur et vérifiez que le navigateur est lancé. Vous pouvez voir une application Navigateur - Page d'accueil comme indiqué dans la capture d'écran ci-dessous -
Vous pouvez voir une application Navigateur - Écran Navigateur comme indiqué dans la capture d'écran ci-dessous -
- Ajouter des notes
- Marquer cette page
- Signaler une erreur
- Suggestions
Enregistrer Fermer