@ -0,0 +1,13 @@
|
||||
*.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 |
@ -0,0 +1,119 @@
|
||||
<component name="ProjectCodeStyleConfiguration"> |
||||
<code_scheme name="Project" version="173"> |
||||
<JetCodeStyleSettings> |
||||
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" /> |
||||
</JetCodeStyleSettings> |
||||
<codeStyleSettings language="XML"> |
||||
<arrangement> |
||||
<rules> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>xmlns:android</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>^$</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>xmlns:.*</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>^$</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
<order>BY_NAME</order> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>.*:id</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>.*:name</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>name</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>^$</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>style</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>^$</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>.*</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>^$</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
<order>BY_NAME</order> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>.*</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
<order>ANDROID_ATTRIBUTE_ORDER</order> |
||||
</rule> |
||||
</section> |
||||
<section> |
||||
<rule> |
||||
<match> |
||||
<AND> |
||||
<NAME>.*</NAME> |
||||
<XML_ATTRIBUTE /> |
||||
<XML_NAMESPACE>.*</XML_NAMESPACE> |
||||
</AND> |
||||
</match> |
||||
<order>BY_NAME</order> |
||||
</rule> |
||||
</section> |
||||
</rules> |
||||
</arrangement> |
||||
</codeStyleSettings> |
||||
<codeStyleSettings language="kotlin"> |
||||
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" /> |
||||
</codeStyleSettings> |
||||
</code_scheme> |
||||
</component> |
@ -0,0 +1,5 @@
|
||||
<component name="ProjectCodeStyleConfiguration"> |
||||
<state> |
||||
<option name="USE_PER_PROJECT_SETTINGS" value="true" /> |
||||
</state> |
||||
</component> |
@ -0,0 +1,7 @@
|
||||
<component name="ProjectDictionaryState"> |
||||
<dictionary name="yc00057"> |
||||
<words> |
||||
<w>coroutine</w> |
||||
</words> |
||||
</dictionary> |
||||
</component> |
@ -0,0 +1,4 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project version="4"> |
||||
<component name="Encoding" addBOMForNewFiles="with NO BOM" /> |
||||
</project> |
@ -0,0 +1,18 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project version="4"> |
||||
<component name="GradleSettings"> |
||||
<option name="linkedExternalProjectsSettings"> |
||||
<GradleProjectSettings> |
||||
<option name="distributionType" value="DEFAULT_WRAPPED" /> |
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" /> |
||||
<option name="modules"> |
||||
<set> |
||||
<option value="$PROJECT_DIR$" /> |
||||
<option value="$PROJECT_DIR$/app" /> |
||||
</set> |
||||
</option> |
||||
<option name="resolveModulePerSourceSet" value="false" /> |
||||
</GradleProjectSettings> |
||||
</option> |
||||
</component> |
||||
</project> |
@ -0,0 +1,14 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project version="4"> |
||||
<component name="CMakeSettings"> |
||||
<configurations> |
||||
<configuration PROFILE_NAME="Debug" CONFIG_NAME="Debug" /> |
||||
</configurations> |
||||
</component> |
||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_7" project-jdk-name="1.8" project-jdk-type="JavaSDK"> |
||||
<output url="file://$PROJECT_DIR$/build/classes" /> |
||||
</component> |
||||
<component name="ProjectType"> |
||||
<option name="id" value="Android" /> |
||||
</component> |
||||
</project> |
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project version="4"> |
||||
<component name="RunConfigurationProducerService"> |
||||
<option name="ignoredProducers"> |
||||
<set> |
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.AllInPackageGradleConfigurationProducer" /> |
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestClassGradleConfigurationProducer" /> |
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestMethodGradleConfigurationProducer" /> |
||||
</set> |
||||
</option> |
||||
</component> |
||||
</project> |
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project version="4"> |
||||
<component name="VcsDirectoryMappings"> |
||||
<mapping directory="$PROJECT_DIR$" vcs="Git" /> |
||||
</component> |
||||
</project> |
@ -0,0 +1,17 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<projectDescription> |
||||
<name>endless-service</name> |
||||
<comment>Project endless-service created by Buildship.</comment> |
||||
<projects> |
||||
</projects> |
||||
<buildSpec> |
||||
<buildCommand> |
||||
<name>org.eclipse.buildship.core.gradleprojectbuilder</name> |
||||
<arguments> |
||||
</arguments> |
||||
</buildCommand> |
||||
</buildSpec> |
||||
<natures> |
||||
<nature>org.eclipse.buildship.core.gradleprojectnature</nature> |
||||
</natures> |
||||
</projectDescription> |
@ -0,0 +1,2 @@
|
||||
connection.project.dir= |
||||
eclipse.preferences.version=1 |
@ -0,0 +1,3 @@
|
||||
# Endless Service |
||||
|
||||
This is just a project showing how to run an Android Foreground Service that will never stop running. |
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<classpath> |
||||
<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-1.8/"/> |
||||
<classpathentry kind="con" path="org.eclipse.buildship.core.gradleclasspathcontainer"/> |
||||
<classpathentry kind="output" path="bin/default"/> |
||||
</classpath> |
@ -0,0 +1,23 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<projectDescription> |
||||
<name>app</name> |
||||
<comment>Project app created by Buildship.</comment> |
||||
<projects> |
||||
</projects> |
||||
<buildSpec> |
||||
<buildCommand> |
||||
<name>org.eclipse.jdt.core.javabuilder</name> |
||||
<arguments> |
||||
</arguments> |
||||
</buildCommand> |
||||
<buildCommand> |
||||
<name>org.eclipse.buildship.core.gradleprojectbuilder</name> |
||||
<arguments> |
||||
</arguments> |
||||
</buildCommand> |
||||
</buildSpec> |
||||
<natures> |
||||
<nature>org.eclipse.jdt.core.javanature</nature> |
||||
<nature>org.eclipse.buildship.core.gradleprojectnature</nature> |
||||
</natures> |
||||
</projectDescription> |
@ -0,0 +1,2 @@
|
||||
connection.project.dir=.. |
||||
eclipse.preferences.version=1 |
@ -0,0 +1,37 @@
|
||||
apply plugin: 'com.android.application' |
||||
|
||||
apply plugin: 'kotlin-android' |
||||
|
||||
apply plugin: 'kotlin-android-extensions' |
||||
|
||||
android { |
||||
compileSdkVersion 28 |
||||
defaultConfig { |
||||
applicationId "co.zzyzx.sensorlogger" |
||||
minSdkVersion 23 |
||||
targetSdkVersion 28 |
||||
versionCode 1 |
||||
versionName "1.0" |
||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" |
||||
} |
||||
buildTypes { |
||||
release { |
||||
minifyEnabled false |
||||
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' |
||||
} |
||||
} |
||||
} |
||||
|
||||
dependencies { |
||||
implementation fileTree(dir: 'libs', include: ['*.jar']) |
||||
implementation"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version" |
||||
implementation 'com.android.support:appcompat-v7:28.0.0' |
||||
implementation 'com.android.support.constraint:constraint-layout:1.1.3' |
||||
implementation 'com.github.kittinunf.fuel:fuel:2.1.0' |
||||
implementation 'com.github.kittinunf.fuel:fuel-android:2.1.0' |
||||
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.0-M2' |
||||
|
||||
testImplementation 'junit:junit:4.12' |
||||
androidTestImplementation 'com.android.support.test:runner:1.0.2' |
||||
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2' |
||||
} |
@ -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 |
@ -0,0 +1,24 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.support.test.InstrumentationRegistry |
||||
import android.support.test.runner.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.getTargetContext() |
||||
assertEquals("co.zzyzx.sensorlogger", appContext.packageName) |
||||
} |
||||
} |
@ -0,0 +1,38 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||
package="co.zzyzx.sensorlogger"> |
||||
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" /> |
||||
<uses-permission android:name="android.permission.INTERNET" /> |
||||
<uses-permission android:name="android.permission.WAKE_LOCK" /> |
||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> |
||||
|
||||
<application |
||||
android:allowBackup="true" |
||||
android:icon="@mipmap/ic_launcher" |
||||
android:label="@string/app_name" |
||||
android:roundIcon="@mipmap/ic_launcher_round" |
||||
android:supportsRtl="true" |
||||
android:theme="@style/AppTheme"> |
||||
|
||||
<service |
||||
android:name=".EndlessService" |
||||
android:enabled="true" |
||||
android:exported="false"> |
||||
</service> |
||||
|
||||
<activity android:name=".MainActivity"> |
||||
<intent-filter> |
||||
<action android:name="android.intent.action.MAIN"/> |
||||
<category android:name="android.intent.category.LAUNCHER"/> |
||||
</intent-filter> |
||||
</activity> |
||||
|
||||
<receiver android:enabled="true" android:name=".StartReceiver"> |
||||
<intent-filter> |
||||
<action android:name="android.intent.action.BOOT_COMPLETED"/> |
||||
</intent-filter> |
||||
</receiver> |
||||
|
||||
</application> |
||||
|
||||
</manifest> |
@ -0,0 +1,6 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
enum class Actions { |
||||
START, |
||||
STOP |
||||
} |
@ -0,0 +1,236 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.app.* |
||||
import android.content.Context |
||||
import android.content.Intent |
||||
import android.graphics.Color |
||||
import android.hardware.Sensor |
||||
import android.hardware.SensorEvent |
||||
import android.hardware.SensorEventListener |
||||
import android.hardware.SensorManager |
||||
import android.os.Build |
||||
import android.os.IBinder |
||||
import android.os.PowerManager |
||||
import android.provider.Settings |
||||
import android.support.v4.app.NotificationCompat |
||||
import android.widget.Toast |
||||
import java.text.SimpleDateFormat |
||||
import java.util.* |
||||
import com.github.kittinunf.fuel.Fuel |
||||
//import com.github.kittinunf.fuel.core.extensions.jsonBody |
||||
import kotlinx.coroutines.* |
||||
|
||||
|
||||
class EndlessService : Service(), SensorEventListener { |
||||
|
||||
private var wakeLock: PowerManager.WakeLock? = null |
||||
private var isServiceStarted = false |
||||
private lateinit var nm: NotificationManager |
||||
private val notificationChannelId = "ENDLESS SERVICE CHANNEL" |
||||
private val notificationId = 1011 |
||||
|
||||
private lateinit var mSensorManager: SensorManager |
||||
private lateinit var mAccelerometer: Sensor |
||||
|
||||
override fun onBind(intent: Intent): IBinder? { |
||||
log("Some component want to bind with the service") |
||||
// We don't provide binding, so return null |
||||
return null |
||||
} |
||||
|
||||
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { |
||||
log("onStartCommand executed with startId: $startId") |
||||
if (intent != null) { |
||||
val action = intent.action |
||||
log("using an intent with action $action") |
||||
when (action) { |
||||
Actions.START.name -> startService() |
||||
Actions.STOP.name -> stopService() |
||||
else -> log("This should never happen. No action in the received intent") |
||||
} |
||||
} else { |
||||
log( |
||||
"with a null intent. It has been probably restarted by the system." |
||||
) |
||||
} |
||||
|
||||
|
||||
mSensorManager.registerListener(this, mAccelerometer, SensorManager.SENSOR_DELAY_NORMAL) |
||||
|
||||
// by returning this we make sure the service is restarted if the system kills the service |
||||
return START_STICKY |
||||
} |
||||
|
||||
override fun onCreate() { |
||||
super.onCreate() |
||||
log("The service has been created".toUpperCase()) |
||||
val notification = createNotification() |
||||
notification.flags = Notification.FLAG_ONGOING_EVENT |
||||
startForeground(notificationId, notification) |
||||
|
||||
mSensorManager = getSystemService(Context.SENSOR_SERVICE) as SensorManager |
||||
mAccelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER) |
||||
|
||||
|
||||
} |
||||
|
||||
override fun onDestroy() { |
||||
super.onDestroy() |
||||
mSensorManager.unregisterListener(this) |
||||
log("The service has been destroyed".toUpperCase()) |
||||
Toast.makeText(this, "Service destroyed", Toast.LENGTH_SHORT).show() |
||||
} |
||||
|
||||
private fun startService() { |
||||
if (isServiceStarted) return |
||||
log("Starting the foreground service task") |
||||
Toast.makeText(this, "Service starting its task", Toast.LENGTH_SHORT).show() |
||||
isServiceStarted = true |
||||
setServiceState(this, ServiceState.STARTED) |
||||
|
||||
// we need this lock so our service gets not affected by Doze Mode |
||||
wakeLock = |
||||
(getSystemService(Context.POWER_SERVICE) as PowerManager).run { |
||||
newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "EndlessService::lock").apply { |
||||
acquire() |
||||
} |
||||
} |
||||
|
||||
// we're starting a loop in a coroutine |
||||
GlobalScope.launch(Dispatchers.IO) { |
||||
while (isServiceStarted) { |
||||
launch(Dispatchers.IO) { |
||||
pingFakeServer() |
||||
} |
||||
delay(1 * 60 * 1000) |
||||
} |
||||
log("End of the loop for the service") |
||||
} |
||||
} |
||||
|
||||
private fun stopService() { |
||||
log("Stopping the foreground service") |
||||
Toast.makeText(this, "Service stopping", Toast.LENGTH_SHORT).show() |
||||
try { |
||||
wakeLock?.let { |
||||
if (it.isHeld) { |
||||
it.release() |
||||
} |
||||
} |
||||
stopForeground(true) |
||||
stopSelf() |
||||
} catch (e: Exception) { |
||||
log("Service stopped without being started: ${e.message}") |
||||
} |
||||
isServiceStarted = false |
||||
setServiceState(this, ServiceState.STOPPED) |
||||
nm.cancelAll() |
||||
} |
||||
|
||||
override fun onAccuracyChanged(p0: Sensor?, p1: Int) { |
||||
|
||||
} |
||||
|
||||
override fun onSensorChanged(evt: SensorEvent) { |
||||
fun Double.format(digits: Int) = java.lang.String.format("%.${digits}f", this) |
||||
val accelGravity = arrayListOf(0.toDouble(), 0.toDouble(), 0.toDouble()) |
||||
|
||||
when (evt.sensor?.type) { |
||||
Sensor.TYPE_ACCELEROMETER -> { |
||||
|
||||
val accelLin = arrayOf(0.toDouble(), 0.toDouble(), 0.toDouble()) |
||||
val alpha = 0.8f |
||||
accelGravity[0] = alpha * accelGravity[0] + (1 - alpha) * evt.values[0] |
||||
accelGravity[1] = alpha * accelGravity[1] + (1 - alpha) * evt.values[1] |
||||
accelGravity[2] = alpha * accelGravity[2] + (1 - alpha) * evt.values[2] |
||||
accelLin[0] = evt.values[0] - accelGravity[0] |
||||
accelLin[1] = evt.values[1] - accelGravity[1] |
||||
accelLin[2] = evt.values[2] - accelGravity[2] |
||||
|
||||
val txt = |
||||
"[accel] (${accelLin[0].format(3)}, ${accelLin[1].format(3)}, ${accelLin[2].format( |
||||
3 |
||||
)})" |
||||
val notification = createNotification(txt) |
||||
nm.notify(notificationId, notification) |
||||
} |
||||
} |
||||
} |
||||
|
||||
|
||||
private fun pingFakeServer() { |
||||
val df = SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.mmmZ") |
||||
val gmtTime = df.format(Date()) |
||||
|
||||
val deviceId = Settings.Secure.getString( |
||||
applicationContext.contentResolver, |
||||
Settings.Secure.ANDROID_ID |
||||
) |
||||
|
||||
// val json = |
||||
// """ |
||||
// { |
||||
// "deviceId": "$deviceId", |
||||
// "createdAt": "$gmtTime" |
||||
// } |
||||
// """ |
||||
try { |
||||
Fuel.get("https://tool.everyday.in.th/ip?format=text") |
||||
// .jsonBody(json) |
||||
.response { _, _, result -> |
||||
val (bytes, error) = result |
||||
if (bytes != null) { |
||||
log("[response bytes] ${String(bytes)}") |
||||
val notification = createNotification("[req] ${gmtTime} - ${String(bytes)}") |
||||
nm.notify(notificationId, notification) |
||||
} else { |
||||
log("[response error] ${error?.message}") |
||||
} |
||||
} |
||||
} catch (e: Exception) { |
||||
log("Error making the request: ${e.message}") |
||||
} |
||||
} |
||||
|
||||
private fun createNotification(text: CharSequence = "Howdy?"): Notification { |
||||
|
||||
// depending on the Android API that we're dealing with we will have |
||||
// to use a specific method to create the notification |
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |
||||
nm = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager |
||||
val channel = NotificationChannel( |
||||
notificationChannelId, |
||||
"SL notifications channel", |
||||
NotificationManager.IMPORTANCE_DEFAULT |
||||
).let { |
||||
it.description = "SL Service channel" |
||||
it.enableLights(true) |
||||
it.lightColor = Color.RED |
||||
it.enableVibration(true) |
||||
it.vibrationPattern = longArrayOf(100, 200, 300, 400, 500, 400, 300, 200, 400) |
||||
it |
||||
} |
||||
nm.createNotificationChannel(channel) |
||||
} |
||||
|
||||
val pendingIntent: PendingIntent = |
||||
Intent(this, MainActivity::class.java).let { notificationIntent -> |
||||
PendingIntent.getActivity(this, 0, notificationIntent, 0) |
||||
} |
||||
|
||||
val builder: Notification.Builder = |
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) Notification.Builder( |
||||
this, |
||||
notificationChannelId |
||||
) else Notification.Builder(this) |
||||
|
||||
return builder |
||||
.setContentTitle("SensorLogger") |
||||
.setContentText(text) |
||||
.setContentIntent(pendingIntent) |
||||
.setSmallIcon(R.mipmap.ic_launcher) |
||||
.setTicker("Ticker text") |
||||
.setPriority(Notification.PRIORITY_DEFAULT) // for under android 26 compatibility |
||||
.build() |
||||
} |
||||
} |
@ -0,0 +1,46 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.content.Intent |
||||
import android.os.Build |
||||
import android.support.v7.app.AppCompatActivity |
||||
import android.os.Bundle |
||||
import android.util.Log |
||||
import android.widget.Button |
||||
|
||||
class MainActivity : AppCompatActivity() { |
||||
|
||||
override fun onCreate(savedInstanceState: Bundle?) { |
||||
super.onCreate(savedInstanceState) |
||||
setContentView(R.layout.activity_main) |
||||
|
||||
title = getString(R.string.app_name) |
||||
|
||||
findViewById<Button>(R.id.btnStartService).let { |
||||
it.setOnClickListener { |
||||
log("START THE FOREGROUND SERVICE ON DEMAND") |
||||
actionOnService(Actions.START) |
||||
} |
||||
} |
||||
|
||||
findViewById<Button>(R.id.btnStopService).let { |
||||
it.setOnClickListener { |
||||
log("STOP THE FOREGROUND SERVICE ON DEMAND") |
||||
actionOnService(Actions.STOP) |
||||
} |
||||
} |
||||
} |
||||
|
||||
private fun actionOnService(action: Actions) { |
||||
if (getServiceState(this) == ServiceState.STOPPED && action == Actions.STOP) return |
||||
Intent(this, EndlessService::class.java).also { |
||||
it.action = action.name |
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |
||||
log("Starting the service in >=26 Mode") |
||||
startForegroundService(it) |
||||
return |
||||
} |
||||
log("Starting the service in < 26 Mode") |
||||
startService(it) |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,30 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.content.Context |
||||
import android.content.SharedPreferences |
||||
|
||||
enum class ServiceState { |
||||
STARTED, |
||||
STOPPED, |
||||
} |
||||
|
||||
private const val name = "SPYSERVICE_KEY" |
||||
private const val key = "SPYSERVICE_STATE" |
||||
|
||||
fun setServiceState(context: Context, state: ServiceState) { |
||||
val sharedPrefs = getPreferences(context) |
||||
sharedPrefs.edit().let { |
||||
it.putString(key, state.name) |
||||
it.apply() |
||||
} |
||||
} |
||||
|
||||
fun getServiceState(context: Context): ServiceState { |
||||
val sharedPrefs = getPreferences(context) |
||||
val value = sharedPrefs.getString(key, ServiceState.STOPPED.name) |
||||
return ServiceState.valueOf(value) |
||||
} |
||||
|
||||
private fun getPreferences(context: Context): SharedPreferences { |
||||
return context.getSharedPreferences(name, 0) |
||||
} |
@ -0,0 +1,24 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.content.BroadcastReceiver |
||||
import android.content.Context |
||||
import android.content.Intent |
||||
import android.os.Build |
||||
|
||||
class StartReceiver : BroadcastReceiver() { |
||||
|
||||
override fun onReceive(context: Context, intent: Intent) { |
||||
if (intent.action == Intent.ACTION_BOOT_COMPLETED && getServiceState(context) == ServiceState.STARTED) { |
||||
Intent(context, EndlessService::class.java).also { |
||||
it.action = Actions.START.name |
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |
||||
log("Starting the service in >=26 Mode from a BroadcastReceiver") |
||||
context.startForegroundService(it) |
||||
return |
||||
} |
||||
log("Starting the service in < 26 Mode from a BroadcastReceiver") |
||||
context.startService(it) |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,7 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
import android.util.Log |
||||
|
||||
fun log(msg: String) { |
||||
Log.d("ENDLESS-SERVICE", msg) |
||||
} |
@ -0,0 +1,34 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:aapt="http://schemas.android.com/aapt" |
||||
android:width="108dp" |
||||
android:height="108dp" |
||||
android:viewportHeight="108" |
||||
android:viewportWidth="108"> |
||||
<path |
||||
android:fillType="evenOdd" |
||||
android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z" |
||||
android:strokeColor="#00000000" |
||||
android:strokeWidth="1"> |
||||
<aapt:attr name="android:fillColor"> |
||||
<gradient |
||||
android:endX="78.5885" |
||||
android:endY="90.9159" |
||||
android:startX="48.7653" |
||||
android:startY="61.0927" |
||||
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="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z" |
||||
android:strokeColor="#00000000" |
||||
android:strokeWidth="1"/> |
||||
</vector> |
@ -0,0 +1,74 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<vector |
||||
xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:height="108dp" |
||||
android:width="108dp" |
||||
android:viewportHeight="108" |
||||
android:viewportWidth="108"> |
||||
<path android:fillColor="#008577" |
||||
android:pathData="M0,0h108v108h-108z"/> |
||||
<path android:fillColor="#00000000" android:pathData="M9,0L9,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,0L19,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M29,0L29,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M39,0L39,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M49,0L49,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M59,0L59,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M69,0L69,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M79,0L79,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M89,0L89,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M99,0L99,108" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,9L108,9" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,19L108,19" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,29L108,29" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,39L108,39" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,49L108,49" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,59L108,59" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,69L108,69" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,79L108,79" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,89L108,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M0,99L108,99" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,29L89,29" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,39L89,39" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,49L89,49" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,59L89,59" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,69L89,69" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M19,79L89,79" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M29,19L29,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M39,19L39,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M49,19L49,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M59,19L59,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M69,19L69,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
<path android:fillColor="#00000000" android:pathData="M79,19L79,89" |
||||
android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/> |
||||
</vector> |
@ -0,0 +1,29 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<android.support.constraint.ConstraintLayout |
||||
xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context=".MainActivity"> |
||||
|
||||
<LinearLayout |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:orientation="vertical"> |
||||
|
||||
|
||||
<Button |
||||
android:id="@+id/btnStartService" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:text="Start Foreground Service" android:paddingTop="20dp" android:layout_marginTop="20dp"/> |
||||
|
||||
<Button |
||||
android:id="@+id/btnStopService" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:text="Stop Foreground Service"/> |
||||
|
||||
</LinearLayout> |
||||
|
||||
</android.support.constraint.ConstraintLayout> |
@ -0,0 +1,5 @@
|
||||
<?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"/> |
||||
</adaptive-icon> |
@ -0,0 +1,5 @@
|
||||
<?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"/> |
||||
</adaptive-icon> |
After Width: | Height: | Size: 2.9 KiB |
After Width: | Height: | Size: 4.8 KiB |
After Width: | Height: | Size: 2.0 KiB |
After Width: | Height: | Size: 2.7 KiB |
After Width: | Height: | Size: 4.4 KiB |
After Width: | Height: | Size: 6.7 KiB |
After Width: | Height: | Size: 6.2 KiB |
After Width: | Height: | Size: 10 KiB |
After Width: | Height: | Size: 8.9 KiB |
After Width: | Height: | Size: 15 KiB |
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<resources> |
||||
<color name="colorPrimary">#008577</color> |
||||
<color name="colorPrimaryDark">#00574B</color> |
||||
<color name="colorAccent">#D81B60</color> |
||||
</resources> |
@ -0,0 +1,3 @@
|
||||
<resources> |
||||
<string name="app_name">SensorLogger</string> |
||||
</resources> |
@ -0,0 +1,11 @@
|
||||
<resources> |
||||
|
||||
<!-- Base application theme. --> |
||||
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> |
||||
<!-- Customize your theme here. --> |
||||
<item name="colorPrimary">@color/colorPrimary</item> |
||||
<item name="colorPrimaryDark">@color/colorPrimaryDark</item> |
||||
<item name="colorAccent">@color/colorAccent</item> |
||||
</style> |
||||
|
||||
</resources> |
@ -0,0 +1,17 @@
|
||||
package co.zzyzx.sensorlogger |
||||
|
||||
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) |
||||
} |
||||
} |
@ -0,0 +1,28 @@
|
||||
// Top-level build file where you can add configuration options common to all sub-projects/modules. |
||||
|
||||
buildscript { |
||||
ext.kotlin_version = '1.3.40' |
||||
repositories { |
||||
google() |
||||
jcenter() |
||||
|
||||
} |
||||
dependencies { |
||||
classpath 'com.android.tools.build:gradle:3.5.0' |
||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" |
||||
// NOTE: Do not place your application dependencies here; they belong |
||||
// in the individual module build.gradle files |
||||
} |
||||
} |
||||
|
||||
allprojects { |
||||
repositories { |
||||
google() |
||||
jcenter() |
||||
|
||||
} |
||||
} |
||||
|
||||
task clean(type: Delete) { |
||||
delete rootProject.buildDir |
||||
} |
@ -0,0 +1,15 @@
|
||||
# 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=-Xmx1536m |
||||
# When configured, Gradle will run in incubating parallel mode. |
||||
# This option should only be used with decoupled projects. More details, visit |
||||
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects |
||||
# org.gradle.parallel=true |
||||
# Kotlin code style for this project: "official" or "obsolete": |
||||
kotlin.code.style=official |
@ -0,0 +1,6 @@
|
||||
#Tue Oct 01 15:24:16 JST 2019 |
||||
distributionBase=GRADLE_USER_HOME |
||||
distributionPath=wrapper/dists |
||||
zipStoreBase=GRADLE_USER_HOME |
||||
zipStorePath=wrapper/dists |
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip |
@ -0,0 +1,172 @@
|
||||
#!/usr/bin/env sh |
||||
|
||||
############################################################################## |
||||
## |
||||
## 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="" |
||||
|
||||
# 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, switch paths to Windows format before running java |
||||
if $cygwin ; 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=$((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" |
||||
|
||||
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong |
||||
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then |
||||
cd "$(dirname "$0")" |
||||
fi |
||||
|
||||
exec "$JAVACMD" "$@" |
@ -0,0 +1,84 @@
|
||||
@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 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= |
||||
|
||||
@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 init |
||||
|
||||
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 init |
||||
|
||||
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 |
||||
|
||||
:init |
||||
@rem Get command-line arguments, handling Windows variants |
||||
|
||||
if not "%OS%" == "Windows_NT" goto win9xME_args |
||||
|
||||
:win9xME_args |
||||
@rem Slurp the command line arguments. |
||||
set CMD_LINE_ARGS= |
||||
set _SKIP=2 |
||||
|
||||
:win9xME_args_slurp |
||||
if "x%~1" == "x" goto execute |
||||
|
||||
set CMD_LINE_ARGS=%* |
||||
|
||||
: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 %CMD_LINE_ARGS% |
||||
|
||||
: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 |