final apk

This commit is contained in:
nuryuda 2026-01-09 16:09:32 +07:00
commit 3f5117a017
56 changed files with 1697 additions and 0 deletions

15
.gitignore vendored Normal file
View File

@ -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

3
.idea/.gitignore generated vendored Normal file
View File

@ -0,0 +1,3 @@
# Default ignored files
/shelf/
/workspace.xml

1
.idea/.name generated Normal file
View File

@ -0,0 +1 @@
Compass

6
.idea/AndroidProjectSystem.xml generated Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AndroidProjectSystem">
<option name="providerId" value="com.android.tools.idea.GradleProjectSystem" />
</component>
</project>

6
.idea/compiler.xml generated Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="CompilerConfiguration">
<bytecodeTargetLevel target="21" />
</component>
</project>

10
.idea/deploymentTargetSelector.xml generated Normal file
View File

@ -0,0 +1,10 @@
<?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>

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

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

19
.idea/gradle.xml generated Normal file
View File

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="GradleMigrationSettings" migrationVersion="1" />
<component name="GradleSettings">
<option name="linkedExternalProjectsSettings">
<GradleProjectSettings>
<option name="testRunner" value="CHOOSE_PER_TEST" />
<option name="externalProjectPath" value="$PROJECT_DIR$" />
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
<option name="modules">
<set>
<option value="$PROJECT_DIR$" />
<option value="$PROJECT_DIR$/app" />
</set>
</option>
</GradleProjectSettings>
</option>
</component>
</project>

10
.idea/migrations.xml generated Normal file
View File

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectMigrations">
<option name="MigrateToGradleLocalJavaHome">
<set>
<option value="$PROJECT_DIR$" />
</set>
</option>
</component>
</project>

9
.idea/misc.xml generated Normal file
View File

@ -0,0 +1,9 @@
<project version="4">
<component name="ExternalStorageConfigurationManager" enabled="true" />
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="jbr-21" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/build/classes" />
</component>
<component name="ProjectType">
<option name="id" value="Android" />
</component>
</project>

17
.idea/runConfigurations.xml generated Normal file
View File

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="RunConfigurationProducerService">
<option name="ignoredProducers">
<set>
<option value="com.intellij.execution.junit.AbstractAllInDirectoryConfigurationProducer" />
<option value="com.intellij.execution.junit.AllInPackageConfigurationProducer" />
<option value="com.intellij.execution.junit.PatternConfigurationProducer" />
<option value="com.intellij.execution.junit.TestInClassConfigurationProducer" />
<option value="com.intellij.execution.junit.UniqueIdConfigurationProducer" />
<option value="com.intellij.execution.junit.testDiscovery.JUnitTestDiscoveryConfigurationProducer" />
<option value="org.jetbrains.kotlin.idea.junit.KotlinJUnitRunConfigurationProducer" />
<option value="org.jetbrains.kotlin.idea.junit.KotlinPatternConfigurationProducer" />
</set>
</option>
</component>
</project>

BIN
Compass.zip Normal file

Binary file not shown.

1
app/.gitignore vendored Normal file
View File

@ -0,0 +1 @@
/build

51
app/build.gradle.kts Normal file
View File

@ -0,0 +1,51 @@
plugins {
alias(libs.plugins.android.application)
alias(libs.plugins.kotlin.android)
id("com.google.gms.google-services")
}
android {
namespace = "com.kelompok6.compass"
compileSdk = 36
defaultConfig {
applicationId = "com.kelompok6.compass"
minSdk = 24
targetSdk = 36
versionCode = 1
versionName = "1.0"
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
isMinifyEnabled = false
proguardFiles(
getDefaultProguardFile("proguard-android-optimize.txt"),
"proguard-rules.pro"
)
}
}
compileOptions {
sourceCompatibility = JavaVersion.VERSION_11
targetCompatibility = JavaVersion.VERSION_11
}
kotlinOptions {
jvmTarget = "11"
}
}
dependencies {
implementation(libs.androidx.core.ktx)
implementation(libs.androidx.appcompat)
implementation(libs.material)
testImplementation(libs.junit)
androidTestImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.espresso.core)
implementation(platform("com.google.firebase:firebase-bom:32.7.0"))
implementation("com.google.firebase:firebase-firestore-ktx")
implementation("com.google.android.gms:play-services-location:21.1.0")
}

29
app/google-services.json Normal file
View File

@ -0,0 +1,29 @@
{
"project_info": {
"project_number": "752769663172",
"project_id": "compass-app-7c3d5",
"storage_bucket": "compass-app-7c3d5.firebasestorage.app"
},
"client": [
{
"client_info": {
"mobilesdk_app_id": "1:752769663172:android:c610b20aa2d6d5cbebc991",
"android_client_info": {
"package_name": "com.kelompok6.compass"
}
},
"oauth_client": [],
"api_key": [
{
"current_key": "AIzaSyCINTsHh1DOoSRV9vKtrLPYeRZ4o0Bt0jc"
}
],
"services": {
"appinvite_service": {
"other_platform_oauth_client": []
}
}
}
],
"configuration_version": "1"
}

21
app/proguard-rules.pro vendored Normal file
View File

@ -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

View File

@ -0,0 +1,24 @@
package com.kelompok6.compass
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.kelompok6.compass", appContext.packageName)
}
}

View File

@ -0,0 +1,28 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-feature android:name="android.hardware.sensor.accelerometer" android:required="true" />
<uses-feature android:name="android.hardware.sensor.magnetometer" android:required="true" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="Compass App"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.Material3.DayNight.NoActionBar">
<activity
android:name=".MainActivity"
android:exported="true"
android:screenOrientation="portrait">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>

View File

@ -0,0 +1,318 @@
package com.kelompok6.compass
import android.Manifest
import android.content.Context
import android.content.pm.PackageManager
import android.hardware.Sensor
import android.hardware.SensorEvent
import android.hardware.SensorEventListener
import android.hardware.SensorManager
import android.location.Location
import android.os.Bundle
import android.view.animation.Animation
import android.view.animation.RotateAnimation
import android.widget.*
import androidx.appcompat.app.AlertDialog
import androidx.appcompat.app.AppCompatActivity
import androidx.core.app.ActivityCompat
import androidx.core.content.ContextCompat
import com.google.android.gms.location.FusedLocationProviderClient
import com.google.android.gms.location.LocationServices
import com.google.android.material.floatingactionbutton.FloatingActionButton
import com.google.firebase.firestore.FirebaseFirestore
import com.google.firebase.firestore.Query
import kotlin.math.roundToInt
class MainActivity : AppCompatActivity(), SensorEventListener {
private lateinit var compassImage: ImageView
private lateinit var degreeText: TextView
private lateinit var directionText: TextView
private lateinit var fabSaveLocation: FloatingActionButton
private lateinit var locationsListView: ListView
private lateinit var locationAdapter: ArrayAdapter<String>
private lateinit var sensorManager: SensorManager
private var magnetometer: Sensor? = null
private var accelerometer: Sensor? = null
private val accelerometerReading = FloatArray(3)
private val magnetometerReading = FloatArray(3)
private val rotationMatrix = FloatArray(9)
private val orientationAngles = FloatArray(3)
private var currentDegree = 0f
private lateinit var fusedLocationClient: FusedLocationProviderClient
private var currentLocation: Location? = null
private val db = FirebaseFirestore.getInstance()
private val savedLocations = mutableListOf<SavedLocation>()
private val locationNames = mutableListOf<String>()
companion object {
private const val LOCATION_PERMISSION_REQUEST = 100
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
initViews()
initSensors()
initLocation()
setupListeners()
loadSavedLocations()
}
private fun initViews() {
compassImage = findViewById(R.id.compassImage)
degreeText = findViewById(R.id.degreeText)
directionText = findViewById(R.id.directionText)
fabSaveLocation = findViewById(R.id.fabSaveLocation)
locationsListView = findViewById(R.id.locationsListView)
locationAdapter = ArrayAdapter(this, android.R.layout.simple_list_item_1, locationNames)
locationsListView.adapter = locationAdapter
}
private fun initSensors() {
sensorManager = getSystemService(Context.SENSOR_SERVICE) as SensorManager
magnetometer = sensorManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD)
accelerometer = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER)
}
private fun initLocation() {
fusedLocationClient = LocationServices.getFusedLocationProviderClient(this)
requestLocationPermission()
}
private fun setupListeners() {
fabSaveLocation.setOnClickListener {
showSaveLocationDialog()
}
locationsListView.setOnItemClickListener { _, _, position, _ ->
val location = savedLocations[position]
showLocationDetails(location)
}
locationsListView.setOnItemLongClickListener { _, _, position, _ ->
showDeleteConfirmation(position)
true
}
}
private fun requestLocationPermission() {
if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION)
!= PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(
this,
arrayOf(Manifest.permission.ACCESS_FINE_LOCATION),
LOCATION_PERMISSION_REQUEST
)
} else {
getCurrentLocation()
}
}
private fun getCurrentLocation() {
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION)
== PackageManager.PERMISSION_GRANTED) {
fusedLocationClient.lastLocation.addOnSuccessListener { location ->
currentLocation = location
}
}
}
private fun showSaveLocationDialog() {
if (currentLocation == null) {
Toast.makeText(this, "Menunggu lokasi...", Toast.LENGTH_SHORT).show()
getCurrentLocation()
return
}
val input = EditText(this)
input.hint = "Nama lokasi (contoh: Rumah, Kantor)"
AlertDialog.Builder(this)
.setTitle("Simpan Lokasi")
.setView(input)
.setPositiveButton("Simpan") { _, _ ->
val name = input.text.toString()
if (name.isNotEmpty()) {
saveLocation(name)
} else {
Toast.makeText(this, "Nama tidak boleh kosong", Toast.LENGTH_SHORT).show()
}
}
.setNegativeButton("Batal", null)
.show()
}
private fun saveLocation(name: String) {
val location = currentLocation ?: return
val savedLocation = hashMapOf(
"name" to name,
"latitude" to location.latitude,
"longitude" to location.longitude,
"timestamp" to System.currentTimeMillis()
)
db.collection("locations")
.add(savedLocation)
.addOnSuccessListener {
Toast.makeText(this, "Lokasi tersimpan!", Toast.LENGTH_SHORT).show()
loadSavedLocations()
}
.addOnFailureListener {
Toast.makeText(this, "Gagal menyimpan: ${it.message}", Toast.LENGTH_SHORT).show()
}
}
private fun loadSavedLocations() {
db.collection("locations")
.orderBy("timestamp", Query.Direction.DESCENDING)
.get()
.addOnSuccessListener { documents ->
savedLocations.clear()
locationNames.clear()
for (document in documents) {
val location = document.toObject(SavedLocation::class.java).copy(id = document.id)
savedLocations.add(location)
locationNames.add("📍 ${location.name}")
}
locationAdapter.notifyDataSetChanged()
}
}
private fun showLocationDetails(location: SavedLocation) {
val message = """
Nama: ${location.name}
Koordinat:
Latitude: ${location.latitude}
Longitude: ${location.longitude}
""".trimIndent()
AlertDialog.Builder(this)
.setTitle("Detail Lokasi")
.setMessage(message)
.setPositiveButton("OK", null)
.show()
}
private fun showDeleteConfirmation(position: Int) {
val location = savedLocations[position]
AlertDialog.Builder(this)
.setTitle("Hapus Lokasi?")
.setMessage("Yakin ingin menghapus '${location.name}'?")
.setPositiveButton("Hapus") { _, _ ->
deleteLocation(location.id, position)
}
.setNegativeButton("Batal", null)
.show()
}
private fun deleteLocation(id: String, position: Int) {
db.collection("locations").document(id)
.delete()
.addOnSuccessListener {
savedLocations.removeAt(position)
locationNames.removeAt(position)
locationAdapter.notifyDataSetChanged()
Toast.makeText(this, "Lokasi dihapus", Toast.LENGTH_SHORT).show()
}
.addOnFailureListener {
Toast.makeText(this, "Gagal menghapus", Toast.LENGTH_SHORT).show()
}
}
override fun onSensorChanged(event: SensorEvent) {
if (event.sensor.type == Sensor.TYPE_ACCELEROMETER) {
System.arraycopy(event.values, 0, accelerometerReading, 0, accelerometerReading.size)
} else if (event.sensor.type == Sensor.TYPE_MAGNETIC_FIELD) {
System.arraycopy(event.values, 0, magnetometerReading, 0, magnetometerReading.size)
}
updateOrientation()
}
private fun updateOrientation() {
SensorManager.getRotationMatrix(
rotationMatrix, null,
accelerometerReading, magnetometerReading
)
SensorManager.getOrientation(rotationMatrix, orientationAngles)
var azimuth = Math.toDegrees(orientationAngles[0].toDouble()).toFloat()
azimuth = (azimuth + 360) % 360
degreeText.text = "${azimuth.roundToInt()}°"
directionText.text = getDirection(azimuth)
val rotateAnimation = RotateAnimation(
currentDegree,
-azimuth,
Animation.RELATIVE_TO_SELF, 0.5f,
Animation.RELATIVE_TO_SELF, 0.5f
)
rotateAnimation.duration = 250
rotateAnimation.fillAfter = true
compassImage.startAnimation(rotateAnimation)
currentDegree = -azimuth
}
private fun getDirection(degree: Float): String {
return when {
degree >= 337.5 || degree < 22.5 -> "Utara (N)"
degree >= 22.5 && degree < 67.5 -> "Timur Laut (NE)"
degree >= 67.5 && degree < 112.5 -> "Timur (E)"
degree >= 112.5 && degree < 157.5 -> "Tenggara (SE)"
degree >= 157.5 && degree < 202.5 -> "Selatan (S)"
degree >= 202.5 && degree < 247.5 -> "Barat Daya (SW)"
degree >= 247.5 && degree < 292.5 -> "Barat (W)"
degree >= 292.5 && degree < 337.5 -> "Barat Laut (NW)"
else -> "Tidak Diketahui"
}
}
override fun onAccuracyChanged(sensor: Sensor?, accuracy: Int) {
// Not needed
}
override fun onResume() {
super.onResume()
magnetometer?.also {
sensorManager.registerListener(this, it, SensorManager.SENSOR_DELAY_GAME)
}
accelerometer?.also {
sensorManager.registerListener(this, it, SensorManager.SENSOR_DELAY_GAME)
}
}
override fun onPause() {
super.onPause()
sensorManager.unregisterListener(this)
}
override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<out String>,
grantResults: IntArray
) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
if (requestCode == LOCATION_PERMISSION_REQUEST) {
if (grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
getCurrentLocation()
}
}
}
}

View File

@ -0,0 +1,9 @@
package com.kelompok6.compass
data class SavedLocation(
val id: String = "",
val name: String = "",
val latitude: Double = 0.0,
val longitude: Double = 0.0,
val timestamp: Long = System.currentTimeMillis()
)

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="oval">
<solid android:color="#1a1a2e" />
<stroke
android:width="3dp"
android:color="#e94560" />
</shape>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="oval">
<solid android:color="#000000" />
<stroke
android:width="2dp"
android:color="#666666" />
</shape>

View File

@ -0,0 +1,54 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="200dp"
android:height="200dp"
android:viewportWidth="200"
android:viewportHeight="200">
<!-- Outer Circle -->
<path
android:pathData="M100,100m-95,0a95,95 0,1 1,190 0a95,95 0,1 1,-190 0"
android:strokeWidth="4"
android:strokeColor="#e94560"
android:fillColor="#16213e"/>
<!-- Compass Rose Lines -->
<path
android:pathData="M100,20 L100,40"
android:strokeWidth="3"
android:strokeColor="#ffffff"/>
<path
android:pathData="M100,160 L100,180"
android:strokeWidth="3"
android:strokeColor="#ffffff"/>
<path
android:pathData="M20,100 L40,100"
android:strokeWidth="3"
android:strokeColor="#ffffff"/>
<path
android:pathData="M160,100 L180,100"
android:strokeWidth="3"
android:strokeColor="#ffffff"/>
<!-- North Marker N -->
<path
android:pathData="M95,10 L95,18 L100,13 L105,18 L105,10"
android:fillColor="#e94560"/>
<!-- Needle - North (Red) -->
<path
android:pathData="M100,100 L90,50 L100,30 L110,50 Z"
android:fillColor="#e94560"/>
<!-- Needle - South (White) -->
<path
android:pathData="M100,100 L90,150 L100,170 L110,150 Z"
android:fillColor="#ffffff"/>
<!-- Center Circle -->
<path
android:pathData="M100,100m-8,0a8,8 0,1 1,16 0a8,8 0,1 1,-16 0"
android:fillColor="#0f3460"
android:strokeWidth="2"
android:strokeColor="#e94560"/>
</vector>

View File

@ -0,0 +1,131 @@
<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="320dp"
android:height="320dp"
android:viewportWidth="320"
android:viewportHeight="320">
<!-- Outer Ring Circle -->
<path
android:fillColor="none"
android:strokeColor="#ffffff"
android:strokeWidth="2"
android:pathData="M 160,10 A 150,150 0 0,1 160,310 A 150,150 0 0,1 160,10" />
<!-- Inner Ring Circle -->
<path
android:fillColor="none"
android:strokeColor="#666666"
android:strokeWidth="1"
android:pathData="M 160,20 A 140,140 0 0,1 160,300 A 140,140 0 0,1 160,20" />
<!-- Major Tick Marks (setiap 10 derajat) -->
<!-- 0° (Utara) -->
<line
android:startX="160"
android:startY="15"
android:endX="160"
android:endY="35"
android:strokeColor="#ffffff"
android:strokeWidth="2" />
<!-- 30° -->
<line
android:startX="235"
android:startY="33"
android:endX="250"
android:endY="45"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 60° -->
<line
android:startX="285"
android:startY="85"
android:endX="297"
android:endY="97"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 90° (Timur) -->
<line
android:startX="305"
android:startY="160"
android:endX="285"
android:endY="160"
android:strokeColor="#ffffff"
android:strokeWidth="2" />
<!-- 120° -->
<line
android:startX="297"
android:startY="223"
android:endX="285"
android:endY="235"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 150° -->
<line
android:startX="250"
android:startY="275"
android:endX="235"
android:endY="287"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 180° (Selatan) -->
<line
android:startX="160"
android:startY="305"
android:endX="160"
android:endY="285"
android:strokeColor="#ffffff"
android:strokeWidth="2" />
<!-- 210° -->
<line
android:startX="85"
android:startY="275"
android:endX="70"
android:endY="287"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 240° -->
<line
android:startX="35"
android:startY="223"
android:endX="23"
android:endY="235"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 270° (Barat) -->
<line
android:startX="15"
android:startY="160"
android:endX="35"
android:endY="160"
android:strokeColor="#ffffff"
android:strokeWidth="2" />
<!-- 300° -->
<line
android:startX="23"
android:startY="97"
android:endX="35"
android:endY="85"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
<!-- 330° -->
<line
android:startX="70"
android:startY="45"
android:endX="85"
android:endY="33"
android:strokeColor="#ffffff"
android:strokeWidth="1.5" />
</vector>

View File

@ -0,0 +1,27 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="800dp"
android:height="800dp"
android:viewportWidth="36"
android:viewportHeight="36">
<path
android:pathData="M18,0L0,5v29l18,2l18,-2V5z"
android:fillColor="#000000"/>
<path
android:pathData="M18,36l18,-2V5L18,0z"
android:fillColor="#292F33"/>
<path
android:pathData="M22.454,14.507v3.407l4.229,0.612L26.683,15.22zM29.454,15.688v3.239l3.299,0.478v-3.161zM18,13.756v3.513l1.683,0.244L19.683,14.04zM36,16.792l-0.539,-0.091v3.096l0.539,0.078z"
android:fillColor="#FFD983"/>
<path
android:pathData="M0,16.792v3.083l0.539,-0.078v-3.096zM16.317,14.04v3.473L18,17.269v-3.513zM3.247,16.244v3.161l3.299,-0.478v-3.239zM9.317,15.22v3.306l4.229,-0.612v-3.407z"
android:fillColor="#FFAC33"/>
<path
android:pathData="M21.389,15.131v-0.042c0,-0.421 -0.143,-0.763 -0.32,-0.763c-0.177,0 -0.32,0.342 -0.32,0.763v0.042c-0.208,0.217 -0.355,0.621 -0.355,1.103c0,0.513 0.162,0.949 0.393,1.152c0.064,0.195 0.163,0.33 0.282,0.33s0.218,-0.135 0.282,-0.33c0.231,-0.203 0.393,-0.639 0.393,-1.152c-0.001,-0.482 -0.147,-0.886 -0.355,-1.103zM28.388,16.2v-0.042c0,-0.421 -0.143,-0.763 -0.32,-0.763c-0.177,0 -0.32,0.342 -0.32,0.763v0.042c-0.208,0.217 -0.355,0.621 -0.355,1.103c0,0.513 0.162,0.949 0.393,1.152c0.064,0.195 0.163,0.33 0.282,0.33s0.218,-0.135 0.282,-0.33c0.231,-0.203 0.393,-0.639 0.393,-1.152c0,-0.481 -0.147,-0.885 -0.355,-1.103zM34.405,17.23v-0.039c0,-0.393 -0.134,-0.712 -0.299,-0.712c-0.165,0 -0.299,0.319 -0.299,0.712v0.039c-0.194,0.203 -0.331,0.58 -0.331,1.03c0,0.479 0.151,0.886 0.367,1.076c0.059,0.182 0.152,0.308 0.263,0.308s0.203,-0.126 0.263,-0.308c0.215,-0.189 0.367,-0.597 0.367,-1.076c0,-0.45 -0.136,-0.827 -0.331,-1.03z"
android:fillColor="#FFD983"/>
<path
android:pathData="M14.611,15.131v-0.042c0,-0.421 0.143,-0.763 0.32,-0.763s0.32,0.342 0.32,0.763v0.042c0.208,0.217 0.355,0.621 0.355,1.103c0,0.513 -0.162,0.949 -0.393,1.152c-0.064,0.195 -0.163,0.33 -0.282,0.33s-0.218,-0.135 -0.282,-0.33c-0.231,-0.203 -0.393,-0.639 -0.393,-1.152c0.001,-0.482 0.147,-0.886 0.355,-1.103zM7.612,16.2v-0.042c0,-0.421 0.143,-0.763 0.32,-0.763s0.32,0.342 0.32,0.763v0.042c0.208,0.217 0.355,0.621 0.355,1.103c0,0.513 -0.162,0.949 -0.393,1.152c-0.064,0.195 -0.163,0.33 -0.282,0.33s-0.218,-0.135 -0.282,-0.33c-0.231,-0.203 -0.393,-0.639 -0.393,-1.152c0,-0.481 0.147,-0.885 0.355,-1.103zM1.595,17.23v-0.039c0,-0.393 0.134,-0.712 0.299,-0.712s0.299,0.319 0.299,0.712v0.039c0.194,0.203 0.331,0.58 0.331,1.03c0,0.479 -0.151,0.886 -0.367,1.076c-0.059,0.182 -0.152,0.308 -0.263,0.308s-0.204,-0.127 -0.264,-0.308c-0.215,-0.189 -0.367,-0.597 -0.367,-1.076c0.001,-0.45 0.137,-0.827 0.332,-1.03zM0,11.146v3.5l18,-3.268L18,7.614z"
android:fillColor="#FFAC33"/>
<path
android:pathData="M18,7.614v3.764l18,3.268v-3.5z"
android:fillColor="#FFD983"/>
</vector>

View File

@ -0,0 +1,170 @@
<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path
android:fillColor="#3DDC84"
android:pathData="M0,0h108v108h-108z" />
<path
android:fillColor="#00000000"
android:pathData="M9,0L9,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,0L19,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,0L29,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,0L39,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,0L49,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,0L59,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,0L69,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,0L79,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M89,0L89,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M99,0L99,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,9L108,9"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,19L108,19"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,29L108,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,39L108,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,49L108,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,59L108,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,69L108,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,79L108,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,89L108,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,99L108,99"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,29L89,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,39L89,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,49L89,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,59L89,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,69L89,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,79L89,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,19L29,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,19L39,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,19L49,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,19L59,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,19L69,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,19L79,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
</vector>

View File

@ -0,0 +1,30 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z">
<aapt:attr name="android:fillColor">
<gradient
android:endX="85.84757"
android:endY="92.4963"
android:startX="42.9492"
android:startY="49.59793"
android:type="linear">
<item
android:color="#44000000"
android:offset="0.0" />
<item
android:color="#00000000"
android:offset="1.0" />
</gradient>
</aapt:attr>
</path>
<path
android:fillColor="#FFFFFF"
android:fillType="nonZero"
android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z"
android:strokeWidth="1"
android:strokeColor="#00000000" />
</vector>

View File

@ -0,0 +1,214 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#1a1a2e">
<!-- Header -->
<TextView
android:id="@+id/headerText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="COMPASS APP"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_marginTop="32dp"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<!-- Compass Container dengan Custom Compass -->
<androidx.cardview.widget.CardView
android:id="@+id/compassCard"
android:layout_width="320dp"
android:layout_height="320dp"
android:layout_marginTop="32dp"
app:cardCornerRadius="160dp"
app:cardElevation="8dp"
app:cardBackgroundColor="#000000"
app:layout_constraintTop_toBottomOf="@id/headerText"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent">
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<!-- Background Circle -->
<View
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@drawable/compass_circle" />
<!-- Compass Ring dengan Derajat dan Arah -->
<ImageView
android:id="@+id/compassRing"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@drawable/compass_ring"
android:contentDescription="Compass Ring" />
<!-- Cardinal Directions Text (U=Utara/N, T=Timur/E, S=Selatan/S, B=Barat/W) -->
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<!-- U (Utara) - Top -->
<TextView
android:id="@+id/directionU"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="U"
android:textSize="28sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_alignParentTop="true"
android:layout_centerHorizontal="true"
android:layout_marginTop="16dp" />
<!-- T (Timur) - Right -->
<TextView
android:id="@+id/directionT"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="T"
android:textSize="28sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_alignParentRight="true"
android:layout_centerVertical="true"
android:layout_marginRight="16dp" />
<!-- S (Selatan) - Bottom -->
<TextView
android:id="@+id/directionS"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="S"
android:textSize="28sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true"
android:layout_marginBottom="16dp" />
<!-- B (Barat) - Left -->
<TextView
android:id="@+id/directionB"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="B"
android:textSize="28sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_alignParentLeft="true"
android:layout_centerVertical="true"
android:layout_marginLeft="16dp" />
</RelativeLayout>
<!-- Center Circle -->
<View
android:layout_width="120dp"
android:layout_height="120dp"
android:layout_gravity="center"
android:background="@drawable/compass_center_circle" />
<!-- Compass Needle (rotating) -->
<ImageView
android:id="@+id/compassNeedle"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@drawable/compass_needle"
android:contentDescription="Compass Needle" />
<!-- Ka'bah Icon - Barat Laut (290.5°) -->
<ImageView
android:id="@+id/kabaahIcon"
android:layout_width="40dp"
android:layout_height="40dp"
android:layout_gravity="center"
android:src="@drawable/ic_kabah"
android:contentDescription="Ka'bah Direction"
android:layout_marginStart="220dp"
android:layout_marginTop="50dp" />
</FrameLayout>
</androidx.cardview.widget.CardView>
<!-- Degree Text -->
<TextView
android:id="@+id/degreeText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0°"
android:textSize="48sp"
android:textStyle="bold"
android:textColor="#e94560"
android:layout_marginTop="24dp"
app:layout_constraintTop_toBottomOf="@id/compassCard"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<!-- Direction Text -->
<TextView
android:id="@+id/directionText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Utara (U)"
android:textSize="20sp"
android:textColor="#ffffff"
android:layout_marginTop="8dp"
app:layout_constraintTop_toBottomOf="@id/degreeText"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<!-- Saved Locations Section -->
<TextView
android:id="@+id/savedLocationsLabel"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Lokasi Tersimpan"
android:textSize="18sp"
android:textStyle="bold"
android:textColor="#ffffff"
android:layout_marginTop="24dp"
android:layout_marginStart="16dp"
app:layout_constraintTop_toBottomOf="@id/directionText"
app:layout_constraintStart_toStartOf="parent" />
<ListView
android:id="@+id/locationsListView"
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_marginTop="12dp"
android:layout_marginStart="16dp"
android:layout_marginEnd="16dp"
android:layout_marginBottom="16dp"
android:background="#16213e"
android:divider="#e94560"
android:dividerHeight="1dp"
android:padding="8dp"
app:layout_constraintTop_toBottomOf="@id/savedLocationsLabel"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<!-- FAB Save Location -->
<com.google.android.material.floatingactionbutton.FloatingActionButton
android:id="@+id/fabSaveLocation"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="24dp"
android:src="@android:drawable/ic_input_add"
app:tint="#ffffff"
app:backgroundTint="#e94560"
android:contentDescription="Save Location"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 982 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.6 KiB

View File

@ -0,0 +1,16 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.Compass" parent="Theme.MaterialComponents.DayNight.DarkActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/purple_200</item>
<item name="colorPrimaryVariant">@color/purple_700</item>
<item name="colorOnPrimary">@color/black</item>
<!-- Secondary brand color. -->
<item name="colorSecondary">@color/teal_200</item>
<item name="colorSecondaryVariant">@color/teal_200</item>
<item name="colorOnSecondary">@color/black</item>
<!-- Status bar color. -->
<item name="android:statusBarColor">?attr/colorPrimaryVariant</item>
<!-- Customize your theme here. -->
</style>
</resources>

View File

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="purple_200">#FFBB86FC</color>
<color name="purple_500">#FF6200EE</color>
<color name="purple_700">#FF3700B3</color>
<color name="teal_200">#FF03DAC5</color>
<color name="teal_700">#FF018786</color>
<color name="black">#FF000000</color>
<color name="white">#FFFFFFFF</color>
</resources>

View File

@ -0,0 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<item name="compassImage" type="id">compassImage</item>
</resources>

View File

@ -0,0 +1,3 @@
<resources>
<string name="app_name">Compass</string>
</resources>

View File

@ -0,0 +1,16 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.Compass" parent="Theme.MaterialComponents.DayNight.DarkActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/purple_500</item>
<item name="colorPrimaryVariant">@color/purple_700</item>
<item name="colorOnPrimary">@color/white</item>
<!-- Secondary brand color. -->
<item name="colorSecondary">@color/teal_200</item>
<item name="colorSecondaryVariant">@color/teal_700</item>
<item name="colorOnSecondary">@color/black</item>
<!-- Status bar color. -->
<item name="android:statusBarColor">?attr/colorPrimaryVariant</item>
<!-- Customize your theme here. -->
</style>
</resources>

View File

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample backup rules file; uncomment and customize as necessary.
See https://developer.android.com/guide/topics/data/autobackup
for details.
Note: This file is ignored for devices older than API 31
See https://developer.android.com/about/versions/12/backup-restore
-->
<full-backup-content>
<!--
<include domain="sharedpref" path="."/>
<exclude domain="sharedpref" path="device.xml"/>
-->
</full-backup-content>

View File

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample data extraction rules file; uncomment and customize as necessary.
See https://developer.android.com/about/versions/12/backup-restore#xml-changes
for details.
-->
<data-extraction-rules>
<cloud-backup>
<!-- TODO: Use <include> and <exclude> to control what is backed up.
<include .../>
<exclude .../>
-->
</cloud-backup>
<!--
<device-transfer>
<include .../>
<exclude .../>
</device-transfer>
-->
</data-extraction-rules>

View File

@ -0,0 +1,17 @@
package com.kelompok6.compass
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)
}
}

6
build.gradle.kts Normal file
View File

@ -0,0 +1,6 @@
// Top-level build file where you can add configuration options common to all sub-projects/modules.
plugins {
alias(libs.plugins.android.application) apply false
alias(libs.plugins.kotlin.android) apply false
id("com.google.gms.google-services") version "4.4.4" apply false
}

23
gradle.properties Normal file
View File

@ -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. For more details, visit
# https://developer.android.com/r/tools/gradle-multi-project-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

22
gradle/libs.versions.toml Normal file
View File

@ -0,0 +1,22 @@
[versions]
agp = "8.13.1"
kotlin = "2.0.21"
coreKtx = "1.17.0"
junit = "4.13.2"
junitVersion = "1.3.0"
espressoCore = "3.7.0"
appcompat = "1.7.1"
material = "1.13.0"
[libraries]
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
junit = { group = "junit", name = "junit", version.ref = "junit" }
androidx-junit = { group = "androidx.test.ext", name = "junit", version.ref = "junitVersion" }
androidx-espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espressoCore" }
androidx-appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" }
material = { group = "com.google.android.material", name = "material", version.ref = "material" }
[plugins]
android-application = { id = "com.android.application", version.ref = "agp" }
kotlin-android = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" }

BIN
gradle/wrapper/gradle-wrapper.jar vendored Normal file

Binary file not shown.

View File

@ -0,0 +1,6 @@
#Thu Dec 04 14:30:28 ICT 2025
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

185
gradlew vendored Normal file
View File

@ -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" "$@"

89
gradlew.bat vendored Normal file
View File

@ -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

24
settings.gradle.kts Normal file
View File

@ -0,0 +1,24 @@
pluginManagement {
repositories {
google {
content {
includeGroupByRegex("com\\.android.*")
includeGroupByRegex("com\\.google.*")
includeGroupByRegex("androidx.*")
}
}
mavenCentral()
gradlePluginPortal()
}
}
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
google()
mavenCentral()
}
}
rootProject.name = "Compass"
include(":app")