commit 1fc3ccc5d48948d670e08f6e0210134610a0060b Author: ismarauu <202210715298@mhs.ubharajaya.ac.id> Date: Thu Dec 12 15:03:00 2024 +0700 first commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa724b7 --- /dev/null +++ b/.gitignore @@ -0,0 +1,15 @@ +*.iml +.gradle +/local.properties +/.idea/caches +/.idea/libraries +/.idea/modules.xml +/.idea/workspace.xml +/.idea/navEditor.xml +/.idea/assetWizardSettings.xml +.DS_Store +/build +/captures +.externalNativeBuild +.cxx +local.properties diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..26d3352 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,3 @@ +# Default ignored files +/shelf/ +/workspace.xml diff --git a/.idea/.name b/.idea/.name new file mode 100644 index 0000000..dd67ebb --- /dev/null +++ b/.idea/.name @@ -0,0 +1 @@ +Aplikasi Absensi \ No newline at end of file diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000..b589d56 --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/deploymentTargetDropDown.xml b/.idea/deploymentTargetDropDown.xml new file mode 100644 index 0000000..f3bf363 --- /dev/null +++ b/.idea/deploymentTargetDropDown.xml @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/deploymentTargetSelector.xml b/.idea/deploymentTargetSelector.xml new file mode 100644 index 0000000..0efcae4 --- /dev/null +++ b/.idea/deploymentTargetSelector.xml @@ -0,0 +1,18 @@ + + + + + + + + + \ No newline at end of file diff --git a/.idea/gradle.xml b/.idea/gradle.xml new file mode 100644 index 0000000..0897082 --- /dev/null +++ b/.idea/gradle.xml @@ -0,0 +1,19 @@ + + + + + + + \ No newline at end of file diff --git a/.idea/inspectionProfiles/Project_Default.xml b/.idea/inspectionProfiles/Project_Default.xml new file mode 100644 index 0000000..5aee9d4 --- /dev/null +++ b/.idea/inspectionProfiles/Project_Default.xml @@ -0,0 +1,6 @@ + + + + \ No newline at end of file diff --git a/.idea/kotlinc.xml b/.idea/kotlinc.xml new file mode 100644 index 0000000..4515aa3 --- /dev/null +++ b/.idea/kotlinc.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/.idea/migrations.xml b/.idea/migrations.xml new file mode 100644 index 0000000..f8051a6 --- /dev/null +++ b/.idea/migrations.xml @@ -0,0 +1,10 @@ + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..b0ca704 --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,25 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..dbf637d --- /dev/null +++ b/README.md @@ -0,0 +1,35 @@ +# Absensi-Apps +Membuat Aplikasi Absensi dengan Android Studio + +# Tutorial Build with Android Studio +https://youtu.be/naA_RTzY9qA + +# Tutorial Build with Step by Step +https://rivaldi48.blogspot.com/2022/06/tutorial-membuat-aplikasi-absensi-dengan-android-studio.html + + + +****If you use the Source Code, please make sure to credit and backlink to [Azhar Rivaldi](https://rivaldi48.blogspot.com/)*** + +## 👇 Click For Support Me : + + + +## 📄 License + +``` +Copyright (C) Azhar Rivaldi + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + +``` diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000..42afabf --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build \ No newline at end of file diff --git a/app/build.gradle b/app/build.gradle new file mode 100644 index 0000000..661f0c2 --- /dev/null +++ b/app/build.gradle @@ -0,0 +1,64 @@ +plugins { + id 'com.android.application' + id 'org.jetbrains.kotlin.android' + id 'kotlin-android-extensions' + id 'kotlin-kapt' +} + +android { + compileSdk 32 + + defaultConfig { + applicationId "com.azhar.absensi" + minSdk 21 + targetSdk 32 + versionCode 1 + versionName "1.0" + + testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" + } + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' + } + } + compileOptions { + sourceCompatibility JavaVersion.VERSION_1_8 + targetCompatibility JavaVersion.VERSION_1_8 + } +} + +dependencies { + + implementation 'androidx.core:core-ktx:1.7.0' + implementation 'androidx.appcompat:appcompat:1.4.1' + implementation 'com.google.android.material:material:1.6.0' + implementation 'com.google.android.gms:play-services-location:19.0.1' + implementation 'androidx.constraintlayout:constraintlayout:2.1.4' + testImplementation 'junit:junit:4.13.2' + androidTestImplementation 'androidx.test.ext:junit:1.1.3' + androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0' + + // Lifecycle Components + implementation 'androidx.lifecycle:lifecycle-livedata:2.4.1' + implementation 'androidx.lifecycle:lifecycle-viewmodel:2.4.1' + implementation "androidx.lifecycle:lifecycle-extensions:2.2.0" + + // Room Database + implementation "androidx.room:room-rxjava3:2.4.2" + implementation "androidx.room:room-runtime:2.4.2" + kapt "androidx.room:room-compiler:2.4.2" + + // Glide + implementation 'com.github.bumptech.glide:glide:4.12.0' + annotationProcessor 'com.github.bumptech.glide:compiler:4.12.0' + + // Rx Java + implementation 'io.reactivex.rxjava3:rxandroid:3.0.0' + implementation 'io.reactivex.rxjava3:rxjava:3.0.0' + + // Custom Permission + implementation 'com.karumi:dexter:6.2.2' +} \ No newline at end of file diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro new file mode 100644 index 0000000..481bb43 --- /dev/null +++ b/app/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/app/src/androidTest/java/com/azhar/absensi/ExampleInstrumentedTest.kt b/app/src/androidTest/java/com/azhar/absensi/ExampleInstrumentedTest.kt new file mode 100644 index 0000000..7d46253 --- /dev/null +++ b/app/src/androidTest/java/com/azhar/absensi/ExampleInstrumentedTest.kt @@ -0,0 +1,24 @@ +package com.azhar.absensi + +import androidx.test.platform.app.InstrumentationRegistry +import androidx.test.ext.junit.runners.AndroidJUnit4 + +import org.junit.Test +import org.junit.runner.RunWith + +import org.junit.Assert.* + +/** + * Instrumented test, which will execute on an Android device. + * + * See [testing documentation](http://d.android.com/tools/testing). + */ +@RunWith(AndroidJUnit4::class) +class ExampleInstrumentedTest { + @Test + fun useAppContext() { + // Context of the app under test. + val appContext = InstrumentationRegistry.getInstrumentation().targetContext + assertEquals("com.azhar.absensi", appContext.packageName) + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..ff21047 --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,60 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/database/AppDatabase.kt b/app/src/main/java/com/azhar/absensi/database/AppDatabase.kt new file mode 100644 index 0000000..a9c1051 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/database/AppDatabase.kt @@ -0,0 +1,20 @@ +package com.azhar.absensi.database + +import androidx.room.Database +import com.azhar.absensi.model.ModelDatabase +import androidx.room.RoomDatabase +import com.azhar.absensi.database.dao.DatabaseDao + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +@Database(entities = [ModelDatabase::class], version = 1, exportSchema = false) +abstract class AppDatabase : RoomDatabase() { + abstract fun databaseDao(): DatabaseDao? +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/database/DatabaseClient.kt b/app/src/main/java/com/azhar/absensi/database/DatabaseClient.kt new file mode 100644 index 0000000..5c9f886 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/database/DatabaseClient.kt @@ -0,0 +1,33 @@ +package com.azhar.absensi.database + +import android.content.Context +import androidx.room.Room + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +class DatabaseClient private constructor(context: Context) { + + var appDatabase: AppDatabase = Room.databaseBuilder(context, AppDatabase::class.java, "absensi_db") + .fallbackToDestructiveMigration() + .build() + + companion object { + private var mInstance: DatabaseClient? = null + @JvmStatic + @Synchronized + fun getInstance(context: Context): DatabaseClient? { + if (mInstance == null) { + mInstance = DatabaseClient(context) + } + return mInstance + } + } + +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/database/dao/DatabaseDao.kt b/app/src/main/java/com/azhar/absensi/database/dao/DatabaseDao.kt new file mode 100644 index 0000000..c9075ac --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/database/dao/DatabaseDao.kt @@ -0,0 +1,32 @@ +package com.azhar.absensi.database.dao + +import androidx.lifecycle.LiveData +import androidx.room.Dao +import androidx.room.Insert +import com.azhar.absensi.model.ModelDatabase +import androidx.room.OnConflictStrategy +import androidx.room.Query + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +@Dao +interface DatabaseDao { + @Query("SELECT * FROM tbl_absensi") + fun getAllHistory(): LiveData> + + @Insert(onConflict = OnConflictStrategy.REPLACE) + fun insertData(vararg modelDatabases: ModelDatabase) + + @Query("DELETE FROM tbl_absensi WHERE uid= :uid") + fun deleteHistoryById(uid: Int) + + @Query("DELETE FROM tbl_absensi") + fun deleteAllHistory() +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/model/ModelDatabase.kt b/app/src/main/java/com/azhar/absensi/model/ModelDatabase.kt new file mode 100644 index 0000000..f744b66 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/model/ModelDatabase.kt @@ -0,0 +1,38 @@ +package com.azhar.absensi.model + +import androidx.room.PrimaryKey +import androidx.room.ColumnInfo +import androidx.room.Entity +import java.io.Serializable + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +@Entity(tableName = "tbl_absensi") +class ModelDatabase : Serializable { + + @PrimaryKey(autoGenerate = true) + @ColumnInfo(name = "uid") + var uid = 0 + + @ColumnInfo(name = "nama") + lateinit var nama: String + + @ColumnInfo(name = "foto_selfie") + lateinit var fotoSelfie: String + + @ColumnInfo(name = "tanggal") + lateinit var tanggal: String + + @ColumnInfo(name = "lokasi") + lateinit var lokasi: String + + @ColumnInfo(name = "keterangan") + lateinit var keterangan: String +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/utils/BitmapManager.kt b/app/src/main/java/com/azhar/absensi/utils/BitmapManager.kt new file mode 100644 index 0000000..5bb5a84 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/utils/BitmapManager.kt @@ -0,0 +1,30 @@ +package com.azhar.absensi.utils + +import android.graphics.Bitmap +import android.graphics.BitmapFactory +import android.util.Base64 +import java.io.ByteArrayOutputStream + +/** + * Created by Azhar Rivaldi on 17-10-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +object BitmapManager { + + fun base64ToBitmap(base64: String): Bitmap { + val decodedString = Base64.decode(base64, Base64.DEFAULT) + return BitmapFactory.decodeByteArray(decodedString, 0, decodedString.size) + } + + fun bitmapToBase64(bitmap: Bitmap): String { + val byteArrayOutputStream = ByteArrayOutputStream() + bitmap.compress(Bitmap.CompressFormat.PNG, 70, byteArrayOutputStream) + val byteArray = byteArrayOutputStream.toByteArray() + return Base64.encodeToString(byteArray, Base64.DEFAULT) + } +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/utils/SessionLogin.kt b/app/src/main/java/com/azhar/absensi/utils/SessionLogin.kt new file mode 100644 index 0000000..face676 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/utils/SessionLogin.kt @@ -0,0 +1,58 @@ +package com.azhar.absensi.utils + +import android.content.Context +import android.content.Intent +import android.content.SharedPreferences +import com.azhar.absensi.view.login.LoginActivity + +/** + * Created by Azhar Rivaldi on 28-12-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +class SessionLogin(var context: Context) { + var pref: SharedPreferences + var editor: SharedPreferences.Editor + var PRIVATE_MODE = 0 + + fun createLoginSession(nama: String) { + editor.putBoolean(IS_LOGIN, true) + editor.putString(KEY_NAMA, nama) + editor.commit() + } + + fun checkLogin() { + if (!isLoggedIn()) { + val intent = Intent(context, LoginActivity::class.java) + intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) + intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK + context.startActivity(intent) + } + } + + fun logoutUser() { + editor.clear() + editor.commit() + val intent = Intent(context, LoginActivity::class.java) + intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) + intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK + context.startActivity(intent) + } + + fun isLoggedIn(): Boolean = pref.getBoolean(IS_LOGIN, false) + + companion object { + private const val PREF_NAME = "AbsensiPref" + private const val IS_LOGIN = "IsLoggedIn" + const val KEY_NAMA = "NAMA" + } + + init { + pref = context.getSharedPreferences(PREF_NAME, PRIVATE_MODE) + editor = pref.edit() + } +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/view/absen/AbsenActivity.kt b/app/src/main/java/com/azhar/absensi/view/absen/AbsenActivity.kt new file mode 100644 index 0000000..a728f5d --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/view/absen/AbsenActivity.kt @@ -0,0 +1,309 @@ +package com.azhar.absensi.view.absen + +import android.Manifest +import android.app.DatePickerDialog +import android.app.DatePickerDialog.OnDateSetListener +import android.app.ProgressDialog +import android.content.Intent +import android.content.pm.PackageManager +import android.graphics.Bitmap +import android.graphics.BitmapFactory +import android.graphics.Matrix +import android.location.Geocoder +import android.os.Bundle +import android.os.Environment +import android.provider.MediaStore +import android.view.MenuItem +import android.view.View +import android.widget.DatePicker +import android.widget.Toast +import androidx.appcompat.app.AppCompatActivity +import androidx.core.app.ActivityCompat +import androidx.core.content.FileProvider +import androidx.exifinterface.media.ExifInterface +import androidx.lifecycle.ViewModelProvider +import com.azhar.absensi.BuildConfig +import com.azhar.absensi.R +import com.azhar.absensi.utils.BitmapManager.bitmapToBase64 +import com.azhar.absensi.viewmodel.AbsenViewModel +import com.bumptech.glide.Glide +import com.bumptech.glide.load.engine.DiskCacheStrategy +import com.google.android.gms.location.LocationServices +import com.karumi.dexter.Dexter +import com.karumi.dexter.MultiplePermissionsReport +import com.karumi.dexter.PermissionToken +import com.karumi.dexter.listener.PermissionRequest +import com.karumi.dexter.listener.multi.MultiplePermissionsListener +import kotlinx.android.synthetic.main.activity_absen.* +import java.io.File +import java.io.IOException +import java.text.SimpleDateFormat +import java.util.* + +class AbsenActivity : AppCompatActivity() { + var REQ_CAMERA = 101 + var strCurrentLatitude = 0.0 + var strCurrentLongitude = 0.0 + var strFilePath: String = "" + var strLatitude = "0" + var strLongitude = "0" + lateinit var fileDirectoty: File + lateinit var imageFilename: File + lateinit var exifInterface: ExifInterface + lateinit var strBase64Photo: String + lateinit var strCurrentLocation: String + lateinit var strTitle: String + lateinit var strTimeStamp: String + lateinit var strImageName: String + lateinit var absenViewModel: AbsenViewModel + lateinit var progressDialog: ProgressDialog + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_absen) + + setInitLayout() + setCurrentLocation() + setUploadData() + } + + private fun setCurrentLocation() { + progressDialog.show() + val fusedLocationClient = LocationServices.getFusedLocationProviderClient(this) + if (ActivityCompat.checkSelfPermission(this, + Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED + && ActivityCompat.checkSelfPermission(this, + Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) { + return + } + fusedLocationClient.lastLocation + .addOnSuccessListener(this) { location -> + progressDialog.dismiss() + if (location != null) { + strCurrentLatitude = location.latitude + strCurrentLongitude = location.longitude + val geocoder = Geocoder(this@AbsenActivity, Locale.getDefault()) + try { + val addressList = + geocoder.getFromLocation(strCurrentLatitude, strCurrentLongitude, 1) + if (addressList != null && addressList.size > 0) { + strCurrentLocation = addressList[0].getAddressLine(0) + inputLokasi.setText(strCurrentLocation) + } + } catch (e: IOException) { + e.printStackTrace() + } + } else { + progressDialog.dismiss() + Toast.makeText(this@AbsenActivity, + "Ups, gagal mendapatkan lokasi. Silahkan periksa GPS atau koneksi internet Anda!", + Toast.LENGTH_SHORT).show() + strLatitude = "0" + strLongitude = "0" + } + } + } + + private fun setInitLayout() { + progressDialog = ProgressDialog(this) + strTitle = intent.extras?.getString(DATA_TITLE).toString() + + if (strTitle != null) { + tvTitle.text = strTitle + } + + setSupportActionBar(toolbar) + if (supportActionBar != null) { + supportActionBar?.setDisplayHomeAsUpEnabled(true) + supportActionBar?.setDisplayShowTitleEnabled(false) + } + + absenViewModel = ViewModelProvider(this, (ViewModelProvider.AndroidViewModelFactory + .getInstance(this.application) as ViewModelProvider.Factory)).get(AbsenViewModel::class.java) + + inputTanggal.setOnClickListener { + val tanggalAbsen = Calendar.getInstance() + val date = + OnDateSetListener { _: DatePicker, year: Int, monthOfYear: Int, dayOfMonth: Int -> + tanggalAbsen[Calendar.YEAR] = year + tanggalAbsen[Calendar.MONTH] = monthOfYear + tanggalAbsen[Calendar.DAY_OF_MONTH] = dayOfMonth + val strFormatDefault = "dd MMMM yyyy HH:mm" + val simpleDateFormat = SimpleDateFormat(strFormatDefault, Locale.getDefault()) + inputTanggal.setText(simpleDateFormat.format(tanggalAbsen.time)) + } + DatePickerDialog( + this@AbsenActivity, date, + tanggalAbsen[Calendar.YEAR], + tanggalAbsen[Calendar.MONTH], + tanggalAbsen[Calendar.DAY_OF_MONTH] + ).show() + } + + layoutImage.setOnClickListener { + Dexter.withContext(this@AbsenActivity) + .withPermissions( + Manifest.permission.CAMERA, + Manifest.permission.WRITE_EXTERNAL_STORAGE, + Manifest.permission.ACCESS_COARSE_LOCATION, + Manifest.permission.ACCESS_FINE_LOCATION + ) + .withListener(object : MultiplePermissionsListener { + override fun onPermissionsChecked(report: MultiplePermissionsReport) { + if (report.areAllPermissionsGranted()) { + try { + val cameraIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE) + cameraIntent.putExtra( + "com.google.assistant.extra.USE_FRONT_CAMERA", + true + ) + cameraIntent.putExtra("android.intent.extra.USE_FRONT_CAMERA", true) + cameraIntent.putExtra("android.intent.extras.LENS_FACING_FRONT", 1) + cameraIntent.putExtra("android.intent.extras.CAMERA_FACING", 1) + + // Samsung + cameraIntent.putExtra("camerafacing", "front") + cameraIntent.putExtra("previous_mode", "front") + + // Huawei + cameraIntent.putExtra("default_camera", "1") + cameraIntent.putExtra( + "default_mode", + "com.huawei.camera2.mode.photo.PhotoMode") + cameraIntent.putExtra( + MediaStore.EXTRA_OUTPUT, + FileProvider.getUriForFile( + this@AbsenActivity, + BuildConfig.APPLICATION_ID + ".provider", + createImageFile() + ) + ) + startActivityForResult(cameraIntent, REQ_CAMERA) + } catch (ex: IOException) { + Toast.makeText(this@AbsenActivity, + "Ups, gagal membuka kamera", Toast.LENGTH_SHORT).show() + } + } + } + + override fun onPermissionRationaleShouldBeShown( + permissions: List, + token: PermissionToken) { + token.continuePermissionRequest() + } + }).check() + } + } + + private fun setUploadData() { + btnAbsen.setOnClickListener { + val strNama = inputNama.text.toString() + val strTanggal = inputTanggal.text.toString() + val strKeterangan = inputKeterangan.text.toString() + if (strFilePath.equals(null) || strNama.isEmpty() || strCurrentLocation.isEmpty() + || strTanggal.isEmpty() || strKeterangan.isEmpty()) { + Toast.makeText(this@AbsenActivity, + "Data tidak boleh ada yang kosong!", Toast.LENGTH_SHORT).show() + } else { + absenViewModel.addDataAbsen( + strBase64Photo, + strNama, + strTanggal, + strCurrentLocation, + strKeterangan) + Toast.makeText(this@AbsenActivity, + "Laporan Anda terkirim, tunggu info selanjutnya ya!", Toast.LENGTH_SHORT).show() + finish() + } + } + } + + @Throws(IOException::class) + private fun createImageFile(): File { + strTimeStamp = SimpleDateFormat("dd MMMM yyyy HH:mm:ss").format(Date()) + strImageName = "IMG_" + fileDirectoty = File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM), "") + imageFilename = File.createTempFile(strImageName, ".jpg", fileDirectoty) + strFilePath = imageFilename.getAbsolutePath() + return imageFilename + } + + public override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { + super.onActivityResult(requestCode, resultCode, data) + convertImage(strFilePath) + } + + private fun convertImage(imageFilePath: String?) { + val imageFile = File(imageFilePath) + if (imageFile.exists()) { + val options = BitmapFactory.Options() + var bitmapImage = BitmapFactory.decodeFile(strFilePath, options) + + try { + exifInterface = ExifInterface(imageFile.absolutePath) + } catch (e: IOException) { + e.printStackTrace() + } + + val orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION, 0) + val matrix = Matrix() + if (orientation == 6) { + matrix.postRotate(90f) + } else if (orientation == 3) { + matrix.postRotate(180f) + } else if (orientation == 8) { + matrix.postRotate(270f) + } + + bitmapImage = Bitmap.createBitmap( + bitmapImage, + 0, + 0, + bitmapImage.width, + bitmapImage.height, + matrix, + true + ) + + if (bitmapImage == null) { + Toast.makeText(this@AbsenActivity, + "Ups, foto kamu belum ada!", Toast.LENGTH_LONG).show() + } else { + val resizeImage = (bitmapImage.height * (512.0 / bitmapImage.width)).toInt() + val scaledBitmap = Bitmap.createScaledBitmap(bitmapImage, 512, resizeImage, true) + Glide.with(this) + .load(scaledBitmap) + .diskCacheStrategy(DiskCacheStrategy.ALL) + .placeholder(R.drawable.ic_photo_camera) + .into(imageSelfie) + strBase64Photo = bitmapToBase64(scaledBitmap) + } + } + } + + override fun onRequestPermissionsResult( + requestCode: Int, + permissions: Array, + grantResults: IntArray) { + super.onRequestPermissionsResult(requestCode, permissions, grantResults) + for (grantResult in grantResults) { + if (grantResult == PackageManager.PERMISSION_GRANTED) { + val intent = intent + finish() + startActivity(intent) + } + } + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + if (item.itemId == android.R.id.home) { + finish() + return true + } + return super.onOptionsItemSelected(item) + } + + companion object { + const val DATA_TITLE = "TITLE" + } +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/view/history/HistoryActivity.kt b/app/src/main/java/com/azhar/absensi/view/history/HistoryActivity.kt new file mode 100644 index 0000000..0aa942e --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/view/history/HistoryActivity.kt @@ -0,0 +1,82 @@ +package com.azhar.absensi.view.history + +import android.content.DialogInterface +import android.os.Bundle +import android.view.MenuItem +import android.view.View +import android.widget.Toast +import androidx.appcompat.app.AlertDialog +import androidx.appcompat.app.AppCompatActivity +import androidx.lifecycle.ViewModelProviders +import androidx.recyclerview.widget.LinearLayoutManager +import com.azhar.absensi.R +import com.azhar.absensi.model.ModelDatabase +import com.azhar.absensi.view.history.HistoryAdapter.HistoryAdapterCallback +import com.azhar.absensi.viewmodel.HistoryViewModel +import kotlinx.android.synthetic.main.activity_history.* + +class HistoryActivity : AppCompatActivity(), HistoryAdapterCallback { + var modelDatabaseList: MutableList = ArrayList() + lateinit var historyAdapter: HistoryAdapter + lateinit var historyViewModel: HistoryViewModel + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_history) + + setInitLayout() + setViewModel() + } + + private fun setInitLayout() { + setSupportActionBar(toolbar) + if (supportActionBar != null) { + supportActionBar?.setDisplayHomeAsUpEnabled(true) + supportActionBar?.setDisplayShowTitleEnabled(false) + } + + tvNotFound.visibility = View.GONE + + historyAdapter = HistoryAdapter(this, modelDatabaseList, this) + rvHistory.setHasFixedSize(true) + rvHistory.layoutManager = LinearLayoutManager(this) + rvHistory.adapter = historyAdapter + } + + private fun setViewModel() { + historyViewModel = ViewModelProviders.of(this).get(HistoryViewModel::class.java) + historyViewModel.dataLaporan.observe(this) { modelDatabases: List -> + if (modelDatabases.isEmpty()) { + tvNotFound.visibility = View.VISIBLE + rvHistory.visibility = View.GONE + } else { + tvNotFound.visibility = View.GONE + rvHistory.visibility = View.VISIBLE + } + historyAdapter.setDataAdapter(modelDatabases) + } + } + + override fun onDelete(modelDatabase: ModelDatabase?) { + val alertDialogBuilder = AlertDialog.Builder(this) + alertDialogBuilder.setMessage("Hapus riwayat ini?") + alertDialogBuilder.setPositiveButton("Ya, Hapus") { dialogInterface, i -> + val uid = modelDatabase!!.uid + historyViewModel.deleteDataById(uid) + Toast.makeText(this@HistoryActivity, "Yeay! Data yang dipilih sudah dihapus", + Toast.LENGTH_SHORT).show() + } + alertDialogBuilder.setNegativeButton("Batal") { dialogInterface: DialogInterface, i: + Int -> dialogInterface.cancel() } + val alertDialog = alertDialogBuilder.create() + alertDialog.show() + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + if (item.itemId == android.R.id.home) { + finish() + return true + } + return super.onOptionsItemSelected(item) + } +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/view/history/HistoryAdapter.kt b/app/src/main/java/com/azhar/absensi/view/history/HistoryAdapter.kt new file mode 100644 index 0000000..7aa5a39 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/view/history/HistoryAdapter.kt @@ -0,0 +1,102 @@ +package com.azhar.absensi.view.history + +import android.content.Context +import android.content.res.ColorStateList +import android.graphics.Color +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.TextView +import androidx.cardview.widget.CardView +import androidx.recyclerview.widget.RecyclerView +import com.azhar.absensi.R +import com.azhar.absensi.model.ModelDatabase +import com.azhar.absensi.utils.BitmapManager.base64ToBitmap +import com.bumptech.glide.Glide +import com.bumptech.glide.load.engine.DiskCacheStrategy +import com.google.android.material.imageview.ShapeableImageView +import kotlinx.android.synthetic.main.list_history_absen.view.* +import java.lang.String +import kotlin.Int + +/** + * Created by Azhar Rivaldi on 30-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +class HistoryAdapter( + var mContext: Context, + var modelDatabase: MutableList, + var mAdapterCallback: HistoryAdapterCallback) : RecyclerView.Adapter() { + + fun setDataAdapter(items: List) { + modelDatabase.clear() + modelDatabase.addAll(items) + notifyDataSetChanged() + } + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { + val view = LayoutInflater.from(parent.context).inflate(R.layout.list_history_absen, parent, false) + return ViewHolder(view) + } + + override fun onBindViewHolder(holder: ViewHolder, position: Int) { + val data = modelDatabase[position] + holder.tvNomor.text = String.valueOf(data.uid) + holder.tvNama.text = data.nama + holder.tvLokasi.text = data.lokasi + holder.tvAbsenTime.text = data.tanggal + holder.tvStatusAbsen.text = data.keterangan + + Glide.with(mContext) + .load(base64ToBitmap(data.fotoSelfie)) + .diskCacheStrategy(DiskCacheStrategy.ALL) + .placeholder(R.drawable.ic_photo_camera) + .into(holder.imageProfile) + + when (data.keterangan) { + "Absen Masuk" -> { + holder.colorStatus.setBackgroundResource(R.drawable.bg_circle_radius) + holder.colorStatus.backgroundTintList = ColorStateList.valueOf(Color.GREEN) + } + "Absen Keluar" -> { + holder.colorStatus.setBackgroundResource(R.drawable.bg_circle_radius) + holder.colorStatus.backgroundTintList = ColorStateList.valueOf(Color.RED) + } + "Izin" -> { + holder.colorStatus.setBackgroundResource(R.drawable.bg_circle_radius) + holder.colorStatus.backgroundTintList = ColorStateList.valueOf(Color.BLUE) + } + } + } + + override fun getItemCount(): Int { + return modelDatabase.size + } + + inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) { + var tvStatusAbsen: TextView = itemView.tvStatusAbsen + var tvNomor: TextView = itemView.tvNomor + var tvNama: TextView = itemView.tvNama + var tvLokasi: TextView = itemView.tvLokasi + var tvAbsenTime: TextView = itemView.tvAbsenTime + var cvHistory: CardView = itemView.cvHistory + var imageProfile: ShapeableImageView = itemView.imageProfile + var colorStatus: View = itemView.colorStatus + + init { + cvHistory.setOnClickListener { + val modelLaundry = modelDatabase[adapterPosition] + mAdapterCallback.onDelete(modelLaundry) + } + } + } + + interface HistoryAdapterCallback { + fun onDelete(modelDatabase: ModelDatabase?) + } +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/view/login/LoginActivity.kt b/app/src/main/java/com/azhar/absensi/view/login/LoginActivity.kt new file mode 100644 index 0000000..8e7ad86 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/view/login/LoginActivity.kt @@ -0,0 +1,82 @@ +package com.azhar.absensi.view.login + +import android.Manifest +import android.content.Intent +import android.content.pm.PackageManager +import android.os.Bundle +import android.widget.Toast +import androidx.appcompat.app.AppCompatActivity +import androidx.core.app.ActivityCompat +import com.azhar.absensi.R +import com.azhar.absensi.utils.SessionLogin +import com.azhar.absensi.view.main.MainActivity +import kotlinx.android.synthetic.main.activity_login.* + +class LoginActivity : AppCompatActivity() { + lateinit var session: SessionLogin + lateinit var strNama: String + lateinit var strPassword: String + var REQ_PERMISSION = 101 + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_login) + + setPermission() + setInitLayout() + } + + private fun setPermission() { + if (ActivityCompat.checkSelfPermission(this, + Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED + && ActivityCompat.checkSelfPermission(this, + Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) { + ActivityCompat.requestPermissions(this, + arrayOf(Manifest.permission.ACCESS_FINE_LOCATION), + REQ_PERMISSION + ) + } + } + + private fun setInitLayout() { + session = SessionLogin(applicationContext) + + if (session.isLoggedIn()) { + startActivity(Intent(this@LoginActivity, MainActivity::class.java)) + finish() + } + + btnLogin.setOnClickListener { + strNama = inputNama.text.toString() + strPassword = inputPassword.text.toString() + + if (strNama.isEmpty() || strPassword.isEmpty()) { + Toast.makeText(this@LoginActivity, "Form tidak boleh kosong!", + Toast.LENGTH_SHORT).show() + } else { + val intent = Intent(this@LoginActivity, MainActivity::class.java) + startActivity(intent) + session.createLoginSession(strNama) + } + } + } + + override fun onRequestPermissionsResult( + requestCode: Int, + permissions: Array, + grantResults: IntArray) { + super.onRequestPermissionsResult(requestCode, permissions, grantResults) + for (grantResult in grantResults) { + if (grantResult == PackageManager.PERMISSION_GRANTED) { + val intent = intent + finish() + startActivity(intent) + } + } + } + + override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { + super.onActivityResult(requestCode, resultCode, data) + } + +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/view/main/MainActivity.kt b/app/src/main/java/com/azhar/absensi/view/main/MainActivity.kt new file mode 100644 index 0000000..cbf2c28 --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/view/main/MainActivity.kt @@ -0,0 +1,69 @@ +package com.azhar.absensi.view.main + +import android.content.Intent +import android.os.Bundle +import androidx.appcompat.app.AlertDialog +import androidx.appcompat.app.AppCompatActivity +import com.azhar.absensi.R +import com.azhar.absensi.utils.SessionLogin +import com.azhar.absensi.view.absen.AbsenActivity +import com.azhar.absensi.view.history.HistoryActivity +import kotlinx.android.synthetic.main.activity_main.* + +class MainActivity : AppCompatActivity() { + + lateinit var strTitle: String + lateinit var session: SessionLogin + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_main) + + setInitLayout() + } + + private fun setInitLayout() { + session = SessionLogin(this) + session.checkLogin() + + cvAbsenMasuk.setOnClickListener { + strTitle = "Absen Masuk" + val intent = Intent(this@MainActivity, AbsenActivity::class.java) + intent.putExtra(AbsenActivity.DATA_TITLE, strTitle) + startActivity(intent) + } + + cvAbsenKeluar.setOnClickListener { + strTitle = "Absen Keluar" + val intent = Intent(this@MainActivity, AbsenActivity::class.java) + intent.putExtra(AbsenActivity.DATA_TITLE, strTitle) + startActivity(intent) + } + + cvPerizinan.setOnClickListener { + strTitle = "Izin" + val intent = Intent(this@MainActivity, AbsenActivity::class.java) + intent.putExtra(AbsenActivity.DATA_TITLE, strTitle) + startActivity(intent) + } + + cvHistory.setOnClickListener { + val intent = Intent(this@MainActivity, HistoryActivity::class.java) + startActivity(intent) + } + + imageLogout.setOnClickListener { + val builder = AlertDialog.Builder(this@MainActivity) + builder.setMessage("Yakin Anda ingin Logout?") + builder.setCancelable(true) + builder.setNegativeButton("Batal") { dialog, which -> dialog.cancel() } + builder.setPositiveButton("Ya") { dialog, which -> + session.logoutUser() + finishAffinity() + } + val alertDialog = builder.create() + alertDialog.show() + } + } + +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/viewmodel/AbsenViewModel.kt b/app/src/main/java/com/azhar/absensi/viewmodel/AbsenViewModel.kt new file mode 100644 index 0000000..9816b6b --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/viewmodel/AbsenViewModel.kt @@ -0,0 +1,41 @@ +package com.azhar.absensi.viewmodel + +import android.app.Application +import androidx.lifecycle.AndroidViewModel +import com.azhar.absensi.database.DatabaseClient.Companion.getInstance +import com.azhar.absensi.database.dao.DatabaseDao +import com.azhar.absensi.model.ModelDatabase +import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers +import io.reactivex.rxjava3.core.Completable +import io.reactivex.rxjava3.schedulers.Schedulers + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +class AbsenViewModel(application: Application) : AndroidViewModel(application) { + var databaseDao: DatabaseDao? = getInstance(application)?.appDatabase?.databaseDao() + + fun addDataAbsen( + foto: String, nama: String, + tanggal: String, lokasi: String, keterangan: String) { + Completable.fromAction { + val modelDatabase = ModelDatabase() + modelDatabase.fotoSelfie = foto + modelDatabase.nama = nama + modelDatabase.tanggal = tanggal + modelDatabase.lokasi = lokasi + modelDatabase.keterangan = keterangan + databaseDao?.insertData(modelDatabase) + } + .subscribeOn(Schedulers.io()) + .observeOn(AndroidSchedulers.mainThread()) + .subscribe() + } + +} \ No newline at end of file diff --git a/app/src/main/java/com/azhar/absensi/viewmodel/HistoryViewModel.kt b/app/src/main/java/com/azhar/absensi/viewmodel/HistoryViewModel.kt new file mode 100644 index 0000000..376e1cf --- /dev/null +++ b/app/src/main/java/com/azhar/absensi/viewmodel/HistoryViewModel.kt @@ -0,0 +1,38 @@ +package com.azhar.absensi.viewmodel + +import android.app.Application +import androidx.lifecycle.AndroidViewModel +import androidx.lifecycle.LiveData +import com.azhar.absensi.database.DatabaseClient.Companion.getInstance +import com.azhar.absensi.database.dao.DatabaseDao +import com.azhar.absensi.model.ModelDatabase +import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers +import io.reactivex.rxjava3.core.Completable +import io.reactivex.rxjava3.schedulers.Schedulers + +/** + * Created by Azhar Rivaldi on 19-11-2021 + * Youtube Channel : https://bit.ly/2PJMowZ + * Github : https://github.com/AzharRivaldi + * Twitter : https://twitter.com/azharrvldi_ + * Instagram : https://www.instagram.com/azhardvls_ + * LinkedIn : https://www.linkedin.com/in/azhar-rivaldi + */ + +class HistoryViewModel(application: Application) : AndroidViewModel(application) { + var dataLaporan: LiveData> + var databaseDao: DatabaseDao? = getInstance(application)?.appDatabase?.databaseDao() + + fun deleteDataById(uid: Int) { + Completable.fromAction { + databaseDao?.deleteHistoryById(uid) + } + .subscribeOn(Schedulers.io()) + .observeOn(AndroidSchedulers.mainThread()) + .subscribe() + } + + init { + dataLaporan = databaseDao!!.getAllHistory() + } +} \ No newline at end of file diff --git a/app/src/main/res/drawable-v24/ic_launcher_foreground.xml b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml new file mode 100644 index 0000000..2b068d1 --- /dev/null +++ b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_circle_radius.xml b/app/src/main/res/drawable/bg_circle_radius.xml new file mode 100644 index 0000000..403ec91 --- /dev/null +++ b/app/src/main/res/drawable/bg_circle_radius.xml @@ -0,0 +1,9 @@ + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_overlay.xml b/app/src/main/res/drawable/bg_overlay.xml new file mode 100644 index 0000000..17a9d2a --- /dev/null +++ b/app/src/main/res/drawable/bg_overlay.xml @@ -0,0 +1,8 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/border_line_dotted.xml b/app/src/main/res/drawable/border_line_dotted.xml new file mode 100644 index 0000000..e5b4095 --- /dev/null +++ b/app/src/main/res/drawable/border_line_dotted.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_arrow.xml b/app/src/main/res/drawable/ic_arrow.xml new file mode 100644 index 0000000..5985b2d --- /dev/null +++ b/app/src/main/res/drawable/ic_arrow.xml @@ -0,0 +1,12 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_back.xml b/app/src/main/res/drawable/ic_back.xml new file mode 100644 index 0000000..005fdaf --- /dev/null +++ b/app/src/main/res/drawable/ic_back.xml @@ -0,0 +1,11 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_history.jpg b/app/src/main/res/drawable/ic_history.jpg new file mode 100644 index 0000000..cbf23d8 Binary files /dev/null and b/app/src/main/res/drawable/ic_history.jpg differ diff --git a/app/src/main/res/drawable/ic_info.xml b/app/src/main/res/drawable/ic_info.xml new file mode 100644 index 0000000..7706011 --- /dev/null +++ b/app/src/main/res/drawable/ic_info.xml @@ -0,0 +1,10 @@ + + + diff --git a/app/src/main/res/drawable/ic_izin.jpg b/app/src/main/res/drawable/ic_izin.jpg new file mode 100644 index 0000000..dfe236c Binary files /dev/null and b/app/src/main/res/drawable/ic_izin.jpg differ diff --git a/app/src/main/res/drawable/ic_keluar.jpg b/app/src/main/res/drawable/ic_keluar.jpg new file mode 100644 index 0000000..1525092 Binary files /dev/null and b/app/src/main/res/drawable/ic_keluar.jpg differ diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 0000000..07d5da9 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,170 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/drawable/ic_logout.xml b/app/src/main/res/drawable/ic_logout.xml new file mode 100644 index 0000000..2059dea --- /dev/null +++ b/app/src/main/res/drawable/ic_logout.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable/ic_masuk.jpg b/app/src/main/res/drawable/ic_masuk.jpg new file mode 100644 index 0000000..d296ce6 Binary files /dev/null and b/app/src/main/res/drawable/ic_masuk.jpg differ diff --git a/app/src/main/res/drawable/ic_photo_camera.xml b/app/src/main/res/drawable/ic_photo_camera.xml new file mode 100644 index 0000000..97262af --- /dev/null +++ b/app/src/main/res/drawable/ic_photo_camera.xml @@ -0,0 +1,13 @@ + + + + diff --git a/app/src/main/res/drawable/logo.png b/app/src/main/res/drawable/logo.png new file mode 100644 index 0000000..419e9cd Binary files /dev/null and b/app/src/main/res/drawable/logo.png differ diff --git a/app/src/main/res/layout/activity_absen.xml b/app/src/main/res/layout/activity_absen.xml new file mode 100644 index 0000000..95e0142 --- /dev/null +++ b/app/src/main/res/layout/activity_absen.xml @@ -0,0 +1,269 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_history.xml b/app/src/main/res/layout/activity_history.xml new file mode 100644 index 0000000..18799e6 --- /dev/null +++ b/app/src/main/res/layout/activity_history.xml @@ -0,0 +1,71 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_login.xml b/app/src/main/res/layout/activity_login.xml new file mode 100644 index 0000000..f1e8266 --- /dev/null +++ b/app/src/main/res/layout/activity_login.xml @@ -0,0 +1,142 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml new file mode 100644 index 0000000..e4e7e86 --- /dev/null +++ b/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,291 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_history_absen.xml b/app/src/main/res/layout/list_history_absen.xml new file mode 100644 index 0000000..73f82d9 --- /dev/null +++ b/app/src/main/res/layout/list_history_absen.xml @@ -0,0 +1,197 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 0000000..eca70cf --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 0000000..eca70cf --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher.webp b/app/src/main/res/mipmap-hdpi/ic_launcher.webp new file mode 100644 index 0000000..c209e78 Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp new file mode 100644 index 0000000..b2dfe3d Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher.webp b/app/src/main/res/mipmap-mdpi/ic_launcher.webp new file mode 100644 index 0000000..4f0f1d6 Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp new file mode 100644 index 0000000..62b611d Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp new file mode 100644 index 0000000..948a307 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..1b9a695 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp new file mode 100644 index 0000000..28d4b77 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9287f50 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp new file mode 100644 index 0000000..aa7d642 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9126ae3 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/values-night/themes.xml b/app/src/main/res/values-night/themes.xml new file mode 100644 index 0000000..806018d --- /dev/null +++ b/app/src/main/res/values-night/themes.xml @@ -0,0 +1,16 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml new file mode 100644 index 0000000..f8c6127 --- /dev/null +++ b/app/src/main/res/values/colors.xml @@ -0,0 +1,10 @@ + + + #FFBB86FC + #FF6200EE + #FF3700B3 + #FF03DAC5 + #FF018786 + #FF000000 + #FFFFFFFF + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml new file mode 100644 index 0000000..19ca70b --- /dev/null +++ b/app/src/main/res/values/strings.xml @@ -0,0 +1,3 @@ + + Aplikasi Absensi + \ No newline at end of file diff --git a/app/src/main/res/values/themes.xml b/app/src/main/res/values/themes.xml new file mode 100644 index 0000000..1362d30 --- /dev/null +++ b/app/src/main/res/values/themes.xml @@ -0,0 +1,20 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/xml/file_paths.xml b/app/src/main/res/xml/file_paths.xml new file mode 100644 index 0000000..65599cc --- /dev/null +++ b/app/src/main/res/xml/file_paths.xml @@ -0,0 +1,18 @@ + + + + + + + + \ No newline at end of file diff --git a/app/src/test/java/com/azhar/absensi/ExampleUnitTest.kt b/app/src/test/java/com/azhar/absensi/ExampleUnitTest.kt new file mode 100644 index 0000000..db448f6 --- /dev/null +++ b/app/src/test/java/com/azhar/absensi/ExampleUnitTest.kt @@ -0,0 +1,17 @@ +package com.azhar.absensi + +import org.junit.Test + +import org.junit.Assert.* + +/** + * Example local unit test, which will execute on the development machine (host). + * + * See [testing documentation](http://d.android.com/tools/testing). + */ +class ExampleUnitTest { + @Test + fun addition_isCorrect() { + assertEquals(4, 2 + 2) + } +} \ No newline at end of file diff --git a/build.gradle b/build.gradle new file mode 100644 index 0000000..84ed49d --- /dev/null +++ b/build.gradle @@ -0,0 +1,10 @@ +// Top-level build file where you can add configuration options common to all sub-projects/modules. +plugins { + id 'com.android.application' version '7.1.1' apply false + id 'com.android.library' version '7.1.1' apply false + id 'org.jetbrains.kotlin.android' version '1.6.10' apply false +} + +task clean(type: Delete) { + delete rootProject.buildDir +} \ No newline at end of file diff --git a/gradle.properties b/gradle.properties new file mode 100644 index 0000000..cd0519b --- /dev/null +++ b/gradle.properties @@ -0,0 +1,23 @@ +# Project-wide Gradle settings. +# IDE (e.g. Android Studio) users: +# Gradle settings configured through the IDE *will override* +# any settings specified in this file. +# For more details on how to configure your build environment visit +# http://www.gradle.org/docs/current/userguide/build_environment.html +# Specifies the JVM arguments used for the daemon process. +# The setting is particularly useful for tweaking memory settings. +org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8 +# When configured, Gradle will run in incubating parallel mode. +# This option should only be used with decoupled projects. More details, visit +# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects +# org.gradle.parallel=true +# AndroidX package structure to make it clearer which packages are bundled with the +# Android operating system, and which are packaged with your app"s APK +# https://developer.android.com/topic/libraries/support-library/androidx-rn +android.useAndroidX=true +# Kotlin code style for this project: "official" or "obsolete": +kotlin.code.style=official +# Enables namespacing of each library's R class so that its R class includes only the +# resources declared in the library itself and none from the library's dependencies, +# thereby reducing the size of the R class for that library +android.nonTransitiveRClass=true \ No newline at end of file diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000..e708b1c Binary files /dev/null and b/gradle/wrapper/gradle-wrapper.jar differ diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..0822e93 --- /dev/null +++ b/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Tue Mar 15 11:40:13 ICT 2022 +distributionBase=GRADLE_USER_HOME +distributionUrl=https\://services.gradle.org/distributions/gradle-7.2-bin.zip +distributionPath=wrapper/dists +zipStorePath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME diff --git a/gradlew b/gradlew new file mode 100644 index 0000000..4f906e0 --- /dev/null +++ b/gradlew @@ -0,0 +1,185 @@ +#!/usr/bin/env sh + +# +# Copyright 2015 the original author or authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn () { + echo "$*" +} + +die () { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MINGW* ) + msys=true + ;; + NONSTOP* ) + nonstop=true + ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin or MSYS, switch paths to Windows format before running java +if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=`expr $i + 1` + done + case $i in + 0) set -- ;; + 1) set -- "$args0" ;; + 2) set -- "$args0" "$args1" ;; + 3) set -- "$args0" "$args1" "$args2" ;; + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Escape application args +save () { + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done + echo " " +} +APP_ARGS=`save "$@"` + +# Collect all arguments for the java command, following the shell quoting and substitution rules +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" + +exec "$JAVACMD" "$@" diff --git a/gradlew.bat b/gradlew.bat new file mode 100644 index 0000000..107acd3 --- /dev/null +++ b/gradlew.bat @@ -0,0 +1,89 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto execute + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/settings.gradle b/settings.gradle new file mode 100644 index 0000000..7c2b15a --- /dev/null +++ b/settings.gradle @@ -0,0 +1,16 @@ +pluginManagement { + repositories { + gradlePluginPortal() + google() + mavenCentral() + } +} +dependencyResolutionManagement { + repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) + repositories { + google() + mavenCentral() + } +} +rootProject.name = "Aplikasi Absensi" +include ':app'