ساخت اپلیکیشن یادداشت با فلاتر و دارت — از صفر تا صد

فلاتر یک فریمورک توسعه موبایل چندپلتفرمی اوپن سورس است که از سوی گوگل عرضه شده است. اپلیکیشنهای فلاتر با دارت نوشته میشوند. فلاتر به صورت پیشفرض مجهز به کامپوننتهای «متریال دیزاین» (Material Design) است و همین امر موجب شده است تا ساخت اپلیکیشن با ظاهر و حس خوب با استفاده از فلاتر بسیار آسان باشد. در فلاتر هر چیزی یک ویجت از نوع باحالت یا بیحالت محسوب میشود. در این راهنما به عنوان یک پروژه برای شروع یادگیری فلاتر، اقدام به ساخت اپلیکیشن یادداشت با فلاتر و دارت خواهیم کرد.
اگر هنوز فلاتر را روی سیستم خود نصب نکردهاید، آن را به همراه یک IDE پشتیبانیشده نصب کنید. راهنماییهای لازم در این صفحه (+) انجام یافته است.
ابتدا پروژه را تنظیم میکنیم. مراحل کار به صورت زیر است:
- یک پروژه فلاتر در اندروید استودیو ایجاد کنید یا دستور flutter create notes را در ترمینال یا CMD وارد نمایید.
- در فایل main.dart کلاس homepage را حذف کرده و یک فایل جدید با کلاس homepage خودتان ایجاد کنید که Stateful Widget را بسط دهد. این کلاس شامل چارچوب کلی اپلیکیشن ما خواهد بود.
- کلاس ویجت باحالت دیگری ایجاد کنید. این کلاس شامل بخش Body است که یک نمای Staggered را برای Home در خود جای داده است. نام آن را StaggeredGridPage میگذاریم.
در این اپلیکیشن تلاش میکنیم که خلاقیت به خرج بدهیم و یادداشتها را به روش Staggered جالبی نمایش دهیم. از این پکیج دارت برای (+) ایجاد نمای شبکهای Staggered استفاده میکنیم. از SQLite نیز برای ذخیره دادههای یادداشتها روی دستگاه استفاده میکنیم.
در ادامه قطعه کدی را از pubspec.yaml میبینید که وابستگیهای فهرست شده را الزام کرده است. آنها را اضافه کرده، فایل را ذخیره کنید و از دستور فلاتر flutter packages get برای نصب وابستگیهای اضافه شده جدید استفاده کنید.
dependencies: flutter: sdk: flutter cupertino_icons: ^0.1.2 flutter_staggered_grid_view: ^0.2.7 auto_size_text: ^1.1.2 sqflite: path: intl: ^0.15.7 share: ^0.6.1
یک کلاس برای یادداشتها ایجاد کنید. ما به تابع toMap برای کوئریهای پایگاه داده نیاز داریم.
- فایل note.dart
class Note { int id; String title; String content; DateTime date_created; DateTime date_last_edited; Color note_color; int is_archived = 0; Note(this.id, this.title, this.content, this.date_created, this.date_last_edited,this.note_color); Map<String, dynamic> toMap(bool forUpdate) { var data = { // 'id': id, since id is auto incremented in the database we don't need to send it to the insert query. 'title': utf8.encode(title), 'content': utf8.encode( content ), 'date_created': epochFromDate( date_created ), 'date_last_edited': epochFromDate( date_last_edited ), 'note_color': note_color.value, 'is_archived': is_archived // for later use for integrating archiving }; if(forUpdate){ data["id"] = this.id; } return data; } // Converting the date time object into int representing seconds passed after midnight 1st Jan, 1970 UTC int epochFromDate(DateTime dt) { return dt.millisecondsSinceEpoch ~/ 1000; } void archiveThisNote(){ is_archived = 1; } }
کد کوئریهای پایگاه داده SQLite برای کلاس note فوق و جدول مربوطه به صورت زیر است:
- فایل SqliteHandler.dart
import 'package:sqflite/sqflite.dart'; import 'package:path/path.dart'; import 'package:sqflite/sqlite_api.dart'; import 'dart:async'; import 'Note.dart'; class NotesDBHandler { final databaseName = "notes.db"; final tableName = "notes"; final fieldMap = { "id": "INTEGER PRIMARY KEY AUTOINCREMENT", "title": "BLOB", "content": "BLOB", "date_created": "INTEGER", "date_last_edited": "INTEGER", "note_color": "INTEGER", "is_archived": "INTEGER" }; static Database _database; Future<Database> get database async { if (_database != null) return _database; _database = await initDB(); return _database; } initDB() async { var path = await getDatabasesPath(); var dbPath = join(path, 'notes.db'); // ignore: argument_type_not_assignable Database dbConnection = await openDatabase( dbPath, version: 1, onCreate: (Database db, int version) async { print("executing create query from onCreate callback"); await db.execute(_buildCreateQuery()); }); await dbConnection.execute(_buildCreateQuery()); _buildCreateQuery(); return dbConnection; } // build the create query dynamically using the column:field dictionary. String _buildCreateQuery() { String query = "CREATE TABLE IF NOT EXISTS "; query += tableName; query += "("; fieldMap.forEach((column, field){ print("$column : $field"); query += "$column $field,"; }); query = query.substring(0, query.length-1); query += " )"; return query; } static Future<String> dbPath() async { String path = await getDatabasesPath(); return path; } Future<int> insertNote(Note note, bool isNew) async { // Get a reference to the database final Database db = await database; print("insert called"); // Insert the Notes into the correct table. await db.insert('notes', isNew ? note.toMap(false) : note.toMap(true), conflictAlgorithm: ConflictAlgorithm.replace, ); if (isNew) { // get latest note which isn't archived, limit by 1 var one = await db.query("notes", orderBy: "date_last_edited desc", where: "is_archived = ?", whereArgs: [0], limit: 1); int latestId = one.first["id"] as int; return latestId; } return note.id; } Future<bool> copyNote(Note note) async { final Database db = await database; try { await db.insert("notes",note.toMap(false), conflictAlgorithm: ConflictAlgorithm.replace); } catch(Error) { print(Error); return false; } return true; } Future<bool> archiveNote(Note note) async { if (note.id != -1) { final Database db = await database; int idToUpdate = note.id; db.update("notes", note.toMap(true), where: "id = ?", whereArgs: [idToUpdate]); } } Future<bool> deleteNote(Note note) async { if(note.id != -1) { final Database db = await database; try { await db.delete("notes",where: "id = ?",whereArgs: [note.id]); return true; } catch (Error){ print("Error deleting ${note.id}: ${Error.toString()}"); return false; } } } Future<List<Map<String,dynamic>>> selectAllNotes() async { final Database db = await database; // query all the notes sorted by last edited var data = await db.query("notes", orderBy: "date_last_edited desc", where: "is_archived = ?", whereArgs: [0]); return data; } }
اینک صفحه اصلی اپلیکیشن متریال باید یک چارچوب (Scaffold) از فایل HomePage.dart داشته باشد که بدنه آن به صورت StaggeredGridView است. در بخش AppBar این چارچوب یک دکمه اکشن قرار میدهیم تا کاربر بتواند بین حالتهای نمایش لیستی و Staggered انتخاب کند. فراموش نکنید که Body را درون SafeArea قرار دهید، چون میخواهیم اپلیکیشن روی گوشیهای مدرن نیز عملکرد مناسبی داشته باشد.
کتابخانه نمای Staggered تعدادی یادداشت برای نما الزام میکند که به صورت دینامیک بر مبنای عرض اندازه صفحه نمایش تعیین میشود. این وضعیت نیازمند این است که تعداد یادداشتهایی که قرار است در کنار هم نمایش یابند، معین شده باشد. در وضعیت افقی گوشی یا روی تبلت، تعداد یادداشتها را به صورت افقی روی 3 عدد و برای وضعیت عمودی روی گوشی روی عدد 2 تنظیم میکنیم.
- فایل StaggeredView.dart
import 'dart:convert'; import 'package:flutter/material.dart'; import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart'; import '../Models/Note.dart'; import '../Models/SqliteHandler.dart'; import '../Models/Utility.dart'; import '../Views/StaggeredTiles.dart'; import 'HomePage.dart'; class StaggeredGridPage extends StatefulWidget { final notesViewType; const StaggeredGridPage({Key key, this.notesViewType}) : super(key: key); @override _StaggeredGridPageState createState() => _StaggeredGridPageState(); } class _StaggeredGridPageState extends State<StaggeredGridPage> { var noteDB = NotesDBHandler(); List<Map<String, dynamic>> _allNotesInQueryResult = []; viewType notesViewType ; @override void initState() { super.initState(); this.notesViewType = widget.notesViewType; } @override void setState(fn) { super.setState(fn); this.notesViewType = widget.notesViewType; } @override Widget build(BuildContext context) { GlobalKey _stagKey = GlobalKey(); if(CentralStation.updateNeeded) { retrieveAllNotesFromDatabase(); } return Container(child: Padding(padding: _paddingForView(context) , child: new StaggeredGridView.count(key: _stagKey, crossAxisSpacing: 6, mainAxisSpacing: 6, crossAxisCount: _colForStaggeredView(context), children: List.generate(_allNotesInQueryResult.length, (i){ return _tileGenerator(i); }), staggeredTiles: _tilesForView() , ), ) ); } int _colForStaggeredView(BuildContext context) { if (widget.notesViewType == viewType.List) { return 1; } // for width larger than 600, return 3 irrelevant of the orientation to accommodate more notes horizontally return MediaQuery.of(context).size.width > 600 ? 3 : 2 ; } List<StaggeredTile> _tilesForView() { // Generate staggered tiles for the view based on the current preference. return List.generate(_allNotesInQueryResult.length,(index){ return StaggeredTile.fit( 1 ); } ) ; } EdgeInsets _paddingForView(BuildContext context){ double width = MediaQuery.of(context).size.width; double padding ; double top_bottom = 8; if (width > 500) { padding = ( width ) * 0.05 ; // 5% padding of width on both side } else { padding = 8; } return EdgeInsets.only(left: padding, right: padding, top: top_bottom, bottom: top_bottom); } MyStaggeredTile _tileGenerator(int i){ return MyStaggeredTile( Note( _allNotesInQueryResult[i]["id"], _allNotesInQueryResult[i]["title"] == null ? "" : utf8.decode(_allNotesInQueryResult[i]["title"]), _allNotesInQueryResult[i]["content"] == null ? "" : utf8.decode(_allNotesInQueryResult[i]["content"]), DateTime.fromMillisecondsSinceEpoch(_allNotesInQueryResult[i]["date_created"] * 1000), DateTime.fromMillisecondsSinceEpoch(_allNotesInQueryResult[i]["date_last_edited"] * 1000), Color(_allNotesInQueryResult[i]["note_color"] )) ); } void retrieveAllNotesFromDatabase() { // queries for all the notes from the database ordered by latest edited note. excludes archived notes. var _testData = noteDB.testSelect(); _testData.then((value){ setState(() { this._allNotesInQueryResult = value; CentralStation.updateNeeded = false; }); }); } }
این نما به کاشیهایی (Tiles) برای نمایش یادداشتها نیاز دارد. آن کاشی که ما برای نما طراحی میکنیم باید عنوان و محتوای یادداشت را به صورت پیشنمایش ارائه کند. برای مدیریت طول مختلف متن یادداشت از یک کتابخانه (+) جهت ایجاد نمای متنی با بسط خودکار استفاده میکنیم. کافی است محدودیت خط را تعریف کنیم تا ویجت به صورت خودکار بسط یابد و محتوا را تا جایی که به این محدودیت میرسد، نمایش دهد.
همانند segue در iOS و Intent در اندروید، برای ناوبری بین صفحهها در فلاتر از Navigator استفاده میکنیم.
- فایل rawStaggeredTile.dart
import 'package:flutter/material.dart'; import 'package:auto_size_text/auto_size_text.dart'; import '../ViewControllers/NotePage.dart'; import '../Models/Note.dart'; import '../Models/Utility.dart'; class MyStaggeredTile extends StatefulWidget { final Note note; MyStaggeredTile(this.note); @override _MyStaggeredTileState createState() => _MyStaggeredTileState(); } class _MyStaggeredTileState extends State<MyStaggeredTile> { String _content ; double _fontSize ; Color tileColor ; String title; @override Widget build(BuildContext context) { _content = widget.note.content; _fontSize = _determineFontSizeForContent(); tileColor = widget.note.note_color; title = widget.note.title; return GestureDetector( onTap: ()=> _noteTapped(context), child: Container( decoration: BoxDecoration( border: tileColor == Colors.white ? Border.all(color: CentralStation.borderColor) : null, color: tileColor, borderRadius: BorderRadius.all(Radius.circular(8))), padding: EdgeInsets.all(8), child: constructChild(),) , ); } void _noteTapped(BuildContext ctx) { CentralStation.updateNeeded = false; Navigator.push(ctx, MaterialPageRoute(builder: (ctx) => NotePage(widget.note))); } Widget constructChild() { List<Widget> contentsOfTiles = []; if(widget.note.title.length != 0) { contentsOfTiles.add( AutoSizeText(title, style: TextStyle(fontSize: _fontSize,fontWeight: FontWeight.bold), maxLines: widget.note.title.length == 0 ? 1 : 3, textScaleFactor: 1.5, ), ); contentsOfTiles.add(Divider(color: Colors.transparent,height: 6,),); } contentsOfTiles.add( AutoSizeText( _content, style: TextStyle(fontSize: _fontSize), maxLines: 10, textScaleFactor: 1.5,) ); return Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisAlignment: MainAxisAlignment.start, children: contentsOfTiles ); } double _determineFontSizeForContent() { int charCount = _content.length + widget.note.title.length ; double fontSize = 20 ; if (charCount > 110 ) { fontSize = 12; } else if (charCount > 80) { fontSize = 14; } else if (charCount > 50) { fontSize = 16; } else if (charCount > 20) { fontSize = 18; } return fontSize; } }
بدین ترتیب کاشیها در نما، ظاهری مانند زیر پیدا میکنند:
اکنون باید یک نما برای ویرایش/ایجاد یادداشت داشته باشیم. در این نما اکشنهای مختلفی برای Undo، آرشیو کردن و موارد دیگر روی یادداشتها در AppBar تعریف میکنیم. اکشنهای دیگری نیز در بخش تحتانی صفحه به صورت گزینههایی برای اشتراک، کپی گرفتن، حذف دائمی و یک انتخابگر رنگ افقی عرضه میشود که امکان تغییر رنگ پسزمینه آن یادداشت خاص را فراهم میسازد.
بدین ترتیب ویجتهای NotePage, BottomSheet و ColorSlider را در کلاسها و فایلهای مختلف قرار میدهیم تا کد تمیزتر و منسجمتر بماند. در زمان انتخاب رنگ از سوی کاربر در ColorSlider، برای تغییر دادن رنگ در همه این ویجتها باید «حالت» (State) را بهروزرسانی کنیم. امکان اتصال این سه ویجت از طریق تابعهای Callback برای پاسخ دادن به تغییرات و بهروزرسانی خودشان وجود دارد.
- فایل rawColorSlider.dart
import 'package:flutter/material.dart'; class ColorSlider extends StatefulWidget { final void Function(Color) callBackColorTapped ; final Color noteColor ; ColorSlider({@required this.callBackColorTapped, @required this.noteColor}); @override _ColorSliderState createState() => _ColorSliderState(); } class _ColorSliderState extends State<ColorSlider> { final colors = [ Color(0xffffffff), // classic white Color(0xfff28b81), // light pink Color(0xfff7bd02), // yellow Color(0xfffbf476), // light yellow Color(0xffcdff90), // light green Color(0xffa7feeb), // turquoise Color(0xffcbf0f8), // light cyan Color(0xffafcbfa), // light blue Color(0xffd7aefc), // plum Color(0xfffbcfe9), // misty rose Color(0xffe6c9a9), // light brown Color(0xffe9eaee) // light gray ]; final Color borderColor = Color(0xffd3d3d3); final Color foregroundColor = Color(0xff595959); final _check = Icon(Icons.check); Color noteColor; int indexOfCurrentColor; @override void initState() { super.initState(); this.noteColor = widget.noteColor; indexOfCurrentColor = colors.indexOf(noteColor); } @override Widget build(BuildContext context) { return ListView( scrollDirection: Axis.horizontal, children: List.generate(colors.length, (index) { return GestureDetector( onTap: ()=> _colorChangeTapped(index), child: Padding( padding: EdgeInsets.only(left: 6, right: 6), child:Container( child: new CircleAvatar( child: _checkOrNot(index), foregroundColor: foregroundColor, backgroundColor: colors[index], ), width: 38.0, height: 38.0, padding: const EdgeInsets.all(1.0), // border width decoration: new BoxDecoration( color: borderColor, // border color shape: BoxShape.circle, ) ) ) ); }) ,); } void _colorChangeTapped(int indexOfColor) { setState(() { noteColor = colors[indexOfColor]; indexOfCurrentColor = indexOfColor; widget.callBackColorTapped(colors[indexOfColor]); }); } Widget _checkOrNot(int index){ if (indexOfCurrentColor == index) { return _check; } return null; } }
در نهایت برخی قابلیتهای مفید دیگر از قبیل Undo کردن تغییرها، آرشیو کردن، اشتراک، کپی کردن یادداشت و حذف دائمی آن را نیز به اپلیکیشن اضافه کردیم. کد کامل این اپلیکیشن فلاتر را میتوانید از این ریپازیتوری گیتهاب (+) دانلود کنید و تغییرهای مورد نظر خود را روی آن اعمال کنید.
اگر این مطلب برای شما مفید بوده است، آموزشهای زیر نیز به شما پیشنهاد میشوند:
- مجموعه آموزشهای برنامه نویسی اندروید
- مجموعه آموزشهای برنامهنویسی
- آموزش فریم ورک Google Flutter برای طراحی اپلیکیشن های موبایل
- مفاهیم مقدماتی فلاتر (Flutter) — به زبان ساده
- گوگل فلاتر (Flutter) از صفر تا صد — ساخت اپلیکیشن به کمک ویجت
==