Universitas Kebudayaan Digital Makassar

Universitas Kebudayaan Digital Makassar (UKDMKS) memadukan inovasi teknologi digital dengan pelestarian budaya lokal. Bergabunglah untuk membangun masa depan berbasis budaya dan teknologi!

android studio

Pengenalan Activity Untuk Aplikasi Android


Activity


Activity  merupakan salah satu komponen penting Android yang berfungsi untuk menampilkan user interface ke layar pengguna. Ini seperti pada saat Anda melihat daftar percakapan pada aplikasi chat atau daftar email pada aplikasi Gmail di ponsel Android Anda. Di dalamnya Anda dapat berinteraksi dengan aplikasi Anda, baik dengan menekan tombol atau menampilkan list.



Seperti ketika Anda membuat project baru di Android Studio, biasanya akan ada dua berkas yang sudah tercipta, yaitu MainActivity dan activity_main.xml.

MainActivity ini disebut sebagai class Activity karena mewarisi (extendssuperclass Activity. Tugasnya yaitu menampilkan layout activity_main.xml dan mengelola interaksi yang ada di dalamnya.

Umumnya dalam sebuah aplikasi terdapat lebih dari satu activity yang saling terhubung dengan tugas yang berbeda-beda. Yang perlu diperhatikan yaitu setiap Activity harus terdaftar di AndroidManifest.xml.  
Secara default, ia akan didaftarkan jika Anda membuat Activity baru dengan cara otomatis. Caranya yaitu klik kanan pada nama package → New →  Activity →  pilih template Activity yang tersedia.

Untuk lebih mendalami activity, kami menyarankan Anda untuk membaca referensi berikut:

Activity Lifecycle

Activity  memiliki daur hidup (lifecycle) dalam sebuah stack pada virtual sandbox yang disiapkan oleh Dalvik Virtual Machine (DVM) atau Android Runtime (ART) yang bersifat last in first out.
201609211453443ec948a46aa87264bbf9fb0e3c
Developer yang baik harus mengetahui secara detail tentang life cycle sebuah activity. Terutama untuk melakukan aksi yang tepat, saat terjadi perubahan state activity. 
Callback methods yang ada dapat digunakan untuk melakukan beragam proses terkait state dari activity. 
Misalnya melakukan semua inisialisasi komponen di onCreate(), melakukan disconnect terhadap koneksi ke server pada onStop() atau onDestroy() dan lain sebagainya.

Pemahaman yang baik tentang daur hidup activity akan membuat implementasi rancangan aplikasi Anda menjadi lebih baik. Hal ini juga akan meminimalisir terjadinya error/bug/force close yang tidak diinginkan.

Last In, First Out (LIFO)

20170515054021b0fdcc4af411dd804660ac803ea3ae2c.PNG

Gambar 1
Gambar 2
Gambar 3
Aktif: Activity 1
onCreate() → onStart() → onResume()
Aktif: Activity 2
Stack append: Activity 2 [ onResume() ]
Activity 1
onStop() → onRestart() → onStart() → onResume()
Aksi: Klik Button1 (Pindah)
Aksi: Klik Hardware Back Button
Aktif: Activity 1
Stack append: Activity 1 [ onStop() ]
Activity 2 [ finish() ]
Stack pop: Activity 2 [ onDestroy() ]

  1. Gambar 1
    Jika Anda memiliki sebuah aplikasi yang terdiri dari 2 activity, maka activity pertama akan dijalankan setelah pengguna meluncurkan aplikasi melalui ikon aplikasi di layar device. Activity yang ada saat ini berada pada posisi activity running setelah melalui beberapa state onCreate (created) → onStart (started) → onResume (resumed) dan masuk ke dalam sebuah stack activity.
    Bila pada activity pertama Anda menekan sebuah tombol untuk menjalankan activity kedua, maka posisi state dari activity pertama berada pada posisi stop. Saat itu, callback onStop() pada activity pertama akan dipanggil.
    Ini terjadi karena activity pertama sudah tidak berada pada layar foreground / tidak lagi ditampilkan. Semua informasi terakhir pada activity pertama akan disimpan secara otomatis.
    Sementara itu, activity kedua masuk ke dalam stack dan menjadi activity terakhir yang masuk.
  2. Gambar 2
    Activity kedua sudah muncul di layar sekarang. Ketika Anda menekan tombol back pada physical button menu utama atau menjalankan metode finish(), maka activity kedua Anda akan dikeluarkan dari stack.
    Pada kondisi di atas, state activity kedua akan berada pada destroy. Oleh karenanya, metode onDestroy() akan dipanggil.
    Kejadian keluar dan masuk stack pada proses di atas menandakan sebuah model Last In, First Out. Activity kedua menjadi yang terakhir masuk stack (Last In) dan yang paling pertama keluar dari stack (First Out).
  3. Gambar 3Activity Pertama akan dimunculkan kembali di layar setelah melalui beberapa state dengan rangkaian callback method yang terpanggil, onStop → onRestart → onStart → onResume.

Detailnya dapat Anda baca di sini:

Saving Activity State

Ketika sebuah activity mengalami pause kemudian resume, maka state dari sebuah activity tersebut dapat terjaga. Sebabnya, obyek activity masih tersimpan di memory sehingga dapat dikembalikan state-nya.
Dengan menjaga state dari activity, maka ketika activity tersebut ditampilkan, kondisinya akan tetap sama dengan kondisi sebelumnya.
Akan tetapi ketika sistem menghancurkan activity untuk keperluan memori misalnya karena memori habis, maka obyek activity dihancurkan. Alhasil, ketika activity ingin ditampilkan kembali diperlukan proses re-create activity yang dihancurkan tadi.
Kejadian di atas adalah hal yang lumrah terjadi. Oleh karena itu, perubahan yang terjadi pada activity perlu disimpan terlebih dahulu sebelum ia dihancurkan. Di sinilah metode onSaveInstanceState() digunakan.
Dalam onSaveInstanceState terdapat bundle yang dapat digunakan untuk menyimpan informasi. Informasi dapat disimpan dengan memanfaatkan fungsi seperti putString() dan putInt().
Ketika activity di-restart, bundle akan diberikan kepada metode onCreate dan onRestoreInstanceState. Bundle tersebut akan dimanfaatkan untuk mengembalikan kembali perubahan yang telah terjadi sebelumnya.
20161102140031b539b0f6990d374fe601740a2173b41e.png
Proses penghancuran activity dapat juga terjadi ketika terdapat perubahan konfigurasi seperti perubahan orientasi layar (portrait-landscape), keyboard availability, dan perubahan bahasa. 
Penghancuran ini akan menjalankan callback method onDestroy dan kemudian menjalankan onCreate. Penghancuran ini dimaksudkan agar activity dapat menyesuaikan diri dengan konfigurasi baru yang muncul pada kejadian-kejadian sebelumnya.
Hal yang perlu diingat ketika menggunakan onSaveInstanceState adalah untuk tidak menyimpan data yang besar pada bundle. Contohnya, hindari penyimpanan data bitmap pada bundle. 
Bila data pada bundle berukuran besar, proses serialisasi dan deserialisasi akan memakan banyak memori.

Tujuan

Codelab ini bertujuan untuk mengimplementasikan komponen activity pada aplikasi pertama yang Anda bangun. Harapannya codelab ini dapat memberi gambaran yang jelas tentang cara kerja activity.
Codelab pertama adalah dengan membuat aplikasi yang dapat menghitung volume balok. Seperti ini tampilannya:201811121622411d5143763cc8715c535d1327e772d305

Logika Dasar

Melakukan input ke dalam objek EditText → melakukan validasi input → menghitung volume balok ketika tombol hitung diklik.

Codelab Membuat Proyek Baru

  1. Buat proyek baru dengan klik File → New → New Project pada Android Studio Anda atau Anda bisa memilih Start a new Android Studio project di bagian dashboard.2018111209000871ac40d0c865aa819a08c988ddde4b4d

  2. Pada bagian ini kita akan memilih tipe activity awal dari template yang telah disediakan. Saat ini Android Studio sudah menyediakan berbagai macam template activity dari yang paling sederhana hingga yang paling kompleks seperti:

    Jenis-jenis template Activity Fungsinya
    Add No Activity Tidak ada activity yang ditambahkan
    Basic Activity Activity dengan template komponen material design seperti FloatingActionButton
    Bottom Navigation Activity Activity dengan tampilan side bar menu di bagian bawah
    Empty Activity Activity dalam bentuk yang paling dasar
    Fragment + ViewModel Activity dengan menerapkan architecture component
    Fullscreen Activity  Activity fullscreen tanpa status bar
    Google AdMob Ads Activity Activity dengan konfigurasi default iklan Admob
    Google Maps Activity Activity dengan menyediakan konfigurasi dasar Google Maps
    Login Activity Activity untuk halaman login
    Master / Detail Flow Activity yang diperuntukan untuk alur aplikasi master detail pada peranti tablet
    Navigation Drawer Activity Activity dengan tampilan side bar menu
    Scrolling Activity Activity dengan kemampuan scroll konten didalamnya secara vertikal
    Settings Activity Activity yang diperuntukan untuk konfigurasi aplikasi
    Tabbed Activity Activity yang diperuntukan untuk menampilkan lebih dari satu tampilan, dapat digeser ke kanan dan ke kiri (swipe) dengan menggunakan komponen ViewPager

    Selain itu, Anda juga bisa memilih target device mana yang akan Anda buat seperti Phone and Tablet, Wear OS, TV, Android Auto atau Android Things.
    2019011614581831311ab983af02c0dc716ffb6a567043Saat ini kita pilih tipe Empty Activity, klik Next untuk melanjutkan.

  3. Selanjutnya masukkan nama aplikasi dan nama package aplikasi Anda. Sebaiknya jangan sama dengan apa yang ada di contoh, karena ini berfungsi sebagai id dari aplikasi yang Anda buat. Kemudian Anda bisa menentukan lokasi proyek yang akan Anda buat. Setelah itu pilih tipe gawai/peranti (device) untuk aplikasi beserta target minimum SDK yang akan digunakan. Pilihan target Android SDK akan mempengaruhi banyaknya peranti yang dapat menggunakan aplikasi. Di sini kita memilih nilai minimum SDK kita pasang ke Level 21 (Lollipop). Klik Finish untuk melanjutkan.20190903095640912aeff2aaa8524a5e7798991ac43491

    Catatan: Untuk Android Studio versi 3.4.1 ke bawah, penggunaan AndroidX belum menjadi default. Jadi, silakan beri tanda centang pada Use AndroidX artifacts di setiap kali membuat project. Informasi lebih detail mengenai AndroidX bisa Anda lihat di sini, dan untuk melihat perubahan antara sebelum dan sesudah AndroidX atau cara migrasi menggunakan AndroidX bisa lihat di sini. Perlu Anda ketahui juga, ketika Anda sudah menggunakan AndroidX maka Anda tidak bisa menambahkan library lama atau sebelum AndroidX.

    Kelas ini menggunakan bahasa Kotlin sebagai bahasa utama. Namun jika anda ingin menggunakan bahasa Java, ubahlah languages dari Kotlin menjadi Java. Jangan khawatir, di dalam tutorial ini kalian bisa melihat kode dalam bahasa Kotlin dan Java secara berdampingan.
  4. Tampilan layar Anda akan seperti contoh di bawah ini:
    2019090310122922b067bbd1587f1c15215d8fa5feb6f9
  5. Di sebelah kanan Anda adalah workspace di mana Activity anda berada dan bernama MainActivity dengan layout-nya activity_main.xml. Di sebelah kiri Anda terdapat struktur proyek, di mana nanti kita akan banyak menambahkan berbagai komponen baru, asset, dan library. Untuk lebih mengenal Android Studio lebih dalam silakan baca materi di sini

Selanjutnya kita akan mulai melakukan pengkodean aplikasi atau lebih enaknya disebut ngoding.
Berikut flow umumnya:
  1. Ngoding Layout untuk user interface aplikasi
  2. Ngoding Activity untuk menambahkan logika aplikasi

Untuk mengoptimalkan proses pengetikan, Anda dapat memanfaatkan code completion dengan menekan ctrl + space. Android Studio juga akan mengimpor package dari komponen yang digunakan. 
Dilarang Keras untuk copy – paste! 
Ngoding pelan-pelan akan membuat Anda lebih jago di masa depan.
Selamat ngoding!

Codelab Layouting

Menambahkan Code Sederhana pada Layout Activity

  1. Silakan pilih tab berkas activity_main.xml pada workspace Anda(res/layout/activity_main.xml).
    201901161518426978d7756dd739230c63bb08f97581e6Pastikan project window pada pilihan Android, seperti di bawah ini:
    20190116151903ed13f27a41c64cb3b1abfc270f21e22cMaka akan ada tampilan seperti ini, kemudian pilih tab Code di sebelah pojok kanan atas.
    20200504155040c3dba1236b44fc8e13e48c9c5980717e.pngDan tambahkan baris-baris berikut:
  1. <?xml version=“1.0” encoding=“utf-8”?>
  2. <LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”
  3.     xmlns:tools=“http://schemas.android.com/tools”
  4.     android:layout_width=“match_parent”
  5.     android:layout_height=“match_parent”
  6.     android:padding=“16dp”
  7.     android:orientation=“vertical”>
  8.  
  9.     <TextView
  10.         android:layout_width=“match_parent”
  11.         android:layout_height=“wrap_content”
  12.         android:text=“Panjang” />
  13.     <EditText
  14.         android:id=“@+id/edt_length”
  15.         android:layout_width=“match_parent”
  16.         android:layout_height=“wrap_content”
  17.         android:inputType=“numberDecimal”
  18.         android:lines=“1” />
  19.     <TextView
  20.         android:layout_width=“match_parent”
  21.         android:layout_height=“wrap_content”
  22.         android:text=“Lebar” />
  23.     <EditText
  24.         android:id=“@+id/edt_width”
  25.         android:layout_width=“match_parent”
  26.         android:layout_height=“wrap_content”
  27.         android:inputType=“numberDecimal”
  28.         android:lines=“1” />
  29.     <TextView
  30.         android:layout_width=“match_parent”
  31.         android:layout_height=“wrap_content”
  32.         android:text=“Tinggi” />
  33.     <EditText
  34.         android:id=“@+id/edt_height”
  35.         android:layout_width=“match_parent”
  36.         android:layout_height=“wrap_content”
  37.         android:inputType=“numberDecimal”
  38.         android:lines=“1” />
  39.     <Button
  40.         android:id=“@+id/btn_calculate”
  41.         android:layout_width=“match_parent”
  42.         android:layout_height=“wrap_content”
  43.         android:text=“Hitung” />
  44.     <TextView
  45.         android:id=“@+id/tv_result”
  46.         android:layout_width=“match_parent”
  47.         android:layout_height=“wrap_content”
  48.         android:gravity=“center”
  49.         android:text=“Hasil”
  50.         android:textSize=“24sp”
  51.         android:textStyle=“bold” />
  52. </LinearLayout>




  1. Perlu diperhatikan root layout (tag layout terluar) yang dipakai di sini adalah LinearLayout. Jika kita menggunakan Android Studio versi 3 ke atas maka secara default root yang dipakai adalah ConstraintLayout. Agar sesuai dengan latihan ini, kita tinggal menggantinya menjadi LinearLayout. Untuk materi tentang Layout akan dibahas nanti pada modul yang berbeda.
  2. Kemudian akan muncul warning pada atribut android:text pada layout tersebut.
    20181112092821af9c83a6062f6880c683b41a9e6d2fd1Ini karena kita melakukan hardcoding pada nilai string-nya. Mari kita hilangkan code warning tersebut dengan menekan Alt+Enter (option + return pada Mac) atau menekan lampu kuning yang muncul pada attribute android:text.
    Akan muncul dialog seperti ini, pilih extract string resource.
    201811120923086c6e448fde7b10eeac1a7ef4fbf25924
  3. Kemudian akan muncul dialog seperti di bawah ini. Sesuaikan dengan nama yang ada.
    201811120927217800750fc9fc17abef37ef35bb3ccb31
  4. Fungsi extract string resource akan secara otomatis menambahkan nilai dari android:text ke dalam berkas res → values → strings.xml.
    2018111209261415bb8ae12764e29af67533cb284e42ccLakukan hal yang sama pada komponen view lainnya hingga tidak ada warning lagi. Jika kita buka berkas strings.xml, maka isinya akan menjadi seperti ini:


    1. <resources>

    2.     <string name="app_name">BarVolume</string>

    3.     <string name="width">Lebar</string>

    4.     <string name="height">Tinggi</string>

    5.     <string name="calculate">Hitung</string>

    6.     <string name="result">Hasil</string>

    7.     <string name="length">Panjang</string>

    8. </resources>


  5. Maka kode di dalam activity_main.xml akan menjadi seperti ini:
  1. <?xml version=“1.0” encoding=“utf-8”?>
  2. <LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”
  3.     xmlns:tools=“http://schemas.android.com/tools”
  4.     android:layout_width=“match_parent”
  5.     android:layout_height=“match_parent”
  6.     android:padding=“16dp”
  7.     android:orientation=“vertical”>
  8.  
  9.     <TextView
  10.         android:layout_width=“match_parent”
  11.         android:layout_height=“wrap_content”
  12.         android:text=“@string/length” />
  13.  
  14.     <EditText
  15.         android:id=“@+id/edt_length”
  16.         android:layout_width=“match_parent”
  17.         android:layout_height=“wrap_content”
  18.         android:inputType=“numberDecimal”
  19.         android:lines=“1” />
  20.  
  21.     <TextView
  22.         android:layout_width=“match_parent”
  23.         android:layout_height=“wrap_content”
  24.         android:text=“@string/width” />
  25.  
  26.     <EditText
  27.         android:id=“@+id/edt_width”
  28.         android:layout_width=“match_parent”
  29.         android:layout_height=“wrap_content”
  30.         android:inputType=“numberDecimal”
  31.         android:lines=“1” />
  32.  
  33.     <TextView
  34.         android:layout_width=“match_parent”
  35.         android:layout_height=“wrap_content”
  36.         android:text=“@string/height” />
  37.  
  38.     <EditText
  39.         android:id=“@+id/edt_height”
  40.         android:layout_width=“match_parent”
  41.         android:layout_height=“wrap_content”
  42.         android:inputType=“numberDecimal”
  43.         android:lines=“1” />
  44.  
  45.     <Button
  46.         android:id=“@+id/btn_calculate”
  47.         android:layout_width=“match_parent”
  48.         android:layout_height=“wrap_content”
  49.         android:text=“@string/calculate” />
  50.  
  51.     <TextView
  52.         android:id=“@+id/tv_result”
  53.         android:layout_width=“match_parent”
  54.         android:layout_height=“wrap_content”
  55.         android:gravity=“center”
  56.         android:text=“@string/result”
  57.         android:textSize=“24sp”
  58.         android:textStyle=“bold” />
  59. </LinearLayout>

  1. Jika Anda perhatikan, hasil layout sementara akan menjadi seperti ini:
    201901161526273401ae29ed3356f270242d25a26182ccSelain menggunakan Teks seperti di atas, Anda juga dapat membuat layout dengan menggunakan design. Untuk tutorialnya dapat Anda lihat divideo berikut:

Codelab Kode Logika

Menambahkan Kode Logika Sederhana pada MainActivity.

  1. Selanjutnya setelah selesai, lanjutkan dengan membuka berkas MainActivity dan lanjutkan ngoding baris-baris di bawah ini.
    Tambahkan beberapa variabel yang akan digunakan untuk menampung view.


  1. private lateinit var edtWidth: EditText

  2. private lateinit var edtHeight: EditText

  3. private lateinit var edtLength: EditText

  4. private lateinit var btnCalculate: Button

  5. private lateinit var tvResult: TextView





  1. private EditText edtWidth;

  2. private EditText edtHeight;

  3. private EditText edtLength;

  4. private Button btnCalculate;

  5. private TextView tvResult;





Kemudian inisiasi variabel yang telah kita buat dengan menambahkan kode berikut di dalam metode onCreate.



  1. override fun onCreate(savedInstanceState: Bundle?) {

  2.     super.onCreate(savedInstanceState)

  3.     setContentView(R.layout.activity_main)

  4.  

  5.     edtWidth = findViewById(R.id.edt_width)

  6.     edtHeight = findViewById(R.id.edt_height)

  7.     edtLength = findViewById(R.id.edt_length)

  8.     btnCalculate = findViewById(R.id.btn_calculate)

  9.     tvResult = findViewById(R.id.tv_result)

  10.  

  11.     btnCalculate.setOnClickListener(this)

  12. }





  1. @Override

  2. protected void onCreate(Bundle savedInstanceState) {

  3.     super.onCreate(savedInstanceState);

  4.     setContentView(R.layout.activity_main);

  5.  

  6.     edtWidth = findViewById(R.id.edt_width);

  7.     edtHeight = findViewById(R.id.edt_height);

  8.     edtLength = findViewById(R.id.edt_length);

  9.     btnCalculate = findViewById(R.id.btn_calculate);

  10.     tvResult = findViewById(R.id.tv_result);

  11.  

  12.     btnCalculate.setOnClickListener(this);

  13.  

  14. }



Akan muncul baris merah pada kata this. Hal ini karena kita belum menambahkan interface di kelas MainActivitySilakan klik di atas baris merah tersebut, kemudian tekan tombol Alt + Enter (option + return pada Mac) atau menekan lampu merah yang muncul lalu pilih aksi berikut untuk implement interface.


Maka secara otomatis akan ada penambahan kode pada kelas MainActivity seperti berikut ini:


  1. class MainActivity : AppCompatActivity(), View.OnClickListener {

  2.  

  3.     ...

  4.  

  5. }






  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  2.  

  3.     ...

  4.  

  5. }





Jika terdapat baris merah seperti ini:
20190903101751c14046f4bbbc7325d8451e5041e4f513

Jangan khawatir! Silakan klik di atas baris merah tersebut, kemudian tekan tombol
 Alt + Enter (option + return pada Mac) atau menekan lampu merah yang muncul lalu pilih implement members (Kotlin) atau implement methods (Java)
Jika terdapat baris merah seperti ini:




Maka secara otomatis akan ada penambahan metode onClick di kelas MainActivity. Setelah itu, tambahkan kode berikut ke dalam metode onClick:


  1. override fun onCreate(savedInstanceState: Bundle?) {

  2.     super.onCreate(savedInstanceState)

  3.     setContentView(R.layout.activity_main)

  4.  

  5.     ...

  6. }

  7.  

  8. override fun onClick(v: View) {

  9.     if (v.id == R.id.btn_calculate) {

  10.         val inputLength = edtLength.text.toString().trim()

  11.         val inputWidth = edtWidth.text.toString().trim()

  12.         val inputHeight = edtHeight.text.toString().trim()


  13.         val volume = inputLength.toDouble() * inputWidth.toDouble() * inputHeight.toDouble()

  14.         tvResult.text = volume.toString()

  15.     }

  16. }





  1. @Override

  2. protected void onCreate(Bundle savedInstanceState) {

  3.     super.onCreate(savedInstanceState);

  4.     setContentView(R.layout.activity_main);

  5.    

  6.     ...

  7.     

  8. }

  9.  

  10. @Override

  11. public void onClick(View v) {

  12.     if (v.getId() == R.id.btn_calculate) {

  13.         String inputLength = edtLength.getText().toString().trim();

  14.         String inputWidth = edtWidth.getText().toString().trim();

  15.         String inputHeight = edtHeight.getText().toString().trim();


  16.         double volume = Double.valueOf(inputLength) * Double.valueOf(inputWidth) * Double.valueOf(inputHeight);

  17.         tvResult.setText(String.valueOf(volume));

  18.     }

  19. }




Akhirnya kelas MainActivity akan memiliki kode seperti berikut ini:


  1. class MainActivity : AppCompatActivity(), View.OnClickListener {

  2.     private lateinit var edtWidth: EditText

  3.     private lateinit var edtHeight: EditText

  4.     private lateinit var edtLength: EditText

  5.     private lateinit var btnCalculate: Button

  6.     private lateinit var tvResult: TextView

  7.  

  8.     override fun onCreate(savedInstanceState: Bundle?) {

  9.         super.onCreate(savedInstanceState)

  10.         setContentView(R.layout.activity_main)

  11.  

  12.         edtWidth = findViewById(R.id.edt_width)

  13.         edtHeight = findViewById(R.id.edt_height)

  14.         edtLength = findViewById(R.id.edt_length)

  15.         btnCalculate = findViewById(R.id.btn_calculate)

  16.         tvResult = findViewById(R.id.tv_result)

  17.  

  18.         btnCalculate.setOnClickListener(this)

  19.     }

  20.  

  21.     override fun onClick(v: View) {

  22.         if (v.id == R.id.btn_calculate) {

  23.             val inputLength = edtLength.text.toString().trim()

  24.             val inputWidth = edtWidth.text.toString().trim()

  25.             val inputHeight = edtHeight.text.toString().trim()

  26.  

  27.             val volume = inputLength.toDouble() * inputWidth.toDouble() * inputHeight.toDouble()

  28.             tvResult.text = volume.toString()

  29.         }

  30.     }

  31. }










  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  2.     private EditText edtWidth, edtHeight, edtLength;

  3.     private Button btnCalculate;

  4.     private TextView tvResult;

  5.  

  6.     @Override

  7.     protected void onCreate(Bundle savedInstanceState) {

  8.         super.onCreate(savedInstanceState);

  9.         setContentView(R.layout.activity_main);

  10.  

  11.         edtWidth = findViewById(R.id.edt_width);

  12.         edtHeight = findViewById(R.id.edt_height);

  13.         edtLength = findViewById(R.id.edt_length);

  14.         btnCalculate = findViewById(R.id.btn_calculate);

  15.         tvResult = findViewById(R.id.tv_result);

  16.  

  17.         btnCalculate.setOnClickListener(this);

  18.     }

  19.  

  20.     @Override

  21.     public void onClick(View v) {

  22.         if (v.getId() == R.id.btn_calculate) {

  23.             String inputLength = edtLength.getText().toString().trim();

  24.             String inputWidth = edtWidth.getText().toString().trim();

  25.             String inputHeight = edtHeight.getText().toString().trim();

  26.  

  27.           double volume = Double.valueOf(inputLength) * Double.valueOf(inputWidth) * Double.valueOf(inputHeight);

  28.             tvResult.setText(String.valueOf(volume));      

  29.         }

  30.     }

  31. }








  1. telah selesai, silakan jalankan aplikasi dengan memilih menu Run → Run ‘app’ dari menu bar.
    201901170910324f3aff89c4b13f86e6f864536dc9d37aSelain cara di atas, Anda juga dapat menekan icon berikut di toolbar:
    202002171624414e0bf3ffd84f26229143bf960490eee7.pngKemudian akan muncul pilihan seperti ini:
    201901170917292fa4e95ece51ff9ab1f2b147f881f674Itu tandanya ADB (Android Debugger) pada peranti yang Anda punya telah terhubung dengan Android Studio. Jika Anda tidak memiliki peranti, maka Anda dapat menggunakan emulator. Ikuti materinya di modul sebelumnya atau lihat materi di sini.

    Kami merekomendasikan Anda menggunakan peranti Android sewaktu mengembangkan aplikasi. Selain karena beban memori pada peranti Anda akan jadi lebih rendah, pendekatan ini juga akan memungkinkan Anda untuk merasakan bagaimana aplikasi berjalan di device sebenarnya.
  2. Pilih OK untuk menjalankan dan tunggu hingga proses building dan instalasi APK selesai. Jika sudah, seharusnya hasilnya akan seperti ini:
201811121622041e168e77704b819eb68061c099ef81b2.gif


  1. Silakan masukkan nilai panjang, lebar, dan tinggi kemudian tekan tombol Hitung dan hasilnya akan ditampilkan di objek textview tvResultNamun masih ada sedikit masalah di sini, yaitu Anda tetap melakukan proses perhitungan walaupun salah satu nilainya kosong. Hal ini akan menyebabkan aplikasi force close karena perhitungan tidak dapat diproses. Maka untuk mengatasinya Anda akan menggunakan percabangan untuk mengecek apakah masing-masing EditText kosong atau tidak.
  2. Silakan buka kembali kelas MainActivity. Tambahkan kode berikut ke dalam metode onClicksebelum melakukan perhitungan.


  1. override fun onClick(v: View) {

  2.     if (v.id == R.id.btn_calculate) {

  3.         val inputLength = edtLength.text.toString().trim()

  4.         val inputWidth = edtWidth.text.toString().trim()

  5.         val inputHeight = edtHeight.text.toString().trim()


  6.         var isEmptyFields = false


  7.         if (inputLength.isEmpty()) {

  8.             isEmptyFields = true

  9.             edtLength.error = "Field ini tidak boleh kosong"

  10.         }


  11.         if (inputWidth.isEmpty()) {

  12.             isEmptyFields = true

  13.             edtWidth.error = "Field ini tidak boleh kosong"

  14.         }


  15.         if (inputHeight.isEmpty()) {

  16.             isEmptyFields = true

  17.             edtHeight.error = "Field ini tidak boleh kosong"

  18.         }


  19.         if (!isEmptyFields) {

  20.             val volume = inputLength.toDouble() * inputWidth.toDouble() * inputHeight.toDouble()

  21.             tvResult.text = volume.toString()

  22.         }

  23. }

  24. }







  1. @Override

  2. public void onClick(View v) {

  3.     if (v.getId() == R.id.btn_calculate) {

  4.         String inputLength = edtLength.getText().toString().trim();

  5.         String inputWidth = edtWidth.getText().toString().trim();

  6.         String inputHeight = edtHeight.getText().toString().trim();


  7.         boolean isEmptyFields = false;


  8.         if (TextUtils.isEmpty(inputLength)) {

  9.             isEmptyFields = true;

  10.             edtLength.setError("Field ini tidak boleh kosong");

  11.         }


  12.         if (TextUtils.isEmpty(inputWidth)) {

  13.             isEmptyFields = true;

  14.             edtWidth.setError("Field ini tidak boleh kosong");

  15.         }


  16.         if (TextUtils.isEmpty(inputHeight)) {

  17.             isEmptyFields = true;

  18.             edtHeight.setError("Field ini tidak boleh kosong");

  19.         }


  20.         if (!isEmptyFields) {

  21.             double volume = Double.valueOf(inputLength) * Double.valueOf(inputWidth) * Double.valueOf(inputHeight);

  22.             tvResult.setText(String.valueOf(volume));

  23.         }

  24.     }

  25. }




Akhirnya kelas MainActivity akan memiliki kode seperti berikut ini:



  1. class MainActivity : AppCompatActivity(), View.OnClickListener {

  2.     private lateinit var edtWidth: EditText

  3.     private lateinit var edtHeight: EditText

  4.     private lateinit var edtLength: EditText

  5.     private lateinit var btnCalculate: Button

  6.     private lateinit var tvResult: TextView

  7.  

  8.     override fun onCreate(savedInstanceState: Bundle?) {

  9.         super.onCreate(savedInstanceState)

  10.         setContentView(R.layout.activity_main)

  11.  

  12.         edtWidth = findViewById(R.id.edt_width)

  13.         edtHeight = findViewById(R.id.edt_height)

  14.         edtLength = findViewById(R.id.edt_length)

  15.         btnCalculate = findViewById(R.id.btn_calculate)

  16.         tvResult = findViewById(R.id.tv_result)

  17.  

  18.         btnCalculate.setOnClickListener(this)

  19.     }

  20.  

  21.     override fun onClick(v: View) {

  22.         if (v.id == R.id.btn_calculate) {

  23.             val inputLength = edtLength.text.toString().trim()

  24.             val inputWidth = edtWidth.text.toString().trim()

  25.             val inputHeight = edtHeight.text.toString().trim()

  26.  

  27.             var isEmptyFields = false

  28.  

  29.             if (inputLength.isEmpty()) {

  30.                 isEmptyFields = true

  31.                 edtLength.error = "Field ini tidak boleh kosong"

  32.             }

  33.  

  34.             if (inputWidth.isEmpty()) {

  35.                 isEmptyFields = true

  36.                 edtWidth.error = "Field ini tidak boleh kosong"

  37.             }

  38.  

  39.             if (inputHeight.isEmpty()) {

  40.                 isEmptyFields = true

  41.                 edtHeight.error = "Field ini tidak boleh kosong"

  42.             }

  43.  

  44.             if (!isEmptyFields) {

  45.                 val volume = inputLength.toDouble() * inputWidth.toDouble() * inputHeight.toDouble()

  46.                 tvResult.text = volume.toString()

  47.             }

  48.         }

  49.     }

  50. }



























  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  2.     private EditText edtWidth, edtHeight, edtLength;

  3.     private Button btnCalculate;

  4.     private TextView tvResult;

  5.  

  6.     @Override

  7.     protected void onCreate(Bundle savedInstanceState) {

  8.         super.onCreate(savedInstanceState);

  9.         setContentView(R.layout.activity_main);

  10.  

  11.         edtWidth = findViewById(R.id.edt_width);

  12.         edtHeight = findViewById(R.id.edt_height);

  13.         edtLength = findViewById(R.id.edt_length);

  14.         btnCalculate = findViewById(R.id.btn_calculate);

  15.         tvResult = findViewById(R.id.tv_result);

  16.  

  17.         btnCalculate.setOnClickListener(this);

  18.     }

  19.  

  20.     @Override

  21.     public void onClick(View v) {

  22.         if (v.getId() == R.id.btn_calculate) {

  23.             String inputLength = edtLength.getText().toString().trim();

  24.             String inputWidth = edtWidth.getText().toString().trim();

  25.             String inputHeight = edtHeight.getText().toString().trim();

  26.  

  27.             boolean isEmptyFields = false;

  28.             boolean isInvalidDouble = false;

  29.  

  30.             if (TextUtils.isEmpty(inputLength)) {

  31.                 isEmptyFields = true;

  32.                 edtLength.setError("Field ini tidak boleh kosong");

  33.             }

  34.  

  35.             if (TextUtils.isEmpty(inputWidth)) {

  36.                 isEmptyFields = true;

  37.                 edtWidth.setError("Field ini tidak boleh kosong");

  38.             }

  39.  

  40.             if (TextUtils.isEmpty(inputHeight)) {

  41.                 isEmptyFields = true;

  42.                 edtHeight.setError("Field ini tidak boleh kosong");

  43.             }

  44.  

  45.             if (!isEmptyFields) {

  46.                 double volume = Double.valueOf(inputLength) * Double.valueOf(inputWidth) * Double.valueOf(inputHeight);

  47.                 tvResult.setText(String.valueOf(volume));

  48.             }

  49.         }

  50.     }

  51. }



























  1. Jalan kembali aplikasi Anda dengan memilih menu Run → Run ‘app’  atau shortcut Shift + F10. Cobalah langsung menekan tombol HITUNG tanpa mengisi EditText, maka aplikasi Anda tidak akan force close dan akan muncul peringatan bahwa “Field ini tidak boleh kosong”.
    2020021716262850c766e9f11b302d8dc24994291d6ad8.png
  2. Apakah kita sudah selesai? Belum! Masih ada yang kurang. Ketika nilai volume sudah dihitung dan kemudian terjadi pergantian orientasi (portrait-landscape) pada peranti, maka hasil perhitungan tadi akan hilang. Hal ini karena di dalam Android, jika melakukan pergantian orientasi, Android akan memanggil fungsi onCreate kembali, sehingga data akan kembali menjadi seperti semula.
    2020022617264047085561e201ba68a797b5b4048ad0f7.pngUntuk mengatasinya, tambahkan metode onSaveInstanceState() pada MainActivity dan sesuaikan seperti berikut:



  1. companion object {

  2.     private const val STATE_RESULT = "state_result"

  3. }

  4.  

  5. ...

  6.  

  7. override fun onSaveInstanceState(outState: Bundle) {

  8.     super.onSaveInstanceState(outState)

  9.     outState.putString(STATE_RESULT, tvResult.text.toString())

  10. }






  1. private static final String STATE_RESULT = "state_result";

  2.  

  3. ...

  4.  

  5. @Override

  6. protected void onSaveInstanceState(Bundle outState) {

  7.     super.onSaveInstanceState(outState);

  8.     outState.putString(STATE_RESULT, tvResult.getText().toString());

  9. }




Kemudian tambahkan juga beberapa baris berikut pada baris terakhir metode onCreate


  1. override fun onCreate(savedInstanceState: Bundle?) {

  2.     super.onCreate(savedInstanceState)

  3.     setContentView(R.layout.activity_main)

  4.  

  5.     ...

  6.  

  7.     if (savedInstanceState != null) {

  8.         val result = savedInstanceState.getString(STATE_RESULT) as String

  9.         tvResult.text = result

  10.     }

  11. }






  1. @Override

  2. protected void onCreate(Bundle savedInstanceState) {

  3.     super.onCreate(savedInstanceState);

  4.     setContentView(R.layout.activity_main);

  5.  

  6.     ...

  7.  

  8.     if (savedInstanceState != null) {

  9.         String result = savedInstanceState.getString(STATE_RESULT);

  10.         tvResult.setText(result);

  11.     }

  12. }



  1. Silakan jalankan kembali aplikasinya. Ulangi proses perhitungan seperti sebelumnya. Kemudian ganti orientasi peranti Anda. Jika sudah benar maka hasil perhitungan tidak akan hilang. 

Bedah Kode

Pembahasan tentang layout xml

Layout merupakan user interface dari suatu activity. Layout dituliskan dalam format xml (extensible markup language).


  1. xml version="1.0" encoding="utf-8"?>


Baris ini mengidentifikasi bahwa berkas ini berformat xml.


  1. xmlns:android="http://schemas.android.com/apk/res/android"


Kode di atas menandakan namespace yang digunakan dalam keseluruhan berkas xml ini.

Macam Views

Di sini kita menggunakan beberapa komponen user interface yang disebut view. Di antaranya:
  • TextView, merupakan komponen view untuk menampilkan teks ke layar.
  • EditText, merupakan komponen view untuk memberikan input teks.
  • Button, merupakan komponen view untuk melakukan sebuah aksi klik.
  • LinearLayout, merupakan komponen view bertipe viewgroup yang menjadi parent dari semua sub komponen view (sub view) di dalamnya. Komponen ini bersifat sebagai kontainer untuk komponen lain dengan orientasi secara vertikal atau horizontal.
Cara membaca:


  1.  <TextView

  2.     android:layout_width="match_parent"

  3.     android:layout_height="wrap_content"

  4.     android:text="@string/calculate"

  5.     android:layout_marginBottom="16dp"/>


Komponen di atas adalah sebuah TextView. Perhatikan gambar di bawah ini. Warna ungu menandakan namespace yang digunakan; warna biru adalah atribut dari komponen dan warna hijau adalah nilai dari atribut. Penjelasannya seperti di bawah ini:
2016110214142154500af4d8d662f855314a704a
20161102141427e71b0d44f5baef3f1395940442
  • match_parent, merupakan sebuah ukuran dimensi sebuah View yang disesuaikan dengan ukuran layar baik secara horizontal pada layout_width dan vertikal jika pada layout_height.
  • wrap_content, merupakan ukuran dimensi sebuah View yang disesuaikan dengan ukuran konten di dalamnya baik secara horizontal pada layout_width dan vertikal jika pada layout_height.
  • @string/calculate, merupakan sebuah value calculate berasal dari berkas strings.xml
20180806171303692bcbb8c9bac3045c0da7b981d59067.
Penggunaan centralize resource value akan memudahkan Anda sewaktu mengembangkan aplikasi Android. Cara tersebut digunakan agar Anda tidak menulis nilai yang sama berulang-ulang. 

Apa itu @+id/ ? 

Salah satu contoh penerapan penggunaan @+id/ sebagai berikut:


  1. <Button

  2.     android:id="@+id/btn_calculate"

  3.     android:layout_width="match_parent"

  4.     android:layout_height="wrap_content"

  5.     android:text="@string/calculate"/>


Penjelasannya sebagai berikut:


  1. android:id="@+id/btn_calculate"


Jika kita memberikan id pada sebuah view maka kita telah memberikan identifier untuk view tersebut. Pemberian id ini dimaksudkan agar kita bisa melakukan manipulasi/pengendalian pada level logic di komponen seperti activity atau fragment.
Id di atas akan diciptakan di berkas R.java dan disimpan dalam bentuk hexa bertipe data integer, public static final int btn_calculate=0x7f0b0057.
2016110214161078fb3d968af6fd3f8cfc0dbfe0

Acuan untuk menyusun tampilan pada RelativeLayout akan dibahas pada modul selanjutnya.

Pembahasan tentang Logika Kode

Kode logika dituliskan ke dalam kelas Kotlin atau Java. Di sinilah semua aktifitas dari suatu aplikasi berjalan.
Activity


  1. class MainActivity : AppCompatActivity()




  1. public class MainActivity extends AppCompatActivity 




Menandakan bahwa kelas Java di atas merupakan sebuah activity karena inherit (turunan) dari superclass bernama AppCompatActivity.

OnClickListener


  1. , View.OnClickListener




  1. implements View.OnClickListener 




Ini adalah listener yang kita implementasikan untuk memantau kejadian klik pada komponen tombol (button).

Views


  1. private lateinit var edtWidth: EditText

  2. private lateinit var edtHeight: EditText

  3. private lateinit var edtLength: EditText

  4. private lateinit var btnCalculate: Button

  5. private lateinit var tvResult: TextView






  1. private EditText edtWidth;

  2. private EditText edtHeight;

  3. private EditText edtLength;

  4. private Button btnCalculate;

  5. private TextView tvResult;





Kode di atas mendeklarasikan semua komponen view yang akan dimanipulasi. Kita deklarasikan secara global agar bisa dikenal di keseluruhan bagian kelas.


OnCreate


  1. override fun onCreate(savedInstanceState: Bundle?) {

  2.     super.onCreate(savedInstanceState)

  3.     setContentView(R.layout.activity_main)

  4.  

  5.     edtWidth = findViewById(R.id.edt_width)

  6.     edtHeight = findViewById(R.id.edt_height)

  7.     edtLength = findViewById(R.id.edt_length)

  8.     btnCalculate = findViewById(R.id.btn_calculate)

  9.     tvResult = findViewById(R.id.tv_result)

  10.  

  11.     btnCalculate.setOnClickListener(this)

  12. }




  1. @Override

  2. protected void onCreate(Bundle savedInstanceState) {

  3.     super.onCreate(savedInstanceState);

  4.     setContentView(R.layout.activity_main);

  5.  

  6.     edtWidth = findViewById(R.id.edt_width);

  7.     edtHeight = findViewById(R.id.edt_height);

  8.     edtLength = findViewById(R.id.edt_length);

  9.     btnCalculate = findViewById(R.id.btn_calculate);

  10.     tvResult = findViewById(R.id.tv_result);

  11.  

  12.     btnCalculate.setOnClickListener(this);

  13. }



Metode onCreate() merupakan metode utama pada activity. Di sinilah kita dapat mengatur layout xml. Semua proses inisialisasi komponen yang digunakan akan dijalankan di sini. 
Inisialisasi komponen dengan findViewById berfungsi untuk menghubungkan variabel yang kita buat sebelumnya dengan id yang sudah kita buat di dalam layout activity_main.xml

SetContentView


  1. setContentView(R.layout.activity_main)




  1. setContentView(R.layout.activity_main);




Maksud baris di atas adalah kelas MainActivity akan menampilkan tampilan yang berasal dari layout activity_main.xml.

Casting View


  1. edtWidth = findViewById(R.id.edt_width)





  1. edtWidth = findViewById(R.id.edt_width);





Maksud dari baris diatas adalah objek edittext edtWidth disesuaikan (cast) dengan komponen edittext ber-id edt_width di layout xml melalui metode findViewById().

SetOnClickListener


  1. class MainActivity : AppCompatActivity(), View.OnClickListener {

  2. ...

  3. btnCalculate.setOnClickListener(this)

  4. ...

  5. override fun onClick(v: View) {

  6. ...

  7. }






  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  2. ...

  3. btnCalculate.setOnClickListener(this);

  4. ...

  5. @Override

  6. public void onClick(View v) {

  7. ...

  8. }







Kita memasang event click listener untuk objek btnCalculate sehingga sebuah aksi dapat dijalankan ketika objek tersebut diklik. 

Keyword this merujuk pada objek Activity saat ini yang telah mengimplementasikan listener OnClickListener sebelumnya.  Sehingga ketika btnCalculate diklik, maka fungsi onClick akan dipanggil dan melukakan proses yang ada di dalamnya.

Selain menggunakan implementation seperti di atas, Anda juga dapat mengimplementasikannya langsung seperti ini:


  1. btnCalculate.setOnClickListener {

  2. val inputLength = edtLength.text.toString().trim()

  3. val inputWidth = edtWidth.text.toString().trim()

  4. val inputHeight = edtHeight.text.toString().trim()

  5. ...

  6. }






  1. btnCalculate.setOnClickListener(new View.OnClickListener() {

  2. @Override

  3. public void onClick(View view) {

  4. String inputLength = edtLength.getText().toString().trim();

  5. String inputWidth = edtWidth.getText().toString().trim();

  6. String inputHeight = edtHeight.getText().toString().trim();

  7. ...

  8. }

  9. });



Dengan menggunakan cara ini Anda tak perlu membuat method onClick dan implement View.OnClickListener di awal kelas. Namun Anda langsung mengimplementasikannya di dalam setOnClickListener.
Mengambil value dari EditText


  1. val inputLength = edtLength.text.toString().trim()

  2. val inputWidth = edtWidth.text.toString().trim()

  3. val inputHeight = edtHeight.text.toString().trim()




  1. String inputLength = edtLength.getText().toString().trim();

  2. String inputWidth = edtWidth.getText().toString().trim();

  3. String inputHeight = edtHeight.getText().toString().trim();



Sintaks .text.toString() di atas berfungsi untuk mengambil isi dari sebuah edittext kemudian menyimpannya dalam sebuah variabel. Tambahan .trim() berfungsi untuk menghiraukan spasi jika ada, sehingga nilai yang didapat hanya berupa angka.

Validasi input yang kosong



  1. var isEmptyFields = false

  2. var isInvalidDouble = false

  3.  

  4. when {

  5.     inputLength.isEmpty() -> {

  6.         isEmptyFields = true

  7.         edtLength.error = "Field ini tidak boleh kosong"

  8.     }

  9.     inputWidth.isEmpty() -> {

  10.         isEmptyFields = true

  11.         edtWidth.error = "Field ini tidak boleh kosong"

  12.     }

  13.     inputHeight.isEmpty() -> {

  14.         isEmptyFields = true

  15.         edtHeight.error = "Field ini tidak boleh kosong"

  16.     }

  17. }





  1. boolean isEmptyFields = false;

  2. boolean isInvalidDouble = false;

  3.  

  4. if (TextUtils.isEmpty(inputLength)) {

  5.     isEmptyFields = true;

  6.     edtLength.setError("Field ini tidak boleh kosong");

  7. }

  8.  

  9. if (TextUtils.isEmpty(inputWidth)) {

  10.     isEmptyFields = true;

  11.     edtWidth.setError("Field ini tidak boleh kosong");

  12. }

  13.  

  14. if (TextUtils.isEmpty(inputHeight)) {

  15.     isEmptyFields = true;

  16.     edtHeight.setError("Field ini tidak boleh kosong");

  17. }




Sintaks .isEmpty() berfungsi untuk mengecek apakah inputan dari edittext itu masih kosong. Jika iya, maka kita akan menampilkan pesan error  dan mengganti variabel boolean isEmptyField menjadi true supaya bisa lanjut ke proses selanjutnya.

Menampilkan data ke EditText


  1. if (!isEmptyFields) {

  2.     val volume = inputLength.toDouble() * inputWidth.toDouble() * inputHeight.toDouble()

  3.     tvResult.text = volume.toString()

  4. }






  1. if (!isEmptyFields) {

  2.     double volume = Double.valueOf(inputLength) * Double.valueOf(inputWidth) * Double.valueOf(inputHeight);

  3.     tvResult.setText(String.valueOf(volume));

  4. }





Sintaks !isEmptyFields memiliki arti “jika semua inputan tidak kosong”. 
Jika kondisi tersebut terpenuhi maka langkah selanjutnya yaitu melakukan proses perhitungan, namun di sini kita tidak dapat langsung melakukan perhitungan. 
Hal ini karena secara default input di edittext tipe datanya berupa String, maka kita perlu untuk merubah tipe datanya terlebih dahulu menjadi double. 
Setelah selesai proses perhitungan, maka selanjutnya yaitu menampilkan hasil pada textview tvResult
Di sini dapat dilihat bahwa kita perlu merubah datanya yang sebelumnya double menjadi string lagi, hal ini karena untuk menampilkan ke dalam textview harus menggunakan tipe data string.

Pembahasan saveInstanceState 


  1. override fun onSaveInstanceState(outState: Bundle) {

  2.     super.onSaveInstanceState(outState)

  3.     outState.putString(STATE_RESULT, tvResult.text.toString())

  4. }





  1. @Override

  2. protected void onSaveInstanceState(Bundle outState) {

  3.     super.onSaveInstanceState(outState);

  4.     outState.putString(STATE_RESULT, tvResult.getText().toString());

  5. }



Perhatikan metode onSaveInstanceStateDi dalam metode tersebut, hasil perhitungan yang ditampilkan pada tvResult dimasukkan pada bundle kemudian disimpan isinya. 

Untuk menyimpan data di sini menggunakan konsep key-value, dengan STATE_RESULT sebagai key dan isi dari tvResult sebagai value

Fungsi onSaveInstanceState akan dipanggil secara otomatis sebelum sebuah Activity hancur. 

Di sini kita perlu menambahkan onSaveInstanceState karena ketika orientasi berubah, activity tersebut akan di-destroy dan memanggil fungsi onCreate lagi, sehingga kita perlu menyimpan nilai hasil perhitungan tersebut supaya data tetap terjaga dan tidak hilang ketika orientasi berubah.


  1. if (savedInstanceState != null) {

  2.     val result = savedInstanceState.getString(STATE_RESULT) as String

  3.     tvResult.text = result

  4. }





  1. if (savedInstanceState != null){

  2.     String result = savedInstanceState.getString(STATE_RESULT);

  3.     tvResult.setText(result);

  4. }




Pada onCreate inilah kita menggunakan nilai bundle yang telah kita simpan sebelumnya pada onSaveInstanceState. Nilai tersebut kita dapatkan dengan menggunakan key yang sama dengan saat menyimpan, yaitu STATE_RESULT. Kemudian kita isikan kembali pada tvResult.

Selamat! Anda telah berhasil membuat dan menjalankan latihan Android pertama di peranti atau emulator. Silakan lanjut ke codelab berikutnya.

Source code dapat Anda unduh di:


Pastikan untuk mengunduh Git terlebih dahulu. Cek tautan di bawah ini:
Asalas | Unlock Anime, Manhwa & Donghua Like Never BeforeFinasteriden: Unlock the Secrets of the Movie World!Marians Woman: Celebrating Beauty Around the WorldArheon - Unveiling the Secrets of Food & Origins WorldwideMPAPER The Ultimate Destination for Anime & Manga FansANMC21: Your Guide to Smarter Living and SuccessMV Agusta of Tampa: Your Automotive News HubTech and Play: Your Hub for Technology, Gaming, and GadgetsRebeccaSommer : Stories & Cultures from Around the WorldUETRABAJANDOJUNTOS - Inside the Music World’s Best-Kept SecretsVandelay Armor - Viral News and Global UpdatesGainesville Sun: Technology & Computers UnveiledGRFX Gaming Party Bus: Journey Through Gaming ErasHouse of Beauty: Celebrating the World's Most Beautiful WomenLearn Mistake: Wisdom for a Better LifeSports Hope Alive: Portal Olahraga DuniaWorld News TW - The Hottest Viral News & Global HeadlinesWriter Sujatha - Life Lessons & Struggles That Inspirehttps://128.199.185.186/https://143.198.89.74/https://165.227.47.178/https://170.64.208.214/https://170.64.192.239/https://46.101.102.216/LVONLINEtelegram lvonlinehttps://www.thecarbongenie.com/https://www.aievea-bijou.com/https://www.slashpolicy.com/https://www.benwestbeech.com/https://www.hh-bags.com/https://www.drupalforfacebook.org/https://www.lvonline.boats/https://www.lvoslot.com/https://www.lvobet.id/https://www.lvoslot.id/https://www.lvonline000.com/https://www.lvonline001.com/https://www.lvonline002.com/https://www.lvonline003.com/https://www.lvonline004.com/https://www.lvonline005.com/https://www.lvonline006.com/https://www.lvonline007.com/https://www.lvonline008.com/https://www.lvonline009.com/https://www.lvonline010.com/https://www.lvonlinepoker.com/https://www.lvonlinebola.com/https://www.lvonlinekasino.com/https://www.lvonline.io/https://www.lvonline.store/https://www.lvonline.online/https://www.situslvonline.us/situs bandar lvonlinehttps://146.190.97.83/https://143.198.209.134/https://188.166.246.204/https://167.172.83.179/https://togelhok.tv/https://www.togelhok.id/https://earthtoweb.com/https://www.elearningfacultymodules.org/https://www.how6youtoknowc.org/https://128.199.71.129/https://167.172.64.185/https://152.42.192.250/https://www.capcut88.com/https://www.capcut88.co/https://towsonsmiles.com/https://www.campur88.com/https://www.campur88.org/https://www.campur88.work/https://www.campur88.xyz/https://www.campur88.lol/https://www.nagacampur.biz/https://www.nagacampur.club/https://www.nagacampur.co/https://www.nagacampur.me/https://www.nagacampur.xyz/https://www.nasicampur88.com/https://165.232.175.185/https://152.42.164.228/https://152.42.194.203/https://152.42.169.214/https://www.campurslot.com/https://www.campurslot.id/https://www.campurslot.co/https://www.campurslot.org/https://www.campurslot.homes/https://www.campurslot.design/Badan Lembaga Pendidikan Ilmu Komputer SubangUniversitas Negeri JeparaLembaga Universitas Kristen MaranthaSMP 3 Negri Nganjukhttps://iklanmalay.com/https://promobola.comhttps://kapsychologists.comhttps://propertycloudsite.comSattar777 NewsVR Slot Online NewsRuby888 Online Slot NewsBerita Global Klik Agenslot228Agen Sloto SG777 NewsGold Club SlotOnline NewsAzar Marra Kech NewsBerita Sidney dan MancanegaraBerita Seputar Sumsel dan DuniaBerita Rehabilitasi Judi OnlineBerita Link Gacor TerupdateItalian Tuition News UpdateWPJS Online NewsBerita Agen Slot RoyalThai Slot Online NewsAll Online Game NewsOnline Game News UpdateAsian Lotre NewsBerita Demo Dana SlotBerita Kalbar ProvLocanda della Maria NewsWye Valley NewsBerita Agen Slot1004Berita Agen Slot33Agen Slot399 NewsPlayboy Slot Online NewsSlot Online BB Slot NewsSlot Online 911 News365 Slot Online NewsEat Atlah Newsambamali canadaInfo Seputar Sepakbolacentre thoughtBerita Hiburanopen etherpadras indo groupresistance manualPrediksi Shiowe want real newsthe poisoned pawnindonesia reclaimed teakswift kennedy and copullip passionmy passion foraim torontoPublic FlashesFriweb TeknologiIngenious Gamersthe late show gardensGishPuppy Newsslot danaOregon Farm Garden NewsViral Pulse GlobaljumpajpPromo Bola soccer Captivates The WorldKapsychologists World First The Science of Mental Health - Understanding Psychiatry: The Science of Mental HealthPropertyCloudSite How to Make Smarter Investments in Today’s MarketArnavichara Ultimate Guide Right Business SoftwareAuscare Disability A Comprehensive Guide to Retirement Homes Finding the Perfect Place to Enjoy Your Golden YearsSeries Mp4 The Future of Entertainment Streaming and Downloadable Video ExplainedAlogirlxinh How to Create a Successful Personal Page or Blog in 2024ihokibethttps://bengbulang-karangpucung.cilacapkab.go.id/https://comunicacion.unsa.edu.ar/https://seychellesbiodiversitychm.sc/https://www.925lms.com/https://www.guisseny.memoire.bzh/https://www.mobiliars.org/https://www.squashparkwieliczka.pl/
Asalas | Unlock Anime, Manhwa & Donghua Like Never BeforeFinasteriden: Unlock the Secrets of the Movie World!Marians Woman: Celebrating Beauty Around the WorldArheon - Unveiling the Secrets of Food & Origins WorldwideMPAPER The Ultimate Destination for Anime & Manga FansANMC21: Your Guide to Smarter Living and SuccessMV Agusta of Tampa: Your Automotive News HubTech and Play: Your Hub for Technology, Gaming, and GadgetsRebeccaSommer : Stories & Cultures from Around the WorldUETRABAJANDOJUNTOS - Inside the Music World’s Best-Kept SecretsVandelay Armor - Viral News and Global UpdatesGainesville Sun: Technology & Computers UnveiledGRFX Gaming Party Bus: Journey Through Gaming ErasHouse of Beauty: Celebrating the World's Most Beautiful WomenLearn Mistake: Wisdom for a Better LifeSports Hope Alive: Portal Olahraga DuniaWorld News TW - The Hottest Viral News & Global HeadlinesWriter Sujatha - Life Lessons & Struggles That Inspirehttps://128.199.185.186/https://143.198.89.74/https://165.227.47.178/https://170.64.208.214/https://170.64.192.239/https://46.101.102.216/LVONLINEtelegram lvonlinehttps://www.thecarbongenie.com/https://www.aievea-bijou.com/https://www.slashpolicy.com/https://www.benwestbeech.com/https://www.hh-bags.com/https://www.lvonline.boats/https://www.lvoslot.com/https://www.lvobet.id/https://www.lvoslot.id/https://www.lvonline000.com/https://www.lvonline001.com/https://www.lvonline002.com/https://www.lvonline003.com/https://www.lvonline004.com/https://www.lvonline005.com/https://www.lvonline006.com/https://www.lvonline007.com/https://www.lvonline008.com/https://www.lvonline009.com/https://www.lvonline010.com/https://www.lvonlinepoker.com/https://www.lvonlinebola.com/https://www.lvonlinekasino.com/https://www.lvonline.io/https://www.lvonline.store/https://www.lvonline.online/https://www.situslvonline.us/situs bandar lvonlinehttps://146.190.97.83/https://143.198.209.134/https://188.166.246.204/https://167.172.83.179/https://togelhok.tv/https://www.togelhok.id/https://earthtoweb.com/https://www.elearningfacultymodules.org/https://www.how6youtoknowc.org/https://128.199.71.129/https://167.172.64.185/https://152.42.192.250/https://www.capcut88.com/https://www.capcut88.co/https://towsonsmiles.com/https://www.campur88.com/https://www.campur88.org/https://www.campur88.work/https://www.campur88.xyz/https://www.campur88.lol/https://www.nagacampur.biz/https://www.nagacampur.club/https://www.nagacampur.co/https://www.nagacampur.me/https://www.nagacampur.xyz/https://www.nasicampur88.com/https://165.232.175.185/https://152.42.164.228/https://152.42.194.203/https://152.42.169.214/https://www.campurslot.com/https://www.campurslot.id/https://www.campurslot.co/https://www.campurslot.org/https://www.campurslot.homes/https://www.campurslot.design/Badan Lembaga Pendidikan Ilmu Komputer SubangUniversitas Negeri JeparaLembaga Universitas Kristen MaranthaSMP 3 Negri Nganjukhttps://iklanmalay.com/https://promobola.comhttps://kapsychologists.comhttps://propertycloudsite.comSattar777 NewsVR Slot Online NewsRuby888 Online Slot NewsBerita Global Klik Agenslot228Agen Sloto SG777 NewsGold Club SlotOnline NewsAzar Marra Kech NewsBerita Sidney dan MancanegaraBerita Seputar Sumsel dan DuniaBerita Rehabilitasi Judi OnlineBerita Link Gacor TerupdateItalian Tuition News UpdateWPJS Online NewsBerita Agen Slot RoyalThai Slot Online NewsAll Online Game NewsOnline Game News UpdateAsian Lotre NewsBerita Demo Dana SlotBerita Kalbar ProvLocanda della Maria NewsWye Valley NewsBerita Agen Slot1004Berita Agen Slot33Agen Slot399 NewsPlayboy Slot Online NewsSlot Online BB Slot NewsSlot Online 911 News365 Slot Online NewsEat Atlah Newsambamali canadaInfo Seputar Sepakbolacentre thoughtBerita Hiburanopen etherpadras indo groupresistance manualPrediksi Shiowe want real newsthe poisoned pawnindonesia reclaimed teakswift kennedy and copullip passionmy passion foraim torontoPublic FlashesFriweb TeknologiIngenious Gamersthe late show gardensGishPuppy Newsslot danaOregon Farm Garden NewsViral Pulse GlobaljumpajpPromo Bola soccer Captivates The WorldKapsychologists World First The Science of Mental Health - Understanding Psychiatry: The Science of Mental HealthPropertyCloudSite How to Make Smarter Investments in Today’s MarketArnavichara Ultimate Guide Right Business SoftwareAuscare Disability A Comprehensive Guide to Retirement Homes Finding the Perfect Place to Enjoy Your Golden YearsSeries Mp4 The Future of Entertainment Streaming and Downloadable Video ExplainedAlogirlxinh How to Create a Successful Personal Page or Blog in 2024ihokibethttps://bengbulang-karangpucung.cilacapkab.go.id/https://comunicacion.unsa.edu.ar/https://seychellesbiodiversitychm.sc/https://www.925lms.com/https://www.guisseny.memoire.bzh/https://www.mobiliars.org/https://www.squashparkwieliczka.pl/https://mok.edu.kz/https://www.mware.cloud/https://www.facilitiescoolingandheating.com.au/https://www.krillpay.ng/https://925worksuite.com/https://www.scenarioaulongcourt-archives.com/http://www.lesateliersdusoleil.fr/