EAS-202310715060-MuhammadYusronAmrullah

This commit is contained in:
202310715060 MUHAMMAD YUSRON AMRULLAH 2026-01-13 22:10:54 +07:00
parent 926d3e0a14
commit 741a6a8fd8
7 changed files with 614 additions and 222 deletions

26
.idea/appInsightsSettings.xml generated Normal file
View File

@ -0,0 +1,26 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AppInsightsSettings">
<option name="tabSettings">
<map>
<entry key="Firebase Crashlytics">
<value>
<InsightsFilterSettings>
<option name="connection">
<ConnectionSetting>
<option name="appId" value="PLACEHOLDER" />
<option name="mobileSdkAppId" value="" />
<option name="projectId" value="" />
<option name="projectNumber" value="" />
</ConnectionSetting>
</option>
<option name="signal" value="SIGNAL_UNSPECIFIED" />
<option name="timeIntervalDays" value="THIRTY_DAYS" />
<option name="visibilityType" value="ALL" />
</InsightsFilterSettings>
</value>
</entry>
</map>
</option>
</component>
</project>

View File

@ -4,10 +4,10 @@
<selectionStates> <selectionStates>
<SelectionState runConfigName="app"> <SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" /> <option name="selectionMode" value="DROPDOWN" />
<DropdownSelection timestamp="2026-01-13T05:20:56.137492Z"> <DropdownSelection timestamp="2026-01-13T12:44:47.478355400Z">
<Target type="DEFAULT_BOOT"> <Target type="DEFAULT_BOOT">
<handle> <handle>
<DeviceId pluginId="PhysicalDevice" identifier="serial=RR8TA08RD8Z" /> <DeviceId pluginId="PhysicalDevice" identifier="serial=y5zxozrgyhcafun7" />
</handle> </handle>
</Target> </Target>
</DropdownSelection> </DropdownSelection>

21
.idea/deviceManager.xml generated Normal file
View File

@ -0,0 +1,21 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="DeviceTable">
<option name="columnSorters">
<list>
<ColumnSorterState>
<option name="column" value="Type" />
<option name="order" value="ASCENDING" />
</ColumnSorterState>
<ColumnSorterState>
<option name="column" value="API" />
<option name="order" value="DESCENDING" />
</ColumnSorterState>
<ColumnSorterState>
<option name="column" value="Name" />
<option name="order" value="ASCENDING" />
</ColumnSorterState>
</list>
</option>
</component>
</project>

View File

@ -6,9 +6,7 @@ plugins {
android { android {
namespace = "id.ac.ubharajaya.sistemakademik" namespace = "id.ac.ubharajaya.sistemakademik"
compileSdk { compileSdk = 36
version = release(36)
}
defaultConfig { defaultConfig {
applicationId = "id.ac.ubharajaya.sistemakademik" applicationId = "id.ac.ubharajaya.sistemakademik"
@ -44,15 +42,19 @@ android {
dependencies { dependencies {
implementation(libs.androidx.core.ktx) implementation(libs.androidx.core.ktx)
implementation(libs.androidx.lifecycle.runtime.ktx) implementation(libs.androidx.lifecycle.runtime.ktx)
implementation(libs.androidx.activity.compose)
implementation("androidx.activity:activity-compose:1.9.0")
implementation(platform(libs.androidx.compose.bom)) implementation(platform(libs.androidx.compose.bom))
implementation(libs.androidx.compose.ui) implementation(libs.androidx.compose.ui)
implementation(libs.androidx.compose.ui.graphics) implementation(libs.androidx.compose.ui.graphics)
implementation(libs.androidx.compose.ui.tooling.preview) implementation(libs.androidx.compose.ui.tooling.preview)
implementation(libs.androidx.compose.material3) implementation(libs.androidx.compose.material3)
// Activity, Navigation, and Icons
implementation(libs.androidx.activity.compose)
implementation(libs.androidx.navigation.compose)
implementation(libs.androidx.compose.material.icons.extended)
// Location (GPS) // Location (GPS)
implementation("com.google.android.gms:play-services-location:21.0.1") implementation(libs.play.services.location)
testImplementation(libs.junit) testImplementation(libs.junit)
androidTestImplementation(libs.androidx.junit) androidTestImplementation(libs.androidx.junit)

View File

@ -1,7 +1,10 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" <manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"> xmlns:tools="http://schemas.android.com/tools">
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" android:required="false" />
<uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/> <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

View File

@ -14,261 +14,593 @@ import androidx.activity.compose.rememberLauncherForActivityResult
import androidx.activity.compose.setContent import androidx.activity.compose.setContent
import androidx.activity.enableEdgeToEdge import androidx.activity.enableEdgeToEdge
import androidx.activity.result.contract.ActivityResultContracts import androidx.activity.result.contract.ActivityResultContracts
import androidx.compose.foundation.Image
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.* import androidx.compose.foundation.layout.*
import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.shape.CircleShape
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.*
import androidx.compose.material3.* import androidx.compose.material3.*
import androidx.compose.runtime.* import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.clip
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.asImageBitmap
import androidx.compose.ui.layout.ContentScale
import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.input.ImeAction
import androidx.compose.ui.text.input.KeyboardType
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.core.content.ContextCompat import androidx.compose.ui.unit.sp
import androidx.core.app.ActivityCompat
import androidx.navigation.NavController
import androidx.navigation.NavType
import androidx.navigation.compose.NavHost
import androidx.navigation.compose.composable
import androidx.navigation.compose.rememberNavController
import androidx.navigation.navArgument
import com.google.android.gms.location.LocationServices import com.google.android.gms.location.LocationServices
import com.google.android.gms.location.Priority
import id.ac.ubharajaya.sistemakademik.ui.theme.SistemAkademikTheme import id.ac.ubharajaya.sistemakademik.ui.theme.SistemAkademikTheme
import org.json.JSONArray
import org.json.JSONObject import org.json.JSONObject
import java.io.ByteArrayOutputStream import java.io.ByteArrayOutputStream
import java.net.HttpURLConnection import java.net.HttpURLConnection
import java.net.URL import java.net.URL
import java.text.SimpleDateFormat
import java.util.*
import kotlin.concurrent.thread import kotlin.concurrent.thread
import kotlin.math.atan2
import kotlin.math.cos
import kotlin.math.sin
import kotlin.math.sqrt
/* ================= UTIL ================= */ /* ================= DATA CLASS ================= */
data class AttendanceRecord(val timestamp: Long, val status: String)
/* ================= UTILS ================= */
fun bitmapToBase64(bitmap: Bitmap): String { fun bitmapToBase64(bitmap: Bitmap): String {
val outputStream = ByteArrayOutputStream() val outputStream = ByteArrayOutputStream()
bitmap.compress(Bitmap.CompressFormat.JPEG, 80, outputStream) bitmap.compress(Bitmap.CompressFormat.JPEG, 70, outputStream)
return Base64.encodeToString(outputStream.toByteArray(), Base64.NO_WRAP) return Base64.encodeToString(outputStream.toByteArray(), Base64.NO_WRAP)
} }
fun kirimKeN8n( object LocationUtils {
context: ComponentActivity, private const val KAMPUS_LATITUDE = -6.2576
latitude: Double, private const val KAMPUS_LONGITUDE = 106.9746
longitude: Double, private const val MAX_RADIUS_METERS = 100.0 // Radius 100 meter
foto: Bitmap
) { fun isWithinCampusRadius(lat: Double, lon: Double): Boolean {
val earthRadius = 6371000.0 // in meters
val dLat = Math.toRadians(lat - KAMPUS_LATITUDE)
val dLon = Math.toRadians(lon - KAMPUS_LONGITUDE)
val lat1 = Math.toRadians(KAMPUS_LATITUDE)
val lat2 = Math.toRadians(lat)
val a = sin(dLat / 2) * sin(dLat / 2) +
sin(dLon / 2) * sin(dLon / 2) * cos(lat1) * cos(lat2)
val c = 2 * atan2(sqrt(a), sqrt(1 - a))
return (earthRadius * c) <= MAX_RADIUS_METERS
}
}
/* ================= MAIN ACTIVITY ================= */
class MainActivity : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
enableEdgeToEdge()
setContent {
SistemAkademikTheme {
AppNavigation(activity = this)
}
}
}
}
/* ================= NAVIGATION ================= */
@Composable
fun AppNavigation(activity: ComponentActivity) {
val navController = rememberNavController()
NavHost(navController = navController, startDestination = "login") {
composable("login") {
LoginScreen(navController = navController)
}
composable(
route = "menu/{npm}/{nama}",
arguments = listOf(
navArgument("npm") { type = NavType.StringType },
navArgument("nama") { type = NavType.StringType }
)
) { backStackEntry ->
val npm = backStackEntry.arguments?.getString("npm") ?: "N/A"
val nama = backStackEntry.arguments?.getString("nama") ?: "N/A"
MenuScreen(navController = navController, npm = npm, nama = nama)
}
composable(
route = "absensi/{npm}/{nama}",
arguments = listOf(
navArgument("npm") { type = NavType.StringType },
navArgument("nama") { type = NavType.StringType }
)
) { backStackEntry ->
val npm = backStackEntry.arguments?.getString("npm") ?: "N/A"
val nama = backStackEntry.arguments?.getString("nama") ?: "N/A"
AbsensiScreen(
activity = activity,
navController = navController,
npm = npm,
nama = nama
)
}
composable(
route = "history/{npm}",
arguments = listOf(navArgument("npm") { type = NavType.StringType })
) { backStackEntry ->
val npm = backStackEntry.arguments?.getString("npm") ?: "N/A"
HistoryScreen(navController = navController, npm = npm, activity = activity)
}
}
}
/* ================= UI SCREENS ================= */
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun LoginScreen(navController: NavController) {
var npm by remember { mutableStateOf("") }
var nama by remember { mutableStateOf("") }
Scaffold(
topBar = {
CenterAlignedTopAppBar(
title = { Text("Login Absensi", color = Color.White, fontSize = 18.sp) },
colors = TopAppBarDefaults.centerAlignedTopAppBarColors(containerColor = Color(0xFF2E7D32))
)
}
) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.fillMaxSize()
.background(Color(0xFFF8FBF8))
.padding(horizontal = 32.dp),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Icon(Icons.Default.Person, contentDescription = null, modifier = Modifier.size(100.dp), tint = Color(0xFF2E7D32))
Spacer(modifier = Modifier.height(40.dp))
OutlinedTextField(
value = npm,
onValueChange = { npm = it },
label = { Text("NPM") },
singleLine = true,
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Number, imeAction = ImeAction.Next),
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(16.dp))
OutlinedTextField(
value = nama,
onValueChange = { nama = it },
label = { Text("Nama Lengkap") },
singleLine = true,
keyboardOptions = KeyboardOptions(imeAction = ImeAction.Done),
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(24.dp))
Button(
onClick = {
if (npm.isNotBlank() && nama.isNotBlank()) {
navController.navigate("menu/$npm/$nama")
}
},
modifier = Modifier.fillMaxWidth().height(50.dp),
enabled = npm.isNotBlank() && nama.isNotBlank(),
colors = ButtonDefaults.buttonColors(containerColor = Color(0xFF1B5E20)),
shape = RoundedCornerShape(10.dp)
) {
Text("LOGIN", fontWeight = FontWeight.Bold, color = Color.White)
}
}
}
}
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun MenuScreen(navController: NavController, npm: String, nama: String) {
Scaffold(
topBar = {
CenterAlignedTopAppBar(
title = { Text("Menu Utama", color = Color.White, fontSize = 18.sp) },
colors = TopAppBarDefaults.centerAlignedTopAppBarColors(containerColor = Color(0xFF2E7D32)),
navigationIcon = {
IconButton(onClick = { navController.popBackStack() }) {
Icon(Icons.Default.ArrowBack, contentDescription = "Logout", tint = Color.White)
}
}
)
}
) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.fillMaxSize()
.background(Color(0xFFF8FBF8))
.padding(16.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text("Selamat Datang,", style = MaterialTheme.typography.bodyLarge)
Text("$nama ($npm)", fontWeight = FontWeight.Bold, fontSize = 20.sp)
Spacer(modifier = Modifier.height(32.dp))
// Tombol Absensi
Button(
onClick = { navController.navigate("absensi/$npm/$nama") },
modifier = Modifier
.fillMaxWidth()
.height(60.dp),
shape = RoundedCornerShape(12.dp)
) {
Icon(Icons.Filled.CameraAlt, contentDescription = null, modifier = Modifier.size(24.dp))
Spacer(modifier = Modifier.width(8.dp))
Text("Absen Kehadiran", fontSize = 16.sp)
}
Spacer(modifier = Modifier.height(16.dp))
// Tombol Riwayat
Button(
onClick = { navController.navigate("history/$npm") },
modifier = Modifier
.fillMaxWidth()
.height(60.dp),
shape = RoundedCornerShape(12.dp)
) {
Icon(Icons.Default.History, contentDescription = null, modifier = Modifier.size(24.dp))
Spacer(modifier = Modifier.width(8.dp))
Text("Riwayat Absensi", fontSize = 16.sp)
}
}
}
}
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun AbsensiScreen(activity: ComponentActivity, navController: NavController, npm: String, nama: String) {
val context = LocalContext.current
var lokasiStatus by remember { mutableStateOf("Mengecek Lokasi...") }
var locationAvailable by remember { mutableStateOf(false) }
var foto by remember { mutableStateOf<Bitmap?>(null) }
var isUploading by remember { mutableStateOf(false) }
var latitude by remember { mutableStateOf<Double?>(null) }
var longitude by remember { mutableStateOf<Double?>(null) }
var matakuliah by remember { mutableStateOf("") }
val fusedLocationClient = remember { LocationServices.getFusedLocationProviderClient(context) }
val cameraLauncher = rememberLauncherForActivityResult(
ActivityResultContracts.StartActivityForResult()
) { result ->
if (result.resultCode == Activity.RESULT_OK) {
val bitmap = result.data?.extras?.get("data") as? Bitmap
if (bitmap != null) foto = bitmap
}
}
val requestPermissionLauncher = rememberLauncherForActivityResult(
ActivityResultContracts.RequestMultiplePermissions()
) { permissions ->
if (permissions[Manifest.permission.ACCESS_FINE_LOCATION] == true) {
if (ActivityCompat.checkSelfPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) {
fusedLocationClient.getCurrentLocation(Priority.PRIORITY_HIGH_ACCURACY, null)
.addOnSuccessListener { loc ->
if (loc != null) {
latitude = loc.latitude
longitude = loc.longitude
locationAvailable = true
lokasiStatus = if (LocationUtils.isWithinCampusRadius(loc.latitude, loc.longitude)) {
"Di Dalam Area Kampus"
} else {
"Di Luar Area Kampus"
}
} else {
lokasiStatus = "Gagal Mendapatkan Lokasi"
locationAvailable = false
}
}.addOnFailureListener {
lokasiStatus = "Gagal Mendapatkan Lokasi"
locationAvailable = false
}
}
} else {
lokasiStatus = "Izin lokasi ditolak."
locationAvailable = false
}
}
LaunchedEffect(Unit) {
requestPermissionLauncher.launch(
arrayOf(Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.CAMERA)
)
}
Scaffold(
topBar = {
CenterAlignedTopAppBar(
title = { Text("Absen Kehadiran", color = Color.White, fontSize = 18.sp) },
navigationIcon = {
IconButton(onClick = { navController.popBackStack() }) {
Icon(Icons.Default.ArrowBack, contentDescription = "Kembali", tint = Color.White)
}
},
colors = TopAppBarDefaults.centerAlignedTopAppBarColors(containerColor = Color(0xFF2E7D32))
)
}
) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.fillMaxSize()
.background(Color(0xFFF8FBF8))
.padding(horizontal = 16.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
Spacer(modifier = Modifier.height(16.dp))
Text("Selamat Datang,", style = MaterialTheme.typography.bodyLarge)
Text("$nama ($npm)", fontWeight = FontWeight.Bold, fontSize = 18.sp)
Spacer(modifier = Modifier.height(16.dp))
OutlinedTextField(
value = matakuliah,
onValueChange = { matakuliah = it },
label = { Text("Mata Kuliah") },
singleLine = true,
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(16.dp))
Row(
modifier = Modifier.padding(top = 8.dp),
verticalAlignment = Alignment.CenterVertically
) {
Text(
text = "Status Lokasi: $lokasiStatus ",
style = MaterialTheme.typography.bodyMedium,
color = if (locationAvailable) Color(0xFF2E7D32) else Color.Red
)
if (locationAvailable) {
Icon(Icons.Default.CheckCircle, "OK", tint = Color(0xFF2E7D32), modifier = Modifier.size(16.dp))
} else {
Icon(Icons.Default.Close, "Error", tint = Color.Red, modifier = Modifier.size(16.dp))
}
}
Spacer(modifier = Modifier.height(20.dp))
Text("Ambil Foto Selfie", fontWeight = FontWeight.SemiBold, fontSize = 16.sp)
Spacer(modifier = Modifier.height(16.dp))
Box(
modifier = Modifier
.size(220.dp)
.clip(CircleShape)
.background(Color(0xFFE0E0E0)),
contentAlignment = Alignment.Center
) {
if (foto != null) {
Image(
bitmap = foto!!.asImageBitmap(),
contentDescription = null,
modifier = Modifier.fillMaxSize(),
contentScale = ContentScale.Crop
)
} else {
Icon(Icons.Default.Person, null, modifier = Modifier.size(80.dp), tint = Color.White)
}
}
Spacer(modifier = Modifier.height(40.dp))
Button(
onClick = {
val intent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
cameraLauncher.launch(intent)
},
modifier = Modifier.fillMaxWidth(0.7f).height(50.dp),
colors = ButtonDefaults.buttonColors(containerColor = Color(0xFF4CAF50)),
shape = RoundedCornerShape(10.dp)
) {
Text("AMBIL FOTO", fontWeight = FontWeight.Bold, color = Color.White)
}
Spacer(modifier = Modifier.height(16.dp))
Button(
onClick = {
isUploading = true
if (latitude != null && longitude != null && foto != null) {
kirimKeServer(activity, npm, nama, matakuliah, latitude!!, longitude!!, foto!!) {
isUploading = false
if (it) navController.popBackStack()
}
} else {
isUploading = false
Toast.makeText(context, "Lokasi atau foto belum siap.", Toast.LENGTH_LONG).show()
}
},
modifier = Modifier.fillMaxWidth(0.7f).height(50.dp),
enabled = foto != null && locationAvailable && matakuliah.isNotBlank() && !isUploading,
colors = ButtonDefaults.buttonColors(containerColor = Color(0xFF1B5E20)),
shape = RoundedCornerShape(10.dp)
) {
Text("KIRIM ABSENSI", fontWeight = FontWeight.Bold)
}
if (isUploading) {
Spacer(modifier = Modifier.height(16.dp))
CircularProgressIndicator(color = Color(0xFF2E7D32))
}
}
}
}
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun HistoryScreen(navController: NavController, npm: String, activity: Activity) {
var historyList by remember { mutableStateOf<List<AttendanceRecord>>(emptyList()) }
var isLoading by remember { mutableStateOf(true) }
LaunchedEffect(npm) {
fetchHistoryFromServer(npm, activity) {
historyList = it
isLoading = false
}
}
Scaffold(
topBar = {
CenterAlignedTopAppBar(
title = { Text("Riwayat Absensi", color = Color.White) },
navigationIcon = {
IconButton(onClick = { navController.popBackStack() }) {
Icon(Icons.Default.ArrowBack, "Kembali", tint = Color.White)
}
},
colors = TopAppBarDefaults.centerAlignedTopAppBarColors(Color(0xFF2E7D32))
)
}
) { innerPadding ->
if (isLoading) {
Box(modifier = Modifier.fillMaxSize(), contentAlignment = Alignment.Center) {
CircularProgressIndicator()
}
} else if (historyList.isEmpty()) {
Box(modifier = Modifier.fillMaxSize(), contentAlignment = Alignment.Center) {
Text("Tidak ada riwayat absensi.")
}
} else {
LazyColumn(
modifier = Modifier
.padding(innerPadding)
.fillMaxSize()
.padding(16.dp)
) {
items(historyList) { record ->
val date = SimpleDateFormat("dd MMMM yyyy, HH:mm", Locale.getDefault()).format(Date(record.timestamp))
Card(
modifier = Modifier
.fillMaxWidth()
.padding(vertical = 4.dp),
elevation = CardDefaults.cardElevation(2.dp)
) {
Row(
modifier = Modifier
.padding(16.dp)
.fillMaxWidth(),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween
) {
Text(text = date, fontWeight = FontWeight.Medium)
Text(
text = record.status,
color = if (record.status == "Sukses") Color(0xFF2E7D32) else Color.Red,
fontWeight = FontWeight.Bold
)
}
}
}
}
}
}
}
/* ================= NETWORK LOGIC ================= */
fun kirimKeServer(activity: Activity, npm: String, nama: String, matakuliah: String, lat: Double, lon: Double, img: Bitmap, onFinish: (Boolean) -> Unit) {
thread { thread {
var isSuccess = false
try { try {
val url = URL("https://n8n.lab.ubharajaya.ac.id/webhook/23c6993d-1792-48fb-ad1c-ffc78a3e6254") val url = URL("https://n8n.lab.ubharajaya.ac.id/webhook/23c6993d-1792-48fb-ad1c-ffc78a3e6254")
// test URL val url = URL("https://n8n.lab.ubharajaya.ac.id/webhook-test/23c6993d-1792-48fb-ad1c-ffc78a3e6254")
val conn = url.openConnection() as HttpURLConnection val conn = url.openConnection() as HttpURLConnection
conn.requestMethod = "POST" conn.requestMethod = "POST"
conn.setRequestProperty("Content-Type", "application/json") conn.setRequestProperty("Content-Type", "application/json")
conn.doOutput = true conn.doOutput = true
val json = JSONObject().apply { val json = JSONObject().apply {
put("npm", "12345") put("npm", npm)
put("nama","Arif R D") put("nama", nama)
put("latitude", latitude) put("mata_kuliah", matakuliah) // FIX: Changed from "matakuliah" to "mata_kuliah"
put("longitude", longitude) put("latitude", lat)
put("longitude", lon)
put("timestamp", System.currentTimeMillis()) put("timestamp", System.currentTimeMillis())
put("foto_base64", bitmapToBase64(foto)) put("foto_base64", bitmapToBase64(img))
} }
conn.outputStream.use { conn.outputStream.write(json.toString().toByteArray())
it.write(json.toString().toByteArray())
val code = conn.responseCode
isSuccess = code == 200
activity.runOnUiThread {
Toast.makeText(activity, if(isSuccess) "Absensi Sukses!" else "Absensi Gagal: Server Error $code", Toast.LENGTH_LONG).show()
onFinish(isSuccess)
} }
} catch (e: Exception) {
val responseCode = conn.responseCode activity.runOnUiThread {
Toast.makeText(activity, "Error: Gagal terhubung ke server.", Toast.LENGTH_SHORT).show()
context.runOnUiThread { onFinish(false)
Toast.makeText(
context,
if (responseCode == 200)
"Absensi diterima server"
else
"Absensi ditolak server",
Toast.LENGTH_SHORT
).show()
}
conn.disconnect()
} catch (_: Exception) {
context.runOnUiThread {
Toast.makeText(
context,
"Gagal kirim ke server",
Toast.LENGTH_SHORT
).show()
} }
} }
} }
} }
/* ================= ACTIVITY ================= */ fun fetchHistoryFromServer(npm: String, activity: Activity, onResult: (List<AttendanceRecord>) -> Unit) {
thread {
try {
// PENTING: URL ini hanya contoh. Ganti dengan URL API Anda untuk mengambil riwayat.
val url = URL("https://n8n.lab.ubharajaya.ac.id/webhook-test/riwayat-absensi?npm=$npm")
val conn = url.openConnection() as HttpURLConnection
conn.requestMethod = "GET"
class MainActivity : ComponentActivity() { if (conn.responseCode == 200) {
val response = conn.inputStream.bufferedReader().use { it.readText() }
override fun onCreate(savedInstanceState: Bundle?) { val jsonArray = JSONArray(response)
super.onCreate(savedInstanceState) val records = mutableListOf<AttendanceRecord>()
enableEdgeToEdge() for (i in 0 until jsonArray.length()) {
val jsonObject = jsonArray.getJSONObject(i)
setContent { records.add(
SistemAkademikTheme { AttendanceRecord(
Scaffold(modifier = Modifier.fillMaxSize()) { innerPadding -> timestamp = jsonObject.getLong("timestamp"),
AbsensiScreen( status = jsonObject.getString("status")
modifier = Modifier.padding(innerPadding), )
activity = this
) )
} }
activity.runOnUiThread { onResult(records) }
} else {
activity.runOnUiThread { onResult(emptyList()) }
} }
} } catch (e: Exception) {
} activity.runOnUiThread { onResult(emptyList()) }
}
/* ================= UI ================= */
@Composable
fun AbsensiScreen(
modifier: Modifier = Modifier,
activity: ComponentActivity
) {
val context = LocalContext.current
var lokasi by remember { mutableStateOf("Koordinat: -") }
var latitude by remember { mutableStateOf<Double?>(null) }
var longitude by remember { mutableStateOf<Double?>(null) }
var foto by remember { mutableStateOf<Bitmap?>(null) }
val fusedLocationClient =
LocationServices.getFusedLocationProviderClient(context)
/* ===== Permission Lokasi ===== */
val locationPermissionLauncher =
rememberLauncherForActivityResult(
ActivityResultContracts.RequestPermission()
) { granted ->
if (granted) {
if (
ContextCompat.checkSelfPermission(
context,
Manifest.permission.ACCESS_FINE_LOCATION
) == PackageManager.PERMISSION_GRANTED
) {
fusedLocationClient.lastLocation
.addOnSuccessListener { location ->
if (location != null) {
latitude = location.latitude
longitude = location.longitude
lokasi =
"Lat: ${location.latitude}\nLon: ${location.longitude}"
} else {
lokasi = "Lokasi tidak tersedia"
}
}
.addOnFailureListener {
lokasi = "Gagal mengambil lokasi"
}
}
} else {
Toast.makeText(
context,
"Izin lokasi ditolak",
Toast.LENGTH_SHORT
).show()
}
}
/* ===== Kamera ===== */
val cameraLauncher =
rememberLauncherForActivityResult(
ActivityResultContracts.StartActivityForResult()
) { result ->
if (result.resultCode == Activity.RESULT_OK) {
val bitmap =
result.data?.extras?.getParcelable("data", Bitmap::class.java)
if (bitmap != null) {
foto = bitmap
Toast.makeText(
context,
"Foto berhasil diambil",
Toast.LENGTH_SHORT
).show()
}
}
}
val cameraPermissionLauncher =
rememberLauncherForActivityResult(
ActivityResultContracts.RequestPermission()
) { granted ->
if (granted) {
val intent =
Intent(MediaStore.ACTION_IMAGE_CAPTURE)
cameraLauncher.launch(intent)
} else {
Toast.makeText(
context,
"Izin kamera ditolak",
Toast.LENGTH_SHORT
).show()
}
}
/* ===== Request Awal ===== */
LaunchedEffect(Unit) {
locationPermissionLauncher.launch(
Manifest.permission.ACCESS_FINE_LOCATION
)
}
/* ===== UI ===== */
Column(
modifier = modifier
.fillMaxSize()
.padding(24.dp),
verticalArrangement = Arrangement.Center
) {
Text(
text = "Absensi Akademik",
style = MaterialTheme.typography.titleLarge
)
Spacer(modifier = Modifier.height(16.dp))
Text(text = lokasi)
Spacer(modifier = Modifier.height(16.dp))
Button(
onClick = {
cameraPermissionLauncher.launch(
Manifest.permission.CAMERA
)
},
modifier = Modifier.fillMaxWidth()
) {
Text("Ambil Foto")
}
Spacer(modifier = Modifier.height(12.dp))
Button(
onClick = {
if (latitude != null && longitude != null && foto != null) {
kirimKeN8n(
activity,
latitude!!,
longitude!!,
foto!!
)
} else {
Toast.makeText(
context,
"Lokasi atau foto belum lengkap",
Toast.LENGTH_SHORT
).show()
}
},
modifier = Modifier.fillMaxWidth()
) {
Text("Kirim Absensi")
} }
} }
} }

View File

@ -8,6 +8,10 @@ espressoCore = "3.7.0"
lifecycleRuntimeKtx = "2.9.4" lifecycleRuntimeKtx = "2.9.4"
activityCompose = "1.11.0" activityCompose = "1.11.0"
composeBom = "2024.09.00" composeBom = "2024.09.00"
activityComposeVersion = "1.12.2"
playServicesLocation = "21.3.0"
navigationCompose = "2.9.6"
materialIconsExtended = "1.6.0"
[libraries] [libraries]
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" } androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
@ -24,6 +28,10 @@ androidx-compose-ui-tooling-preview = { group = "androidx.compose.ui", name = "u
androidx-compose-ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" } androidx-compose-ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" }
androidx-compose-ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" } androidx-compose-ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" }
androidx-compose-material3 = { group = "androidx.compose.material3", name = "material3" } androidx-compose-material3 = { group = "androidx.compose.material3", name = "material3" }
activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activityComposeVersion" }
play-services-location = { group = "com.google.android.gms", name = "play-services-location", version.ref = "playServicesLocation" }
androidx-navigation-compose = { group = "androidx.navigation", name = "navigation-compose", version.ref = "navigationCompose" }
androidx-compose-material-icons-extended = { group = "androidx.compose.material", name = "material-icons-extended", version.ref = "materialIconsExtended" }
[plugins] [plugins]
android-application = { id = "com.android.application", version.ref = "agp" } android-application = { id = "com.android.application", version.ref = "agp" }