Skip to content

Latest commit

 

History

History
1463 lines (1184 loc) · 47.9 KB

README.md

File metadata and controls

1463 lines (1184 loc) · 47.9 KB

🏕️ Adventurer's Inventory 🎒

Daffa Mohamad Fathoni 2206824161 PBP E

Tugas 9

Apakah bisa kita melakukan pengambilan data JSON tanpa membuat model terlebih dahulu? Jika iya, apakah hal tersebut lebih baik daripada membuat model sebelum melakukan pengambilan data JSON?

Pengambilan data JSON sendiri tidak memerlukan pembuatan model terlebih dahulu. JSON (JavaScript Object Notation) adalah format data ringan yang digunakan untuk pertukaran data antar aplikasi. Jadi, Anda dapat mengambil data JSON langsung tanpa membangun model khusus.

Namun, apakah Anda memerlukan pembuatan model tergantung pada tujuan pengambilan data tersebut. Jika Anda hanya ingin membaca atau mengambil data dari file JSON atau endpoint API, Anda tidak perlu membuat model. Anda dapat menggunakan bahasa pemrograman atau alat yang mendukung manipulasi data JSON, seperti Python dengan library json atau JavaScript dengan JSON.parse().

Jelaskan fungsi dari CookieRequest dan jelaskan mengapa instance CookieRequest perlu untuk dibagikan ke semua komponen di aplikasi Flutter.

Dalam konteks aplikasi Flutter yang berkomunikasi dengan server, penggunaan cookies seringkali terlibat dalam mekanisme otentikasi atau manajemen sesi. Cookies dapat digunakan untuk menyimpan informasi otentikasi atau data sesi yang diperlukan oleh server.

Namun, secara umum, jika Anda memiliki kebutuhan untuk menggunakan cookies di seluruh aplikasi Flutter Anda, Anda mungkin ingin membuat singleton atau objek yang dapat diakses secara global untuk menangani manajemen cookies tersebut. Alasan untuk ini dapat mencakup:

Konsistensi: Memiliki satu instance yang menangani manajemen cookies dapat memastikan konsistensi pengelolaan cookies di seluruh aplikasi. Jika Anda membuat instance CookieRequest dan membagikannya ke semua komponen, Anda dapat memastikan bahwa semua komponen menggunakan cookies yang sama.

Dapat Dikonfigurasi Secara Sentral: Dengan menggunakan singleton atau objek yang dapat diakses secara global, Anda dapat dengan mudah menambahkan konfigurasi atau opsi terkait cookies di satu tempat. Ini memudahkan pengelolaan dan perubahan konfigurasi di masa depan.

Kemudahan Penggunaan: Dengan menggunakan satu instance untuk menangani cookies, Anda mengurangi kebutuhan untuk melewatkan atau menyimpan cookies di banyak tempat. Ini dapat meningkatkan kemudahan penggunaan dan pemeliharaan.

Contoh sederhana dalam Flutter bisa terlihat seperti ini:

final request = context.watch<CookieRequest>();

Dengan menggunakan CookieRequest seperti di atas, Anda dapat membuat instance CookieRequest di satu tempat dan menggunakannya di seluruh aplikasi Anda untuk menangani cookies.

Jelaskan mekanisme pengambilan data dari JSON hingga dapat ditampilkan pada Flutter.

Mekanisme pengambilan data dari JSON hingga dapat ditampilkan pada aplikasi Flutter melibatkan beberapa langkah utama. Ini mencakup pengiriman request HTTP ke server, penerimaan response dalam format JSON, parsing data JSON tersebut, dan akhirnya menampilkan data dalam UI Flutter. Berikut adalah penjelasan rinci dari setiap langkah:

Mengirim Request HTTP: Pertama, aplikasi Flutter mengirimkan request HTTP ke server atau endpoint API yang menyediakan data JSON. Biasanya, ini dilakukan menggunakan paket http yang tersedia di Flutter.

Contoh kode untuk mengirim request:

Copy code
var url = Uri.parse('https://example.com/data');
var response = await http.get(url);

Menerima Response JSON: Setelah request terkirim, aplikasi akan menerima response dari server. Jika request berhasil, response ini akan berisi data dalam format JSON.

Menguraikan (Parsing) JSON: Langkah selanjutnya adalah menguraikan response JSON tersebut menjadi struktur data yang bisa digunakan oleh Flutter. Ada dua cara umum untuk melakukan ini:

Menggunakan Dynamic Typing: Menguraikan JSON menjadi Map<String, dynamic> atau List menggunakan jsonDecode dari dart:convert. Metode ini tidak memerlukan pembuatan model kelas terlebih dahulu.

Contoh:

Copy code
var jsonData = jsonDecode(response.body);

Menggunakan Model Kelas: Membuat model kelas untuk merepresentasikan data JSON dan menggunakan jsonDecode untuk menguraikan JSON ke dalam instance dari model tersebut. Ini memberikan keuntungan dari type safety dan kemudahan pengelolaan.

Contoh model:

Copy code
class DataModel {
  final String id;
  final String name;
  // Konstruktor dan metode lainnya
}

Contoh penguraian:

Copy code
var dataModel = DataModel.fromJson(jsonDecode(response.body));

Menampilkan Data pada UI: Setelah data JSON berhasil diuraikan, langkah terakhir adalah menampilkannya dalam UI Flutter. Ini bisa dilakukan dengan menggunakan berbagai widget Flutter, seperti Text, ListView, Card, dll.

Contoh:

Copy code
ListView.builder(
  itemCount: data.length,
  itemBuilder: (context, index) {
    return ListTile(
      title: Text(data[index].name),
      // Widget lainnya
    );
  },
)

Kesimpulan: Proses pengambilan data dari JSON hingga ditampilkan pada Flutter melibatkan beberapa langkah penting, mulai dari pengiriman request, parsing JSON, dan akhirnya menampilkan data dalam UI. Dengan menggunakan paket http untuk request dan dart:convert untuk parsing JSON, Flutter menyediakan cara yang efisien dan fleksibel untuk menangani data JSON.

Jelaskan mekanisme autentikasi dari input data akun pada Flutter ke Django hingga selesainya proses autentikasi oleh Django dan tampilnya menu pada Flutter.

Mekanisme autentikasi dari input data akun pada Flutter hingga proses autentikasi oleh Django dan tampilnya menu pada Flutter melibatkan beberapa langkah terintegrasi. Berikut adalah langkah-langkah tersebut:

Input Data Akun pada Flutter: Pengguna memasukkan data akun seperti username dan password melalui antarmuka pengguna (UI) Flutter. Ini umumnya dilakukan menggunakan widget seperti TextFormField atau input widget serupa.

Mengirim Data ke Server Django: Setelah pengguna menekan tombol login, aplikasi Flutter akan mengirimkan data ke server Django. Proses ini biasanya dilakukan dengan membuat request HTTP POST menggunakan paket seperti http di Flutter.

Contoh kode untuk mengirim data:

Copy code
var url = Uri.parse('http://yourdjangoapp.com/api/login/');
var response = await http.post(
  url,
  body: {'username': 'user', 'password': 'pass'}
);

Proses Autentikasi di Django: Di sisi server Django, data yang diterima akan diproses. Django akan memeriksa apakah kombinasi username dan password cocok dengan yang ada di database.

Jika valid, Django akan menghasilkan token (misalnya menggunakan Django Rest Framework Token Authentication) dan mengirimkannya sebagai response. Jika tidak valid, Django akan mengirimkan response error. Menerima Response di Flutter: Aplikasi Flutter akan menerima response dari Django. Jika autentikasi berhasil, aplikasi menyimpan token tersebut (biasanya di SharedPreferences untuk penggunaan selanjutnya). Jika gagal, aplikasi dapat menampilkan pesan kesalahan.

Contoh:

Copy code
if (authSuccess) {
  // Menyimpan token
  // Navigasi ke halaman utama atau menu utama aplikasi
} else {
  // Menampilkan pesan kesalahan
}

Menampilkan Menu di Flutter: Setelah autentikasi berhasil, aplikasi Flutter dapat menavigasikan pengguna ke halaman utama atau menu utama aplikasi. Ini biasanya dilakukan dengan menggunakan Navigator untuk mengubah halaman.

Contoh:

Copy code
if (authSuccess) {
  Navigator.pushReplacement(
    context,
    MaterialPageRoute(builder: (context) => MainMenu()),
  );
}

Penggunaan Token untuk Request Berikutnya: Token yang disimpan digunakan untuk autentikasi pada request-request berikutnya ke server Django. Header Authorization dengan token biasanya ditambahkan pada setiap request yang memerlukan autentikasi.

Contoh:

Copy code
var response = await http.get(
  protectedUrl,
  headers: {'Authorization': 'Token $yourToken'},
);

Kesimpulan: Proses autentikasi antara aplikasi Flutter dan server Django melibatkan pertukaran data antara klien dan server, validasi kredensial oleh Django, dan penggunaan token untuk sesi yang terautentikasi. Flutter bertanggung jawab untuk mengumpulkan data pengguna dan menampilkan UI sesuai dengan status autentikasi, sementara Django menangani verifikasi kredensial dan mengeluarkan token.

Sebutkan seluruh widget yang kamu pakai pada tugas ini dan jelaskan fungsinya masing-masing.

MaterialApp: Pada Flutter, MaterialApp digunakan sebagai titik awal aplikasi yang mengikuti panduan desain Material Design. Ini mengatur tema secara global untuk aplikasi dan menyediakan navigasi antar-halaman.

Scaffold: Scaffold adalah widget yang menyediakan struktur dasar layout Material. Ini mencakup elemen-elemen seperti app bar, body, floating action button, dan lainnya.

AppBar: AppBar digunakan untuk menampilkan bar aplikasi di bagian atas layar. Biasanya berisi judul aplikasi, ikon, dan tindakan yang dapat dilakukan oleh pengguna.

TextFormField: Widget ini memungkinkan pengguna memasukkan data melalui formulir teks. Umumnya digunakan untuk input seperti username dan password.

ListView: ListView adalah widget scrollable yang digunakan untuk menampilkan daftar item dalam bentuk linear, baik secara vertikal maupun horizontal.

Text: Text digunakan untuk menampilkan teks sederhana pada antarmuka pengguna.

Image: Image digunakan untuk menampilkan gambar dari berbagai sumber, seperti jaringan atau aset lokal.

Navigator: Navigator digunakan untuk mengatur rute halaman di aplikasi Flutter. Ini memungkinkan navigasi antar-halaman.

FutureBuilder: FutureBuilder membangun widget berdasarkan hasil terakhir dari interaksi Future, seperti request HTTP. Ini mempermudah penanganan pembaruan UI berdasarkan data yang diterima dari proses async.

Card: Card adalah widget Material dengan sudut melengkung dan bayangan. Biasanya digunakan untuk menampilkan konten secara terorganisir.

Padding: Padding digunakan untuk menambahkan padding di sekeliling widget anaknya, membantu mengontrol ruang di sekitar elemen.

Column/Row: Column dan Row digunakan untuk menata widget anaknya secara vertikal (Column) atau horizontal (Row).

Icon: Icon digunakan untuk menampilkan ikon dari set Material Icons.

IconButton: IconButton adalah tombol dengan area tekan yang menampilkan ikon. Sering digunakan di AppBar untuk aksi cepat.

CircularProgressIndicator: CircularProgressIndicator digunakan untuk menampilkan indikator loading berputar ketika aplikasi sedang melakukan tugas yang memerlukan waktu.

SharedPreferences: Bukan widget, tetapi digunakan untuk menyimpan data secara lokal di perangkat. Berguna untuk menyimpan preferensi pengguna atau data aplikasi kecil lainnya.

Jelaskan bagaimana cara kamu mengimplementasikan checklist di atas secara step-by-step! (bukan hanya sekadar mengikuti tutorial).
  • Memastikan deployment proyek tugas Django kamu telah berjalan dengan baik.

Saya memastikannya dengan memperbaiki http menjadi https dan redeploy melalui github actions.

  • Membuat halaman login pada proyek tugas Flutter.
import 'package:adventurers_inventory/screens/menu.dart';
import 'package:flutter/material.dart';
import 'package:pbp_django_auth/pbp_django_auth.dart';
import 'package:provider/provider.dart';

void main() {
    runApp(const LoginApp());
}

class LoginApp extends StatelessWidget {
const LoginApp({super.key});

@override
Widget build(BuildContext context) {
    return MaterialApp(
        title: 'Login',
        theme: ThemeData(
            primarySwatch: Colors.blue,
    ),
    home: const LoginPage(),
    );
    }
}

class LoginPage extends StatefulWidget {
    const LoginPage({super.key});

    @override
    _LoginPageState createState() => _LoginPageState();
}

class _LoginPageState extends State<LoginPage> {
    final TextEditingController _usernameController = TextEditingController();
    final TextEditingController _passwordController = TextEditingController();

    @override
    Widget build(BuildContext context) {
        final request = context.watch<CookieRequest>();
        return Scaffold(
            appBar: AppBar(
                title: const Text('Login'),
            ),
            body: Container(
                padding: const EdgeInsets.all(16.0),
                child: Column(
                    mainAxisAlignment: MainAxisAlignment.center,
                    children: [
                        TextField(
                            controller: _usernameController,
                            decoration: const InputDecoration(
                                labelText: 'Username',
                            ),
                        ),
                        const SizedBox(height: 12.0),
                        TextField(
                            controller: _passwordController,
                            decoration: const InputDecoration(
                                labelText: 'Password',
                            ),
                            obscureText: true,
                        ),
                        const SizedBox(height: 24.0),
                        ElevatedButton(
                            onPressed: () async {
                                String username = _usernameController.text;
                                String password = _passwordController.text;

                                // Cek kredensial
                                // Ganti URL dan jangan lupa tambahkan trailing slash (/) di akhir URL!
                                // Untuk menyambungkan Android emulator dengan Django pada localhost,
                                // gunakan URL http://10.0.2.2/
                                final response = await request.login("https://daffa-mohamad-tugas.pbp.cs.ui.ac.id/", {
                                'username': username,
                                'password': password,
                                });
                    
                                if (request.loggedIn) {
                                    String message = response['message'];
                                    String uname = response['username'];
                                    Navigator.pushReplacement(
                                        context,
                                        MaterialPageRoute(builder: (context) => MyHomePage()),
                                    );
                                    ScaffoldMessenger.of(context)
                                        ..hideCurrentSnackBar()
                                        ..showSnackBar(
                                            SnackBar(content: Text("$message Selamat datang, $uname.")));
                                    } else {
                                    showDialog(
                                        context: context,
                                        builder: (context) => AlertDialog(
                                            title: const Text('Login Gagal'),
                                            content:
                                                Text(response['message']),
                                            actions: [
                                                TextButton(
                                                    child: const Text('OK'),
                                                    onPressed: () {
                                                        Navigator.pop(context);
                                                    },
                                                ),
                                            ],
                                        ),
                                    );
                                }
                            },
                            child: const Text('Login'),
                        ),
                    ],
                ),
            ),
        );
    }
}
  • Mengintegrasikan sistem autentikasi Django dengan proyek tugas Flutter.
  1. Membuat django-app bernama authentication pada project Django yang telah dibuat sebelumnya.

  2. Menambahkan authentication ke INSTALLED_APPS pada main project settings.py aplikasi Django.

  3. Jalankan perintah pip install django-cors-headers untuk menginstal library yang dibutuhkan.

  4. Tambahkan properti-properti corsheaders pada main project settings.py aplikasi Django.

  5. Membuat autentikasi login pada views.py

from django.shortcuts import render
from django.contrib.auth import authenticate, login as auth_login
from django.http import JsonResponse
from django.views.decorators.csrf import csrf_exempt

@csrf_exempt
def login(request):
    username = request.POST['username']
    password = request.POST['password']
    user = authenticate(username=username, password=password)
    if user is not None:
        if user.is_active:
            auth_login(request, user)
            # Status login sukses.
            return JsonResponse({
                "username": user.username,
                "status": True,
                "message": "Login sukses!"
                # Tambahkan data lainnya jika ingin mengirim data ke Flutter.
            }, status=200)
        else:
            return JsonResponse({
                "status": False,
                "message": "Login gagal, akun dinonaktifkan."
            }, status=401)

    else:
        return JsonResponse({
            "status": False,
            "message": "Login gagal, periksa kembali email atau kata sandi."
        }, status=401)
  1. Lalu melakukan routing pada urls.py
from django.urls import path
from authentication.views import login

app_name = 'authentication'

urlpatterns = [
    path('login/', login, name='login'),
]
  • Membuat model kustom sesuai dengan proyek aplikasi Django.

  • Membuat halaman yang berisi daftar semua item yang terdapat pada endpoint JSON di Django yang telah kamu deploy.

  • Tampilkan name, amount, dan description dari masing-masing item pada halaman ini.

Tugas 8

Jelaskan perbedaan antara Navigator.push() dan Navigator.pushReplacement(), disertai dengan contoh mengenai penggunaan kedua metode tersebut yang tepat!

Navigator.push() dan Navigator.pushReplacement() adalah dua metode yang digunakan dalam Flutter untuk mengelola navigasi antar halaman (routing).

Navigator.push() Navigator.pushReplacement()
Digunakan untuk menambahkan halaman baru ke dalam tumpukan navigasi. Digunakan untuk mengganti halaman saat ini dengan halaman baru.
Mempush halaman baru ke atas halaman saat ini. Halaman saat ini dihapus dari tumpukan dan diganti dengan halaman baru.
Menambahkan halaman baru ke tumpukan, sehingga pengguna dapat kembali ke halaman sebelumnya. Berguna ketika Anda ingin mencegah pengguna untuk kembali ke halaman sebelumnya.

Contoh untuk Navigator.push()

// Contoh menggunakan Navigator.push()
FlatButton(
  onPressed: () {
    Navigator.push(
      context,
      MaterialPageRoute(builder: (context) => HalamanBaru()),
    );
  },
  child: Text('Buka Halaman Baru'),
)

Contoh untuk Navigator.pushReplacement()

// Contoh menggunakan Navigator.pushReplacement()
FlatButton(
  onPressed: () {
    Navigator.pushReplacement(
      context,
      MaterialPageRoute(builder: (context) => HalamanBaru()),
    );
  },
  child: Text('Ganti ke Halaman Baru'),
)
Jelaskan masing-masing layout widget pada Flutter dan konteks penggunaannya masing-masing!

1. Single-child Layout Widgets

Berlaku untuk widget yang memiliki satu child widget di dalamnya

  • Container: Digunakan untuk mengelompokkan dan mengatur widget lainnya.
Container(
  padding: EdgeInsets.all(16.0),
  margin: EdgeInsets.symmetric(vertical: 8.0),
  color: Colors.blue,
  child: Text('Isi Container'),
)
  • Center: Menempatkan satu child widget di tengah parent widget.
Center(
  child: Text('Tengah'),
)
  • Align: Menempatkan satu child widget dengan spesifikasi alignment.
Align(
  alignment: Alignment.bottomRight,
  child: Text('Di Bawah Kanan'),
)

2. Multi-child Layout Widgets

Widget yang dapat memiliki lebih dari satu child widget.

  • Row and Column: Menempatkan child widget secara horizontal (Row) atau vertikal (Column).
Row(
  children: <Widget>[
    Icon(Icons.star),
    Text('Bintang'),
  ],
)

Column(
  children: <Widget>[
    Text('Baris 1'),
    Text('Baris 2'),
  ],
)
  • Stack: Menumpuk child widget di atas satu sama lain.
Stack(
  children: <Widget>[
    Positioned(
      top: 10.0,
      left: 10.0,
      child: Text('Atas Kiri'),
    ),
    Positioned(
      bottom: 10.0,
      right: 10.0,
      child: Text('Bawah Kanan'),
    ),
  ],
)
  • ListView and GridView: Menampilkan daftar atau grid dari child widget.
ListView(
  children: <Widget>[
    ListTile(title: Text('Item 1')),
    ListTile(title: Text('Item 2')),
    // ...
  ],
)

GridView.builder(
  gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
    crossAxisCount: 2,
  ),
  itemBuilder: (context, index) => MyGridItem(index),
)

3. Silver Widgets

  • SilverAppBar: Membuat AppBar yang dapat di-scroll.
CustomScrollView(
  slivers: <Widget>[
    SliverAppBar(
      title: Text('AppBar yang Bisa Di-Scroll'),
      floating: true,
      pinned: true,
    ),
    // ... SliverList atau SliverGrid
  ],
)

*SilverList and SilverGrid: Menampilkan daftar atau grid dalam konteks ScrollView.

CustomScrollView(
  slivers: <Widget>[
    SliverList(
      delegate: SliverChildBuilderDelegate(
        (context, index) => ListTile(title: Text('Item $index')),
        childCount: 10,
      ),
    ),
    // ... SliverAppBar atau SliverGrid
  ],
)
Sebutkan apa saja elemen input pada form yang kamu pakai pada tugas kali ini dan jelaskan mengapa kamu menggunakan elemen input tersebut!
  1. TextFormField untuk Nama Produk:

Alasan Penggunaan: Digunakan untuk memasukkan teks, dalam hal ini, nama produk. TextFormField memberikan kemudahan untuk mengelola input teks dengan menyediakan berbagai fitur seperti validasi dan interaksi yang nyaman.

Padding(
  padding: const EdgeInsets.all(8.0),
  child: TextFormField(
    decoration: InputDecoration(
      hintText: "Nama Produk",
      labelText: "Nama Produk",
      border: OutlineInputBorder(
        borderRadius: BorderRadius.circular(5.0),
      ),
    ),
    onChanged: (String? value) {
      setState(() {
        _name = value!;
      });
    },
    validator: (String? value) {
      if (value == null || value.isEmpty) {
        return "Nama tidak boleh kosong!";
      }
      return null;
    },
  ),
),
  1. TextFormField untuk Harga:

Alasan Penggunaan: Digunakan lagi untuk memasukkan harga produk. TextFormField memungkinkan input numerik dan menyediakan validasi untuk memastikan bahwa yang dimasukkan adalah angka.

Padding(
  padding: const EdgeInsets.all(8.0),
  child: TextFormField(
    decoration: InputDecoration(
      hintText: "Harga",
      labelText: "Harga",
      border: OutlineInputBorder(
        borderRadius: BorderRadius.circular(5.0),
      ),
    ),
    onChanged: (String? value) {
      setState(() {
        _amount = int.parse(value!);
      });
    },
    validator: (String? value) {
      if (value == null || value.isEmpty) {
        return "Harga tidak boleh kosong!";
      }
      if (int.tryParse(value) == null) {
        return "Harga harus berupa angka!";
      }
      return null;
    },
  ),
),
  1. TextFormField untuk Deskripsi:

Alasan Penggunaan: Digunakan ketiga kalinya untuk memasukkan deskripsi produk. TextFormField memudahkan pengguna memasukkan teks dan menyediakan validasi sederhana untuk memastikan deskripsi tidak kosong.

Padding(
  padding: const EdgeInsets.all(8.0),
  child: TextFormField(
    decoration: InputDecoration(
      hintText: "Deskripsi",
      labelText: "Deskripsi",
      border: OutlineInputBorder(
        borderRadius: BorderRadius.circular(5.0),
      ),
    ),
    onChanged: (String? value) {
      setState(() {
        _description = value!;
      });
    },
    validator: (String? value) {
      if (value == null || value.isEmpty) {
        return "Deskripsi tidak boleh kosong!";
      }
      return null;
    },
  ),
),
  1. Elemen lain:
  • Decorations: TextFormField memberikan opsi dekorasi yang kaya, seperti hintText, labelText, dan border. Ini membantu meningkatkan pengalaman pengguna dengan memberikan petunjuk visual.

  • OnChanged dan State Management: Dengan menggunakan onChanged, aplikasi dapat merespons perubahan nilai di dalam TextFormField. Selain itu, kita menggunakan state management untuk menyimpan nilai dari setiap TextFormField sehingga dapat diakses saat tombol "Save" ditekan.

  • Validator: TextFormField memungkinkan penerapan validator untuk memastikan bahwa data yang dimasukkan sesuai dengan aturan yang diinginkan. Validator ini membantu untuk memberikan umpan balik kepada pengguna jika input tidak sesuai dengan yang diharapkan.

Bagaimana penerapan clean architecture pada aplikasi Flutter?

Penerapan Clean Architecture pada aplikasi Flutter melibatkan strukturisasi proyek untuk memisahkan lapisan bisnis, presentasi, dan infrastruktur. Berikut adalah langkah-langkah singkatnya:

  1. Entities:
  • Definisikan entitas atau model bisnis.
  • Pastikan entitas tidak bergantung pada lapisan lainnya.
  1. Use Cases (Interactors):
  • Implementasikan operasi bisnis dalam use cases.
  • Pastikan use cases bersifat independen dan dapat digunakan di berbagai konteks.
  1. Interface Adapters:
  • Terapkan interface adapters untuk mengonversi data antara lapisan bisnis dan presentasi.
  • Pisahkan presentasi dari logika bisnis dengan menggunakan presenter atau view model.
  1. Frameworks & Drivers:
  • Tempatkan kode infrastruktur dan kerangka kerja eksternal di lapisan ini.
  • Ketergantungan dari lapisan bisnis ke lapisan ini melalui interface.
  1. Modularisasi Kode:
  • Pisahkan proyek menjadi modul-modul sesuai dengan lapisan bisnis, data, dan presentasi.
  1. Dependency Injection:
  • Gunakan Dependency Injection untuk memberikan ketergantungan antara lapisan.
  • Ini memudahkan pengujian dan meningkatkan fleksibilitas dalam perubahan implementasi.
  1. Testing:
  • Gunakan unit testing untuk menguji logika bisnis dan use cases secara terpisah.
  • Gunakan widget testing untuk menguji UI dan integrasi komponen.

Contoh struktur proyek yang menerapkan Clean Architecture di Flutter mungkin memiliki folder seperti "core" untuk lapisan bisnis, "data" untuk infrastruktur dan akses data, dan "presentation" untuk tampilan dan antarmuka pengguna. Selain itu, konsisten dalam menerapkan prinsip SOLID dan memisahkan tanggung jawab setiap lapisan adalah kunci untuk mendapatkan manfaat maksimal dari Clean Architecture.

Jelaskan bagaimana cara kamu mengimplementasikan checklist di atas secara step-by-step! (bukan hanya sekadar mengikuti tutorial)
  • Membuat minimal satu halaman baru pada aplikasi, yaitu halaman formulir tambah item baru dengan ketentuan sebagai berikut:

Membuat sebuah page baru dengan class ShopFormPage yang mengextend StatefulWidget

import 'package:flutter/material.dart';
// TODO: Impor drawer yang sudah dibuat sebelumnya
import 'package:adventurers_inventory/widgets/left_drawer.dart';
import 'package:adventurers_inventory/widgets/model_item.dart';

class ShopFormPage extends StatefulWidget {
  const ShopFormPage({super.key});

  @override
  State<ShopFormPage> createState() => _ShopFormPageState();
}

class _ShopFormPageState extends State<ShopFormPage> {
  final _formKey = GlobalKey<FormState>();
  String _name = "";
  int _amount = 0;
  String _description = "";

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Center(
          child: Text(
            'Form Tambah Produk',
          ),
        ),
        backgroundColor: Colors.indigo,
        foregroundColor: Colors.white,
      ),
      // Tambahkan drawer yang sudah dibuat di sini
      drawer: const LeftDrawer(),
      body: Form(
        key: _formKey,
        child: SingleChildScrollView(
            child:
                Column(crossAxisAlignment: CrossAxisAlignment.start, children:[])),
      ),
    );
  }
}
  • Memakai minimal tiga elemen input, yaitu name, amount, description. Tambahkan elemen input sesuai dengan model pada aplikasi tugas Django yang telah kamu buat.

Pada bagian ini dibuat 3 child input dengan dicontain oleh Padding, 3 input memakai TextFormField untuk Nama, Amount, dan Description

[
Padding(
    padding: const EdgeInsets.all(8.0),
    child: TextFormField(
      decoration: InputDecoration(
        hintText: "Nama Produk",
        labelText: "Nama Produk",
        border: OutlineInputBorder(
          borderRadius: BorderRadius.circular(5.0),
        ),
      ),
      onChanged: (String? value) {
        setState(() {
          _name = value!;
        });
      },
      validator: (String? value) {
        if (value == null || value.isEmpty) {
          return "Nama tidak boleh kosong!";
        }
        return null;
      },
    ),
  ),
  Padding(
    padding: const EdgeInsets.all(8.0),
    child: TextFormField(
      decoration: InputDecoration(
        hintText: "Harga",
        labelText: "Harga",
        border: OutlineInputBorder(
          borderRadius: BorderRadius.circular(5.0),
        ),
      ),
      onChanged: (String? value) {
        setState(() {
          _amount = int.parse(value!);
        });
      },
      validator: (String? value) {
        if (value == null || value.isEmpty) {
          return "Harga tidak boleh kosong!";
        }
        if (int.tryParse(value) == null) {
          return "Harga harus berupa angka!";
        }
        return null;
      },
    ),
  ),
  Padding(
    padding: const EdgeInsets.all(8.0),
    child: TextFormField(
      decoration: InputDecoration(
        hintText: "Deskripsi",
        labelText: "Deskripsi",
        border: OutlineInputBorder(
          borderRadius: BorderRadius.circular(5.0),
        ),
      ),
      onChanged: (String? value) {
        setState(() {
          _description = value!;
        });
      },
      validator: (String? value) {
        if (value == null || value.isEmpty) {
          return "Deskripsi tidak boleh kosong!";
        }
        return null;
      },
    ),
  ),
]
  • Memiliki sebuah tombol Save.

Pada tombol Save, dibuat widget ElevatedButton yang dibungkus oleh widget Padding dan Align. Dan ketika dipencet akan memunculkan sebuah Popup bernama AlertDialog

Align(
  alignment: Alignment.bottomCenter,
  child: Padding(
    padding: const EdgeInsets.all(8.0),
    child: ElevatedButton(
      style: ButtonStyle(
        backgroundColor: MaterialStateProperty.all(Colors.indigo),
      ),
      onPressed: () {
        if (_formKey.currentState!.validate()) {
          listItem.add(Item(_name, _amount, _description));
          showDialog(
            context: context,
            builder: (context) {
              return AlertDialog(
                title: const Text('Produk berhasil tersimpan'),
                content: SingleChildScrollView(
                  child: Column(
                    crossAxisAlignment: CrossAxisAlignment.start,
                    children: [
                      Text('Nama: $_name'),
                      Text('Amount: $_amount'),
                      Text('Description: $_description')
                    ],
                  ),
                ),
                actions: [
                  TextButton(
                    child: const Text('OK'),
                    onPressed: () {
                      Navigator.pop(context);
                    },
                  ),
                ],
              );
            },
          );
          _formKey.currentState!.reset();
        }
      },
      child: const Text(
        "Save",
        style: TextStyle(color: Colors.white),
      ),
    ),
  ),
),
  • Setiap elemen input di formulir juga harus divalidasi dengan ketentuan sebagai berikut:

  • Setiap elemen input tidak boleh kosong.

Validasi input diimplementasikan dengan validator seperti dibawah

validator: (String? value) {
  if (value == null || value.isEmpty) {
    return "Deskripsi tidak boleh kosong!";
  }
  return null;
},
  • Setiap elemen input harus berisi data dengan tipe data atribut modelnya.

Sama juga untuk validasi data dengan kode if (int.tryParse(value) == null)

validator: (String? value) {
  if (value == null || value.isEmpty) {
    return "Harga tidak boleh kosong!";
  }
  if (int.tryParse(value) == null) {
    return "Harga harus berupa angka!";
  }
  return null;
},
  • Mengarahkan pengguna ke halaman form tambah item baru ketika menekan tombol Tambah Item pada halaman utama.

Untuk perpindahan halaman digunakan Navigator untuk berpindah-pindah, pada halaman utama digunakan push untuk menyimpan pada stack agar nantinya bisa kembali ke halaman utama

onTap: () {
  // Memunculkan SnackBar ketika diklik
  ScaffoldMessenger.of(context)
    ..hideCurrentSnackBar()
    ..showSnackBar(SnackBar(
        content: Text("Kamu telah menekan tombol ${item.name}!"),
        backgroundColor: item.color,
      )
    );
  // Navigate ke route yang sesuai (tergantung jenis tombol)
  if (item.name == "Tambah Item") {
    // Gunakan Navigator.push untuk melakukan navigasi ke MaterialPageRoute yang mencakup ShopFormPage.
    Navigator.push(
          context,
          MaterialPageRoute(
            builder: (context) => ShopFormPage(),
          ));
  }
},
  • Memunculkan data sesuai isi dari formulir yang diisi dalam sebuah pop-up setelah menekan tombol Save pada halaman formulir tambah item baru.

Pada AlertDialog ini bekerja sebagai popup yang akan menunjukkan bahwa Produk sukses diinput beserta informasi2 yang ada.

return AlertDialog(
  title: const Text('Produk berhasil tersimpan'),
  content: SingleChildScrollView(
    child: Column(
      crossAxisAlignment: CrossAxisAlignment.start,
      children: [
        Text('Nama: $_name'),
        Text('Amount: $_amount'),
        Text('Description: $_description')
      ],
    ),
  ),
  actions: [
    TextButton(
      child: const Text('OK'),
      onPressed: () {
        Navigator.pop(context);
      },
    ),
  ],
);
  • Membuat sebuah drawer pada aplikasi dengan ketentuan sebagai berikut:
import 'package:adventurers_inventory/items_list.dart';
import 'package:flutter/material.dart';
import 'package:adventurers_inventory/menu.dart';
// Impor halaman ShopFormPage jika sudah dibuat
import 'package:adventurers_inventory/shoplist_form.dart';

class LeftDrawer extends StatelessWidget {
  const LeftDrawer({super.key});

  @override
  Widget build(BuildContext context) {
    return Drawer(
      child: ListView(
        children: [
          const DrawerHeader(
            decoration: BoxDecoration(
              color: Colors.indigo,
            ),
            child: Column(
              children: [
                Text(
                  'Shopping List',
                  textAlign: TextAlign.center,
                  style: TextStyle(
                    fontSize: 30,
                    fontWeight: FontWeight.bold,
                    color: Colors.white,
                  ),
                ),
                Padding(padding: EdgeInsets.all(10)),
                Text("Catat seluruh keperluan belanjamu di sini!",
                    //Tambahkan gaya teks dengan center alignment, font ukuran 15, warna putih, dan weight biasa
                    textAlign: TextAlign.center,
                    style: TextStyle(
                      fontSize: 15,
                      color: Colors.white,
                      fontWeight: FontWeight.normal,
                    ),
                    ),
              ],
            ),
          ),
          // Bagian routing
        ],
      ),
    );
  }
}
  • Drawer minimal memiliki dua buah opsi, yaitu Halaman Utama dan Tambah Item.

Untuk pilihan opsi pada drawer ini bekerja sebagai navigator pada drawer maka digunakan Navigator dan pushReplacement untuk langsung mengganti halaman tersebut

ListTile(
  leading: const Icon(Icons.home_outlined),
  title: const Text('Halaman Utama'),
  // Bagian redirection ke MyHomePage
  onTap: () {
    Navigator.pushReplacement(
        context,
        MaterialPageRoute(
          builder: (context) => MyHomePage(),
        ));
  },
),
ListTile(
  leading: const Icon(Icons.add_shopping_cart),
  title: const Text('Tambah Item'),
  // Bagian redirection ke ShopFormPage
  onTap: () {
    /*
    Buatlah routing ke ShopFormPage di sini,
    setelah halaman ShopFormPage sudah dibuat.
    */
    Navigator.pushReplacement(
        context,
        MaterialPageRoute(
          builder: (context) => ShopFormPage(),
        ));
  },
),
ListTile(
  leading: const Icon(Icons.checklist),
  title: const Text('Lihat Item'),
  // Bagian redirection ke ItemsListPage
  onTap: () {
    /*
    Buatlah routing ke ItemsListPage di sini,
    setelah halaman ItemsListPage sudah dibuat.
    */
    Navigator.pushReplacement(
        context,
        MaterialPageRoute(
          builder: (context) => ItemsListPage(),
        ));
  },
),
  • Ketika memiih opsi Halaman Utama, maka aplikasi akan mengarahkan pengguna ke halaman utama.

Diimplementasikann Navigator.pushReplacement untuk menggantikan halaman ke HomePage

onTap: () {
  Navigator.pushReplacement(
      context,
      MaterialPageRoute(
        builder: (context) => MyHomePage(),
      ));
},
  • Ketika memiih opsi (Tambah Item), maka aplikasi akan mengarahkan pengguna ke halaman form tambah item baru.

Diimplementasikan yang sama untuk perpindahan halaman ke input form dengan pushReplacement

onTap: () {
  /*
  Buatlah routing ke ShopFormPage di sini,
  setelah halaman ShopFormPage sudah dibuat.
  */
  Navigator.pushReplacement(
      context,
      MaterialPageRoute(
        builder: (context) => ShopFormPage(),
      ));
},

Tugas 7

Apa perbedaan utama antara stateless dan stateful widget dalam konteks pengembangan aplikasi Flutter?

state pada flutter adalah data atau variabel yang digunakan untuk membangun UI (User Interface) di flutter yang dapat berubah selama waktu aplikasi berjalan. state merujuk kepada data yang memengaruhi tampilan atau perilaku dari suatu widget. State adalah salah satu konsep utama dalam pembuatan aplikasi Flutter.

Terdapat dua jenis state:

Stateless Widget Stateful Widget
tidak memiliki data yang berubah selama masa hidupnya data yang dapat berubah selama masa hidupnya
dapat memanggil method build sekali ketika widget pertama kali dimuat dapat memanggil method build ulang ketika ada perubahan state dengan menggunakan method setState
tidak memerlukan class tambahan untuk mengatur state memerlukan class tambahan yang meng-extend State untuk mengatur state
menampilkan data yang sifatnya statis atau tidak perlu adanya perubahan nilai menampilkan data yang sifatnya dinamis atau membutuhkan perubahan nilai

Contoh kode Stateless Widget

class MyApp extends statelessWidget{
    @override
    Widget build(BuildContext context){
        return MaterialApp(
            ...
        )
    }
}

Contoh kode Stateful Widget

class MyHomePage extends statefulWidget {
    MyHomePage({Key key, this.title}) : super(key: key);
    final String title;

    @override
    _MyHomePageState createState() => _MyHomePageState();
}
Sebutkan seluruh widget yang kamu gunakan untuk menyelesaikan tugas ini dan jelaskan fungsinya masing-masing.
  • MaterialApp: Ini adalah widget yang digunakan untuk menginisialisasi aplikasi Flutter, dan di dalamnya, Anda dapat mengatur konfigurasi global seperti tema.

  • Scaffold: Ini adalah widget yang memberikan kerangka dasar untuk aplikasi Anda, termasuk AppBar, FloatingActionButton, dan lain-lain.

  • AppBar: Widget ini digunakan untuk menampilkan bilah atas (app bar) yang berisi judul aplikasi.

  • SingleChildScrollView: Widget ini digunakan untuk membuat area yang dapat di-scroll ketika kontennya lebih besar dari layar.

  • Padding: Widget ini digunakan untuk memberikan jarak (padding) di sekitar kontennya.

  • Column: Widget ini digunakan untuk menampilkan child widgets secara vertikal dalam kolom.

  • GridView.count: Widget ini digunakan untuk menampilkan item dalam grid dengan jumlah kolom yang telah ditentukan.

  • InkWell: Widget ini digunakan untuk membuat area responsif terhadap sentuhan (tap) dengan efek splash.

  • SnackBar: Widget ini digunakan untuk menampilkan pesan singkat (biasanya sebagai respons atas tindakan pengguna) di bagian bawah layar.

  • Material: Ini adalah widget yang mengelilingi konten card untuk memberikan efek Material Design.

  • Container: Widget ini digunakan untuk mengelilingi ikon dan teks dalam card.

  • Icon: Widget ini digunakan untuk menampilkan ikon.

  • Text: Widget ini digunakan untuk menampilkan teks.

Jelaskan bagaimana cara kamu mengimplementasikan checklist di atas secara step-by-step (bukan hanya sekadar mengikuti tutorial)
  • Membuat sebuah program Flutter baru dengan tema inventory seperti tugas-tugas sebelumnya.
  1. Membuat project Flutter baru dengan command flutter create adventurers_inventory dan dilanjutkan cd adventurers_inventory

  2. Inisiasi git dengan git init dan menyambungkan dengan repository remote di GitHub dengan git remote add origin https://github.com/fathonidf/adventurers-inventory-mobile.git

  3. Menjalankan project Flutter dengan flutter run

  • Membuat tiga tombol sederhana dengan ikon dan teks untuk:
  1. Pertama-tama, membuat main.dart sebagai inisiasi MyApp di awal dengan kode berikut
import 'package:flutter/material.dart';
import 'package:adventurers_inventory/menu.dart';

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        colorScheme: ColorScheme.fromSeed(seedColor: Colors.indigo),
        useMaterial3: true,
      ),
      home: MyHomePage(),
    );
  }
}

Pada kode di atas, menjadikan home dengan memanggil fungsi dari class MyHomePage yang berada pada menu.dart. Agar dapat diakses, maka file diimport dengan import 'package:adventurers_inventory/menu.dart';

  1. Membuat dan mendefinisikan suatu class untuk barang-barang yang akan dijual sebagai berikut
class ShopItem {
  final String name;
  final IconData icon;
  final Color color;

  ShopItem(this.name, this.icon, this.color);
}

attribute tambahan yaitu color untuk pengerjaan bonus pada Tugas 7 ini.

  1. Kemudian menambahkan class MyHomePage yang merupakan StatelessWidget kode pada menu.dart sebagai berikut
class MyHomePage extends StatelessWidget {
  MyHomePage({Key? key}) : super(key: key);
  final List<ShopItem> items = [
    ShopItem("Lihat Item", Icons.checklist, Colors.red),
    ShopItem("Tambah Item", Icons.add_shopping_cart, Colors.yellow.shade600),
    ShopItem("Logout", Icons.logout, Colors.green),
  ];

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Text(
          'Adventurer\'s Inventory ',
        ),
      ),
      body: SingleChildScrollView(
        // Widget wrapper yang dapat discroll
        child: Padding(
          padding: const EdgeInsets.all(10.0), // Set padding dari halaman
          child: Column(
            // Widget untuk menampilkan children secara vertikal
            children: <Widget>[
              const Padding(
                padding: EdgeInsets.only(top: 10.0, bottom: 10.0),
                // Widget Text untuk menampilkan tulisan dengan alignment center dan style yang sesuai
                child: Text(
                  'Inventory Store', // Text yang menandakan toko
                  textAlign: TextAlign.center,
                  style: TextStyle(
                    fontSize: 30,
                    fontWeight: FontWeight.bold,
                  ),
                ),
              ),
              // Grid layout
              GridView.count(
                // Container pada card kita.
                primary: true,
                padding: const EdgeInsets.all(20),
                crossAxisSpacing: 10,
                mainAxisSpacing: 10,
                crossAxisCount: 3,
                shrinkWrap: true,
                children: items.map((ShopItem item) {
                  // Iterasi untuk setiap item
                  return ShopCard(item);
                }).toList(),
              ),
            ],
          ),
        ),
      ),
    );
  }
}

Kode tersebut membuat suatu home page container dengan menampilkan Widget button-button dengan judul "Inventory Store". Pada final List<ShopItem> items menginstansiasi object baru dengan menambahkannya pada list items meliputi "Lihat Item", "Tambah Produk", dan "Logout".

Penambahan instansiasi attribute color untuk bonus tugas 7

  1. Selanjutnya membuat widget stateless baru untuk menampilkan ShopCard tersebut
class ShopCard extends StatelessWidget {
  final ShopItem item;

  const ShopCard(this.item, {super.key}); // Constructor

  @override
  Widget build(BuildContext context) {
    return Material(
      child: InkWell(
        // Area responsive terhadap sentuhan
        onTap: () {
          // Memunculkan SnackBar ketika diklik
          ScaffoldMessenger.of(context)
            ..hideCurrentSnackBar()
            ..showSnackBar(SnackBar(
                content: Text("Kamu telah menekan tombol ${item.name}!"),
                backgroundColor: item.color,
              )
            );
        },
        child: Container(
          // Container untuk menyimpan Icon dan Text
          color: item.color,
          padding: const EdgeInsets.all(8),
          child: Center(
            child: Column(
              mainAxisAlignment: MainAxisAlignment.center,
              children: [
                Icon(
                  item.icon,
                  color: Colors.white,
                  size: 30.0,
                ),
                const Padding(padding: EdgeInsets.all(3)),
                Text(
                  item.name,
                  textAlign: TextAlign.center,
                  style: const TextStyle(color: Colors.white),
                ),
              ],
            ),
          ),
        ),
      ),
    );
  }
}

Card tersebut akan ditampilkan dengan iterasi oleh MyHomePage.

  • Melihat daftar item Lihat Item
    final List<ShopItem> items = [
    ShopItem("Lihat Item", Icons.checklist, Colors.red),
];
  • Menambah item Tambah Item
    final List<ShopItem> items = [
    ShopItem("Tambah Item", Icons.add_shopping_cart, Colors.yellow.shade600),
];
  • Logout Logout
    final List<ShopItem> items = [
    ShopItem("Logout", Icons.logout, Colors.green),
];
  • Memunculkan Snackbar dengan tulisan:

    • "Kamu telah menekan tombol Lihat Item" ketika tombol Lihat Item ditekan.
    • "Kamu telah menekan tombol Tambah Item" ketika tombol Tambah Item ditekan.
    • "Kamu telah menekan tombol Logout" ketika tombol Logout ditekan.

    Hal tersebut dilakukan dengan menjadikannya responsive melalui onTap: () dan menambahkan ScaffoldMessenger.of(context)

    onTap: () {
          // Memunculkan SnackBar ketika diklik
          ScaffoldMessenger.of(context)
            ..hideCurrentSnackBar()
            ..showSnackBar(SnackBar(
                content: Text("Kamu telah menekan tombol ${item.name}!"),
                backgroundColor: item.color,
              )
            );
        },

    Karena setiap Card dimunculkan secara iterasi dari MyHomePage, maka tiap Card dapat mengakses item yang diinisiasi oleh constructor sehingga mengkonkatenasinya dengan ${item.name}

  • Melakukan add-commit-push ke GitHub.

git add .

git commit -m "Selesai Tugas 7"

git push -u origin main