Compare commits

..

35 Commits

Author SHA1 Message Date
f289d0495f Delete SwiftVersion 2025-11-27 15:40:52 +07:00
7c7b9c0ec6 Add SwiftVersion 2025-11-27 15:40:27 +07:00
373c8595d7 Update README.md 2025-11-27 15:13:23 +07:00
d6dafed995 Update README.md 2025-11-13 20:19:16 +07:00
b42d846e01 Update README.md 2025-11-13 20:19:00 +07:00
2638da3f05 kasdka 2025-11-13 14:53:43 +07:00
08c24bdfdd Updated 2025-11-13 14:48:04 +07:00
d3c1410a11 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:47:37 +07:00
1ce89ad197 Updated 2025-11-13 14:47:11 +07:00
228d9f8f77 Hadi P. 2025-11-13 14:46:11 +07:00
5dffb47576 Hadi P 2025-11-13 14:45:15 +07:00
992c0d627c Hadi 2025-11-13 14:44:54 +07:00
4f1f8577bc Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:44:26 +07:00
6370eba0da Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:43:41 +07:00
c4aee9ee3c a. 2025-11-13 14:43:25 +07:00
5cccdf5368 a. 2025-11-13 14:42:12 +07:00
b73d429200 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:42:08 +07:00
89c60eb81e Hadi 2025-11-13 14:41:29 +07:00
6fcbb52b90 askd.., 2025-11-13 14:40:56 +07:00
cd156ca010 kasdka 2025-11-13 14:39:02 +07:00
4488dfd0cb Hadi 2025-11-13 14:38:11 +07:00
4281e24379 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:37:45 +07:00
5170efe60e askd.. 2025-11-13 14:37:17 +07:00
4ac1678d48 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:37:00 +07:00
357a25744c Hadi 2025-11-13 14:36:29 +07:00
cf5adbd923 askd. 2025-11-13 14:36:18 +07:00
d77a9409de Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:35:21 +07:00
a115da0783 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	README.md
2025-11-13 14:35:01 +07:00
8f9ca68257 Merge remote-tracking branch 'origin/master' 2025-11-13 14:34:53 +07:00
83fabde258 askd 2025-11-13 14:34:41 +07:00
8f24ef2640 Merge remote-tracking branch 'origin/master' 2025-11-13 14:34:33 +07:00
3e3da371dc Hadi 2025-11-13 14:34:28 +07:00
6d30bb4a88 nama 2025-11-13 14:33:21 +07:00
wafi14-art
ae1ae8cd80 Fadhlul Wafi 2025-11-13 14:33:21 +07:00
55d17abb93 Praktikum 1 2025-11-13 14:29:32 +07:00
8 changed files with 42 additions and 270 deletions

View File

@ -1,10 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="deploymentTargetSelector">
<selectionStates>
<SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" />
</SelectionState>
</selectionStates>
</component>
</project>

View File

@ -25,4 +25,12 @@ Repository ini digunakan untuk praktikum perkuliahan pemrograman mobile.
- 14. Nabila suwandira (202310715066)
- 15. Indris Alpasela (202310715200)
- 16. Raihan Ariq Muzakki (202310715297)
- 17. Dirson Ali Wardana (202310715246)
- 17. Dirson Ali Wardana (202310715246)
- 18. Dimas Hendri Pamungkas (202310715274)
- 19. Fadhlul Wafi (202310715188)
- 20. Muhammad Yusron Amrullah (202310715060)
- 21. Muhammad Fadillah (202310715213)
- 22. Hadi Guna Prakoso (202310715312)
- 23. Muhammad Rafi (202310715191)
- 24. Muhammad Rafly Al-Fathir (202310715043)
- 25. Jeremia Sebastian Marpaung (202310715096)

View File

@ -50,8 +50,6 @@ dependencies {
implementation(libs.androidx.compose.ui.graphics)
implementation(libs.androidx.compose.ui.tooling.preview)
implementation(libs.androidx.compose.material3)
implementation("com.squareup.okhttp3:okhttp:4.11.0")
testImplementation(libs.junit)
androidTestImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.espresso.core)
@ -59,4 +57,8 @@ dependencies {
androidTestImplementation(libs.androidx.compose.ui.test.junit4)
debugImplementation(libs.androidx.compose.ui.tooling)
debugImplementation(libs.androidx.compose.ui.test.manifest)
//praktikum 1
implementation("com.squareup.okhttp3:okhttp:4.11.0")
implementation("androidx.compose.material3:material3:1.1.1")
}

View File

@ -22,12 +22,6 @@
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".JalurEvakuasiActivity"
android:exported="false"/>
</application>
<uses-permission android:name="android.permission.INTERNET" />
</manifest>

View File

@ -1,47 +0,0 @@
package id.ac.ubharajaya.panicbutton
import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.foundation.Image
import androidx.compose.foundation.layout.*
import androidx.compose.material3.Button
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.res.painterResource
import androidx.compose.ui.unit.dp
class JalurEvakuasiActivity : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
JalurEvakuasiScreen(onClose = { finish() })
}
}
}
@Composable
fun JalurEvakuasiScreen(onClose: () -> Unit) {
Column(
modifier = Modifier
.fillMaxSize()
.padding(16.dp),
horizontalAlignment = Alignment.CenterHorizontally
) {
Image(
painter = painterResource(id = R.drawable.evakuasi),
contentDescription = "Jalur Evakuasi",
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(20.dp))
Button(onClick = onClose) {
Text("Close")
}
}
}

View File

@ -1,25 +1,18 @@
package id.ac.ubharajaya.panicbutton
import android.content.Intent
import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.foundation.border
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.*
import androidx.compose.foundation.text.BasicTextField
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.*
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.activity.enableEdgeToEdge
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.padding
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Button
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.platform.LocalFocusManager
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.input.TextFieldValue
import androidx.compose.ui.text.input.ImeAction
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import androidx.compose.ui.tooling.preview.Preview
import id.ac.ubharajaya.panicbutton.ui.theme.PanicButtonTheme
import okhttp3.MediaType.Companion.toMediaType
import okhttp3.OkHttpClient
import okhttp3.Request
@ -28,200 +21,32 @@ import okhttp3.RequestBody
class MainActivity : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
enableEdgeToEdge()
setContent {
MyApp(
onClose = { finish() },
onOpenEvacuation = {
startActivity(Intent(this, JalurEvakuasiActivity::class.java))
PanicButtonTheme {
Scaffold(modifier = Modifier.fillMaxSize()) { innerPadding ->
Greeting(
name = "Android",
modifier = Modifier.padding(innerPadding)
)
}
)
}
}
}
}
// Emoji untuk tiap kondisi
fun getEmoji(condition: String): String {
return when (condition) {
"Kebakaran" -> "🔥"
"Banjir" -> "⛈️"
"Tsunami" -> "🌊"
"Gunung Meletus" -> "🌋"
"Gempa Bumi" -> "🌏"
"Huru Hara/Demonstrasi" -> "👿"
"Binatang Buas" -> "🐍"
"Radiasi Nuklir" -> "☢️"
"Biohazard" -> "☣️"
else -> ""
}
}
@Composable
fun MyApp(onClose: () -> Unit, onOpenEvacuation: () -> Unit) {
val focusManager = LocalFocusManager.current
var message by remember { mutableStateOf("Klik tombol untuk mengirim notifikasi") }
var selectedConditions by remember { mutableStateOf(mutableSetOf<String>()) }
var additionalNotes by remember { mutableStateOf(TextFieldValue("")) }
fun Greeting(name: String, modifier: Modifier = Modifier) {
Text(
text = "Hello $name!",
modifier = modifier
)
}
Column(
modifier = Modifier
.fillMaxSize()
.padding(16.dp),
verticalArrangement = Arrangement.Top,
horizontalAlignment = Alignment.Start
) {
// Judul
Text(
text = "Terjadi Kondisi Darurat:",
fontSize = 26.sp,
fontWeight = FontWeight.Bold,
color = Color.Red,
modifier = Modifier.padding(bottom = 16.dp)
)
// Daftar kondisi
listOf(
"Kebakaran",
"Banjir",
"Tsunami",
"Gunung Meletus",
"Gempa Bumi",
"Huru Hara/Demonstrasi",
"Binatang Buas",
"Radiasi Nuklir",
"Biohazard",
"Lainnya"
).forEach { condition ->
val emoji = getEmoji(condition)
Row(
modifier = Modifier
.fillMaxWidth()
.clickable {
focusManager.clearFocus()
selectedConditions = selectedConditions.toMutableSet().apply {
if (contains(condition)) remove(condition) else add(condition)
}
}
.padding(vertical = 4.dp),
verticalAlignment = Alignment.CenterVertically
) {
Checkbox(
checked = selectedConditions.contains(condition),
onCheckedChange = { isChecked ->
focusManager.clearFocus()
selectedConditions = selectedConditions.toMutableSet().apply {
if (isChecked) add(condition) else remove(condition)
}
}
)
Text(
text = "$emoji $condition",
modifier = Modifier.padding(start = 8.dp)
)
}
}
// Catatan tambahan
Spacer(modifier = Modifier.height(16.dp))
Text(text = "Catatan tambahan:", fontWeight = FontWeight.Bold)
BasicTextField(
value = additionalNotes,
onValueChange = { additionalNotes = it },
modifier = Modifier
.fillMaxWidth()
.height(100.dp)
.padding(8.dp)
.border(1.dp, Color.Gray),
keyboardOptions = KeyboardOptions(imeAction = ImeAction.Done)
)
// ================================================
// Tombol Kirim + Lihat Jalur Evakuasi (Berdampingan)
// ================================================
Spacer(modifier = Modifier.height(16.dp))
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween
) {
// Tombol Kirim Laporan
Button(
onClick = {
val notes = additionalNotes.text
val conditions = selectedConditions.joinToString(", ") {
"${getEmoji(it)} $it"
}
val report = "Kondisi: $conditions\nCatatan: $notes"
sendNotification(report) { response ->
message = response
}
},
modifier = Modifier.weight(1f),
colors = ButtonDefaults.buttonColors(containerColor = Color.Red)
) {
Text(text = "Kirim Laporan", color = Color.White)
}
Spacer(modifier = Modifier.width(12.dp))
// Tombol Jalur Evakuasi
Button(
onClick = onOpenEvacuation,
modifier = Modifier.weight(1f),
colors = ButtonDefaults.buttonColors(containerColor = Color(0xFF0080FF))
) {
Text(text = "Lihat Jalur Evakuasi", color = Color.White)
}
}
Spacer(modifier = Modifier.height(16.dp))
Text(
text = "“JANGAN PANIK! SEGERA EVAKUASI\nDIRI ANDA KE TITIK KUMPUL”",
color = Color.Red,
fontSize = 15.sp
)
Spacer(modifier = Modifier.height(20.dp))
// Tombol Tutup (opsional)
Button(onClick = onClose) {
Text("Tutup")
}
Spacer(modifier = Modifier.height(16.dp))
Text(text = message)
@Preview(showBackground = true)
@Composable
fun GreetingPreview() {
PanicButtonTheme {
Greeting("Android")
}
}
// ===============================================================
// Fungsi Kirim Notifikasi ke Server ntfy.ubharajaya.ac.id
// ===============================================================
fun sendNotification(report: String, onResult: (String) -> Unit) {
val client = OkHttpClient()
val url = "https://ntfy.ubharajaya.ac.id/panic-button"
val requestBody = RequestBody.create("text/plain".toMediaType(), report)
val request = Request.Builder()
.url(url)
.addHeader("Title", "Alert")
.addHeader("Priority", "urgent")
.post(requestBody)
.build()
Thread {
try {
val response = client.newCall(request).execute()
if (response.isSuccessful) {
onResult("Notifikasi berhasil dikirim!")
} else {
onResult("Gagal mengirim notifikasi: ${response.code}")
}
} catch (e: Exception) {
onResult("Error: ${e.message}")
}
}.start()
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.8 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 269 KiB