├── .gitignore ├── .idea ├── .gitignore ├── compiler.xml ├── gradle.xml ├── jarRepositories.xml ├── misc.xml └── vcs.xml ├── NotificationTimer ├── .gitignore ├── build.gradle ├── consumer-rules.pro ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── io │ │ └── joon │ │ └── notificationtimer │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ └── java │ │ └── io │ │ └── joon │ │ └── notificationtimer │ │ ├── NotificationTimer.kt │ │ └── TimerService.kt │ └── test │ └── java │ └── io │ └── joon │ └── notificationtimer │ └── ExampleUnitTest.kt ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── io │ │ └── joon │ │ └── notitimer │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── io │ │ │ └── joon │ │ │ └── notitimer │ │ │ └── MainActivity.kt │ └── res │ │ ├── drawable-v24 │ │ └── ic_launcher_foreground.xml │ │ ├── drawable │ │ ├── ic_launcher_background.xml │ │ ├── ic_pause_noti.xml │ │ ├── ic_play_noti.xml │ │ ├── ic_stop_noti.xml │ │ └── ic_timer.jpeg │ │ ├── layout │ │ └── activity_main.xml │ │ ├── mipmap-anydpi-v26 │ │ ├── ic_launcher.xml │ │ └── ic_launcher_round.xml │ │ ├── mipmap-hdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-mdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── values-night │ │ └── themes.xml │ │ └── values │ │ ├── colors.xml │ │ ├── strings.xml │ │ └── themes.xml │ └── test │ └── java │ └── io │ └── joon │ └── notitimer │ └── ExampleUnitTest.kt ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/caches 5 | /.idea/libraries 6 | /.idea/modules.xml 7 | /.idea/workspace.xml 8 | /.idea/navEditor.xml 9 | /.idea/assetWizardSettings.xml 10 | .DS_Store 11 | /build 12 | /captures 13 | .externalNativeBuild 14 | .cxx 15 | local.properties 16 | -------------------------------------------------------------------------------- /.idea/.gitignore: -------------------------------------------------------------------------------- 1 | # Default ignored files 2 | /shelf/ 3 | /workspace.xml 4 | -------------------------------------------------------------------------------- /.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 22 | 23 | -------------------------------------------------------------------------------- /.idea/jarRepositories.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 9 | 10 | 14 | 15 | 19 | 20 | 24 | 25 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 9 | -------------------------------------------------------------------------------- /.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /NotificationTimer/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /NotificationTimer/build.gradle: -------------------------------------------------------------------------------- 1 | plugins { 2 | id 'com.android.library' 3 | id 'kotlin-android' 4 | id 'com.github.dcendents.android-maven' 5 | } 6 | group = 'com.github.Jun-Hub' 7 | 8 | android { 9 | compileSdkVersion 29 10 | 11 | defaultConfig { 12 | minSdkVersion 21 13 | targetSdkVersion 29 14 | versionCode 1 15 | versionName "1.0" 16 | 17 | testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" 18 | consumerProguardFiles "consumer-rules.pro" 19 | } 20 | 21 | buildTypes { 22 | release { 23 | minifyEnabled false 24 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' 25 | } 26 | } 27 | compileOptions { 28 | sourceCompatibility JavaVersion.VERSION_1_8 29 | targetCompatibility JavaVersion.VERSION_1_8 30 | } 31 | kotlinOptions { 32 | jvmTarget = '1.8' 33 | } 34 | } 35 | 36 | dependencies { 37 | 38 | implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" 39 | implementation 'androidx.core:core-ktx:1.3.2' 40 | implementation 'androidx.appcompat:appcompat:1.2.0' 41 | implementation 'com.google.android.material:material:1.3.0' 42 | testImplementation 'junit:junit:4.+' 43 | androidTestImplementation 'androidx.test.ext:junit:1.1.2' 44 | androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0' 45 | implementation 'androidx.media:media:1.2.1' 46 | } -------------------------------------------------------------------------------- /NotificationTimer/consumer-rules.pro: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Jun-Hub/NotificationTimer/1625c0ce12a94ce8b43b4b7583225366f53b7fd9/NotificationTimer/consumer-rules.pro -------------------------------------------------------------------------------- /NotificationTimer/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /NotificationTimer/src/androidTest/java/io/joon/notificationtimer/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notificationtimer 2 | 3 | import androidx.test.platform.app.InstrumentationRegistry 4 | import androidx.test.ext.junit.runners.AndroidJUnit4 5 | 6 | import org.junit.Test 7 | import org.junit.runner.RunWith 8 | 9 | import org.junit.Assert.* 10 | 11 | /** 12 | * Instrumented test, which will execute on an Android device. 13 | * 14 | * See [testing documentation](http://d.android.com/tools/testing). 15 | */ 16 | @RunWith(AndroidJUnit4::class) 17 | class ExampleInstrumentedTest { 18 | @Test 19 | fun useAppContext() { 20 | // Context of the app under test. 21 | val appContext = InstrumentationRegistry.getInstrumentation().targetContext 22 | assertEquals("io.joon.notificationtimer.test", appContext.packageName) 23 | } 24 | } -------------------------------------------------------------------------------- /NotificationTimer/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | -------------------------------------------------------------------------------- /NotificationTimer/src/main/java/io/joon/notificationtimer/NotificationTimer.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notificationtimer 2 | 3 | import android.app.Notification 4 | import android.app.NotificationChannel 5 | import android.app.NotificationManager 6 | import android.app.PendingIntent 7 | import android.content.Context 8 | import android.content.Intent 9 | import android.os.Build 10 | import androidx.core.app.NotificationCompat 11 | import androidx.core.app.NotificationManagerCompat 12 | import androidx.core.content.ContextCompat 13 | import kotlin.properties.Delegates 14 | 15 | interface Timer { 16 | fun play(context: Context, timeMillis: Long) 17 | fun pause(context: Context) 18 | fun stop(context: Context) 19 | fun terminate(context: Context) 20 | } 21 | 22 | typealias onFinishListener = () -> Unit 23 | typealias onTickListener = (Long) -> Unit 24 | 25 | object NotificationTimer: Timer { 26 | 27 | private var notiIcon:Int? = null 28 | private var notiTitle: CharSequence = "" 29 | private var showWhen = false 30 | private var notiColor = 0x66FFFFFF 31 | private var notificationPriority = NotificationCompat.PRIORITY_LOW 32 | private var isAutoCancel = false 33 | private var isOnlyAlertOnce = true 34 | private var isControlMode = false 35 | private var contentPendingIntent: PendingIntent? = null 36 | private var playBtnIcon: Int? = null 37 | private var pauseBtnIcon: Int? = null 38 | private var stopBtnIcon: Int? = null 39 | private var finishListener: onFinishListener? = null 40 | private var tickListener: onTickListener? = null 41 | 42 | private lateinit var channelId: String 43 | private lateinit var notificationManager: NotificationManagerCompat 44 | private lateinit var pausePendingIntent: PendingIntent 45 | private lateinit var stopPendingIntent: PendingIntent 46 | 47 | private var setStartTime by Delegates.notNull() 48 | 49 | override fun play(context: Context, timeMillis: Long) { 50 | if(TimerService.state == TimerState.RUNNING) return 51 | 52 | val playIntent = Intent(context, TimerService::class.java).apply { 53 | action = "PLAY" 54 | putExtra("setTime", timeMillis) 55 | putExtra("forReplay", TimerService.state == TimerState.PAUSED) 56 | } 57 | ContextCompat.startForegroundService(context, playIntent) 58 | } 59 | 60 | override fun pause(context: Context) { 61 | if(TimerService.state != TimerState.RUNNING) return 62 | 63 | val pauseIntent = Intent(context, TimerService::class.java).apply { 64 | action = "PAUSE" 65 | } 66 | ContextCompat.startForegroundService(context, pauseIntent) 67 | } 68 | 69 | override fun stop(context: Context) { 70 | if(TimerService.state == TimerState.STOPPED) return 71 | 72 | val stopIntent = Intent(context, TimerService::class.java).apply { 73 | action = "STOP" 74 | } 75 | ContextCompat.startForegroundService(context, stopIntent) 76 | } 77 | 78 | override fun terminate(context: Context) { 79 | if(!::notificationManager.isInitialized && TimerService.state == TimerState.TERMINATED) return 80 | 81 | val terminateIntent = Intent(context, TimerService::class.java).apply { 82 | action = "TERMINATE" 83 | } 84 | ContextCompat.startForegroundService(context, terminateIntent) 85 | } 86 | 87 | fun createNotification(context: Context, setTime: Long): Notification { 88 | channelId = "${context.packageName}.timer" 89 | notificationManager = NotificationManagerCompat.from(context) 90 | 91 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { 92 | val channel = 93 | NotificationChannel(channelId, "timer", NotificationManager.IMPORTANCE_LOW).apply { setShowBadge(false) } 94 | notificationManager.createNotificationChannel(channel) 95 | } 96 | 97 | val pauseIntent = Intent(context, TimerService::class.java).apply { action = "PAUSE" } 98 | val stopIntent = Intent(context, TimerService::class.java).apply { action = "STOP" } 99 | 100 | pausePendingIntent = PendingIntent.getService(context, 29, pauseIntent, PendingIntent.FLAG_UPDATE_CURRENT) 101 | stopPendingIntent = PendingIntent.getService(context, 29, stopIntent, PendingIntent.FLAG_UPDATE_CURRENT) 102 | 103 | this.setStartTime = setTime 104 | 105 | val minutesUntilFinished = (setTime/1000 - 1) / 60 106 | val secondsInMinuteUntilFinished = ((setTime/1000 - 1) - minutesUntilFinished * 60) 107 | val secondsStr = secondsInMinuteUntilFinished.toString() 108 | val showTime = 109 | "$minutesUntilFinished : ${if (secondsStr.length == 2) secondsStr else "0$secondsStr"}" 110 | 111 | return playStateNotification(context, showTime) 112 | } 113 | 114 | fun updateTimeLeft(context: Context, timeLeft: String) = notificationManager.notify(55, playStateNotification(context, timeLeft)) 115 | 116 | fun updatePauseState(context: Context, timeLeft: String) = notificationManager.notify(55, pauseStateNotification(context, timeLeft)) 117 | 118 | fun updateStopState(context: Context, timeLeft: String, timeUp: Boolean = false) { 119 | notificationManager.notify(55, standByStateNotification(context, timeLeft)) 120 | if(timeUp) 121 | finishListener?.invoke() 122 | } 123 | 124 | fun updateUntilFinished(millisUntilFinished: Long) = tickListener?.invoke(millisUntilFinished) 125 | 126 | fun removeNotification() = notificationManager.cancelAll() 127 | 128 | private fun baseNotificationBuilder(context: Context, timeLeft: String) = 129 | NotificationCompat.Builder(context, channelId).apply { 130 | notiIcon?.let { setSmallIcon(it) } 131 | setContentTitle(notiTitle) 132 | setContentText(timeLeft) 133 | setShowWhen(showWhen) 134 | color = notiColor 135 | priority = notificationPriority 136 | setAutoCancel(isAutoCancel) 137 | setOnlyAlertOnce(isOnlyAlertOnce) 138 | contentPendingIntent?.let { setContentIntent(it) } 139 | if(isControlMode) 140 | setStyle(androidx.media.app.NotificationCompat.MediaStyle().setShowActionsInCompactView(0, 1)) 141 | } 142 | 143 | private fun playStateNotification(context: Context, timeLeft: String): Notification = 144 | baseNotificationBuilder(context, timeLeft).apply { 145 | if(isControlMode) { 146 | pauseBtnIcon?.let { addAction(it, "pause", pausePendingIntent) } 147 | stopBtnIcon?.let { addAction(it, "stop", stopPendingIntent) } 148 | } 149 | }.build() 150 | 151 | private fun pauseStateNotification(context: Context, timeLeft: String): Notification = 152 | baseNotificationBuilder(context, timeLeft).apply { 153 | if(isControlMode) { 154 | playBtnIcon?.let { addAction(it, "play", getPlayPendingIntent(context, true)) } 155 | stopBtnIcon?.let { addAction(it, "stop", stopPendingIntent) } 156 | } 157 | }.build() 158 | 159 | private fun standByStateNotification(context: Context, timeLeft: String): Notification = 160 | baseNotificationBuilder(context, timeLeft).apply { 161 | if(isControlMode) { 162 | playBtnIcon?.let { addAction(it, "play", getPlayPendingIntent(context)) } 163 | stopBtnIcon?.let { addAction(it, "stop", stopPendingIntent) } 164 | } 165 | }.build() 166 | 167 | private fun getPlayPendingIntent(context: Context, isPausingState: Boolean = false): PendingIntent { 168 | val playIntent = Intent(context, TimerService::class.java).apply { 169 | action = "PLAY" 170 | putExtra("setTime", setStartTime) 171 | putExtra("forReplay", isPausingState) 172 | } 173 | 174 | return PendingIntent.getService(context, 29, playIntent, PendingIntent.FLAG_UPDATE_CURRENT) 175 | } 176 | 177 | class Builder(private val context: Context) { 178 | 179 | fun setSmallIcon(icon: Int): Builder { 180 | notiIcon = icon 181 | return this 182 | } 183 | 184 | fun setContentTitle(title: CharSequence): Builder { 185 | notiTitle = title 186 | return this 187 | } 188 | 189 | fun setShowWhen(show: Boolean): Builder { 190 | showWhen = show 191 | return this 192 | } 193 | 194 | fun setColor(color: Int): Builder { 195 | notiColor = color 196 | return this 197 | } 198 | 199 | fun setPriority(priority: Int): Builder { 200 | notificationPriority = priority 201 | return this 202 | } 203 | 204 | fun setAutoCancel(autoCancel: Boolean): Builder { 205 | isAutoCancel = autoCancel 206 | return this 207 | } 208 | 209 | fun setOnlyAlertOnce(onlyAlertOnce: Boolean): Builder { 210 | isOnlyAlertOnce = onlyAlertOnce 211 | return this 212 | } 213 | 214 | fun setControlMode(controlMode: Boolean): Builder { 215 | isControlMode = controlMode 216 | return this 217 | } 218 | 219 | fun setContentIntent(intent: PendingIntent): Builder { 220 | contentPendingIntent = intent 221 | return this 222 | } 223 | 224 | fun setPlayButtonIcon(icon: Int): Builder { 225 | playBtnIcon = icon 226 | return this 227 | } 228 | 229 | fun setPauseButtonIcon(icon: Int): Builder { 230 | pauseBtnIcon = icon 231 | return this 232 | } 233 | 234 | fun setStopButtonIcon(icon: Int): Builder { 235 | stopBtnIcon = icon 236 | return this 237 | } 238 | 239 | fun setOnFinishListener(listener: onFinishListener): Builder { 240 | finishListener = listener 241 | return this 242 | } 243 | 244 | fun setOnTickListener(listener: onTickListener): Builder { 245 | tickListener = listener 246 | return this 247 | } 248 | 249 | fun play(timeMillis: Long) = play(context, timeMillis) 250 | 251 | fun pause() = pause(context) 252 | 253 | fun stop() = stop(context) 254 | 255 | fun terminate() = terminate(context) 256 | } 257 | } -------------------------------------------------------------------------------- /NotificationTimer/src/main/java/io/joon/notificationtimer/TimerService.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notificationtimer 2 | 3 | import android.app.Service 4 | import android.content.Intent 5 | import android.os.* 6 | 7 | enum class TimerState { STOPPED, PAUSED, RUNNING, TERMINATED } 8 | 9 | class TimerService: Service() { 10 | 11 | companion object { 12 | var state = TimerState.TERMINATED 13 | } 14 | 15 | private lateinit var timer: CountDownTimer 16 | 17 | private val foreGroundId = 55 18 | private var secondsRemaining: Long = 0 19 | private var setTime:Long = 0 20 | private lateinit var showTime: String 21 | 22 | override fun onBind(intent: Intent): IBinder? = null 23 | 24 | override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { 25 | if (intent != null) { 26 | when (intent.action) { 27 | "PLAY" -> { 28 | playTimer( 29 | intent.getLongExtra("setTime", 0L), 30 | intent.getBooleanExtra("forReplay", false)) 31 | } 32 | "PAUSE" -> pauseTimer() 33 | "STOP" -> stopTimer() 34 | "TERMINATE" -> terminateTimer() 35 | } 36 | } 37 | return START_NOT_STICKY 38 | } 39 | 40 | override fun onTaskRemoved(rootIntent: Intent?) { 41 | super.onTaskRemoved(rootIntent) 42 | 43 | if (::timer.isInitialized) { 44 | timer.cancel() 45 | state = TimerState.TERMINATED 46 | } 47 | NotificationTimer.removeNotification() 48 | stopSelf() 49 | } 50 | 51 | private fun playTimer(setTime: Long, isReplay: Boolean) { 52 | 53 | if (!isReplay) { 54 | this.setTime = setTime 55 | secondsRemaining = setTime 56 | startForeground(foreGroundId, NotificationTimer.createNotification(this, setTime)) 57 | } 58 | 59 | timer = object : CountDownTimer(secondsRemaining, 1000) { 60 | override fun onFinish() { 61 | state = TimerState.STOPPED 62 | //초기 세팅됬었던 카운트다운 시간값을 노티에 재세팅 63 | val minutesUntilFinished = setTime/1000 / 60 64 | val secondsInMinuteUntilFinished = ((setTime/1000) - minutesUntilFinished * 60) 65 | val secondsStr = secondsInMinuteUntilFinished.toString() 66 | val showTime = "$minutesUntilFinished : ${if (secondsStr.length == 2) secondsStr else "0$secondsStr"}" 67 | NotificationTimer.updateStopState(this@TimerService, showTime, true) 68 | } 69 | 70 | override fun onTick(millisUntilFinished: Long) { 71 | NotificationTimer.updateUntilFinished(millisUntilFinished + (1000-(millisUntilFinished%1000)) - 1000) 72 | secondsRemaining = millisUntilFinished 73 | updateCountdownUI() 74 | } 75 | }.start() 76 | 77 | state = TimerState.RUNNING 78 | } 79 | 80 | private fun pauseTimer() { 81 | if (::timer.isInitialized) { 82 | timer.cancel() 83 | state = TimerState.PAUSED 84 | NotificationTimer.updatePauseState(this, showTime) 85 | } 86 | } 87 | 88 | private fun stopTimer() { 89 | if (::timer.isInitialized) { 90 | timer.cancel() 91 | state = TimerState.STOPPED 92 | val minutesUntilFinished = setTime/1000 / 60 93 | val secondsInMinuteUntilFinished = ((setTime/1000) - minutesUntilFinished * 60) 94 | val secondsStr = secondsInMinuteUntilFinished.toString() 95 | val showTime = "$minutesUntilFinished : ${if (secondsStr.length == 2) secondsStr else "0$secondsStr"}" 96 | NotificationTimer.updateStopState(this@TimerService, showTime) 97 | } 98 | } 99 | 100 | private fun terminateTimer() { 101 | if (::timer.isInitialized) { 102 | timer.cancel() 103 | state = TimerState.TERMINATED 104 | NotificationTimer.removeNotification() 105 | stopSelf() 106 | } 107 | } 108 | 109 | private fun updateCountdownUI() { 110 | val minutesUntilFinished = (secondsRemaining/1000) / 60 111 | val secondsInMinuteUntilFinished = ((secondsRemaining/1000) - minutesUntilFinished * 60) 112 | val secondsStr = secondsInMinuteUntilFinished.toString() 113 | showTime = "$minutesUntilFinished : ${if (secondsStr.length == 2) secondsStr else "0$secondsStr"}" 114 | 115 | NotificationTimer.updateTimeLeft(this, showTime) 116 | } 117 | 118 | } -------------------------------------------------------------------------------- /NotificationTimer/src/test/java/io/joon/notificationtimer/ExampleUnitTest.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notificationtimer 2 | 3 | import org.junit.Test 4 | 5 | import org.junit.Assert.* 6 | 7 | /** 8 | * Example local unit test, which will execute on the development machine (host). 9 | * 10 | * See [testing documentation](http://d.android.com/tools/testing). 11 | */ 12 | class ExampleUnitTest { 13 | @Test 14 | fun addition_isCorrect() { 15 | assertEquals(4, 2 + 2) 16 | } 17 | } -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # NotificationTimer 2 | Timer in Notification for Android that supports play, pause, stop by buttons. 3 |
4 |
5 | 6 | Demo 7 | ------- 8 | 9 | 10 |
11 |
12 | 13 | Dependency 14 | -------------- 15 | Add this in your root ```build.gradle``` file (not your module ```build.gradle``` file) 16 | ``` 17 | allprojects { 18 | repositories { 19 | ... 20 | maven { url 'https://jitpack.io' } 21 | } 22 | } 23 | ``` 24 | Then, Add the library to your module ```build.gradle``` 25 | ``` 26 | dependencies { 27 | implementation 'com.github.Jun-Hub:NotificationTimer:latest.release.version' 28 | } 29 | ``` 30 |
31 |
32 | 33 | Features 34 | ---------------- 35 | * This is Singleton Pattern which means you can't create more than one Timer. 36 | * There are 2 types of Notification. One is 'default mode' that show a simple timer. The other one is 'control mode' that show with control buttons(play, pause, stop) 37 | 38 |
39 |
40 | 41 | Usage 42 | -------------- 43 | Add this in your ```manifest``` file 44 | ``` 45 | 46 | 47 | 48 | ``` 49 | And simple usage: 50 | 51 | ``` kotlin 52 | NotificationTimer.Builder(context) 53 | .setSmallIcon(R.drawable.ic_timer) 54 | .play(timeMillis) 55 | ``` 56 | control mode usage: 57 | 58 | ``` kotlin 59 | NotificationTimer.Builder(context) 60 | .setSmallIcon(R.drawable.ic_timer) 61 | .setPlayButtonIcon(R.drawable.ic_play_noti) //*required. (If you set ControlMode as 'true') 62 | .setPauseButtonIcon(R.drawable.ic_pause_noti) //*required. (If you set ControlMode as 'true') 63 | .setStopButtonIcon(R.drawable.ic_stop_noti) //*required. (If you set ControlMode as 'true') 64 | .setControlMode(true) 65 | .play(timeMillis) 66 | ``` 67 |
68 |
69 | 70 | Detail Usage 71 | ------------------ 72 | ```kotlin 73 | val notiTimer = NotificationTimer.Builder(context) 74 | 75 | .setSmallIcon(R.drawable.ic_timer) //*required 76 | 77 | .setPlayButtonIcon(R.drawable.ic_play_noti) //should use xml file (not jpg or png something) 78 | 79 | .setPauseButtonIcon(R.drawable.ic_pause_noti) //should use xml file (not jpg or png something) 80 | 81 | .setStopButtonIcon(R.drawable.ic_stop_noti) //should use xml file (not jpg or png something) 82 | 83 | .setControlMode(true) //set as control mode(with play, pause, stop buttons) default value is 'false' 84 | 85 | .setColor(R.color.sexy_blue) //same as NotificationCompat. 86 | 87 | .setShowWhen(false) //same as NotificationCompat. default value is 'false' 88 | 89 | .setAutoCancel(false) //same as NotificationCompat. default value is 'false' 90 | 91 | .setOnlyAlertOnce(true) //same as NotificationCompat. default value is 'true' 92 | 93 | .setPriority(NotificationCompat.PRIORITY_LOW) //same as NotificationCompat. default value is 'PRIORITY_LOW' 94 | 95 | .setContentIntent(pendingIntent) //same as NotificationCompat. 96 | 97 | .setContentTitle("This is NotiTimer!!") //same as NotificationCompat. 98 | 99 | .setOnTickListener { untilFinish -> //callback timeMillis until finished 100 | time_until_finish_text.text = untilFinish.toString() 101 | } 102 | 103 | .setOnFinishListener { //callback when timer finished 104 | Toast.makeText(this, "Timer finished", Toast.LENGTH_SHORT).show() 105 | } 106 | 107 | notiTimer.play(timeMillis) //play Timer. But when Timer state is Paused, it will execute 'Replay' by itself. 108 | 109 | notiTimer.pause() //pause Timer. 110 | 111 | notiTimer.stop() //stop Timer. And set time which is you set before. 112 | 113 | notiTimer.terminate() //terminate Timer. And remove Notification as well. 114 | ``` 115 | What if you want current Timer state?: 116 | 117 | ```kotlin 118 | if(TimerService.state==TimerState.RUNNING) { 119 | //TODO your code 120 | } 121 | 122 | if(TimerService.state==TimerState.PAUSED) { 123 | //TODO your code 124 | } 125 | 126 | if(TimerService.state==TimerState.STOPPED) { 127 | //TODO your code 128 | } 129 | 130 | if(TimerService.state==TimerState.TERMINATED) { 131 | //TODO your code 132 | } 133 | ``` 134 |
135 |
136 | 137 | License 138 | ----------- 139 | 140 | Copyright 2021 Joon Lee 141 | 142 | Licensed under the Apache License, Version 2.0 (the "License"); 143 | you may not use this file except in compliance with the License. 144 | You may obtain a copy of the License at 145 | 146 | http://www.apache.org/licenses/LICENSE-2.0 147 | 148 | Unless required by applicable law or agreed to in writing, software 149 | distributed under the License is distributed on an "AS IS" BASIS, 150 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 151 | See the License for the specific language governing permissions and 152 | limitations under the License. 153 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | plugins { 2 | id 'com.android.application' 3 | id 'kotlin-android' 4 | id 'kotlin-android-extensions' 5 | } 6 | 7 | android { 8 | compileSdkVersion 29 9 | 10 | defaultConfig { 11 | applicationId "io.joon.notitimer" 12 | minSdkVersion 21 13 | targetSdkVersion 29 14 | versionCode 1 15 | versionName "1.0" 16 | 17 | testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" 18 | } 19 | 20 | buildTypes { 21 | release { 22 | minifyEnabled false 23 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' 24 | } 25 | } 26 | compileOptions { 27 | sourceCompatibility JavaVersion.VERSION_1_8 28 | targetCompatibility JavaVersion.VERSION_1_8 29 | } 30 | kotlinOptions { 31 | jvmTarget = '1.8' 32 | } 33 | } 34 | 35 | dependencies { 36 | 37 | implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" 38 | implementation 'androidx.core:core-ktx:1.3.2' 39 | implementation 'androidx.appcompat:appcompat:1.2.0' 40 | implementation 'com.google.android.material:material:1.2.1' 41 | implementation 'androidx.constraintlayout:constraintlayout:2.0.4' 42 | implementation project(path: ':NotificationTimer') 43 | testImplementation 'junit:junit:4.+' 44 | androidTestImplementation 'androidx.test.ext:junit:1.1.2' 45 | androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0' 46 | } -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /app/src/androidTest/java/io/joon/notitimer/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notitimer 2 | 3 | import androidx.test.platform.app.InstrumentationRegistry 4 | import androidx.test.ext.junit.runners.AndroidJUnit4 5 | 6 | import org.junit.Test 7 | import org.junit.runner.RunWith 8 | 9 | import org.junit.Assert.* 10 | 11 | /** 12 | * Instrumented test, which will execute on an Android device. 13 | * 14 | * See [testing documentation](http://d.android.com/tools/testing). 15 | */ 16 | @RunWith(AndroidJUnit4::class) 17 | class ExampleInstrumentedTest { 18 | @Test 19 | fun useAppContext() { 20 | // Context of the app under test. 21 | val appContext = InstrumentationRegistry.getInstrumentation().targetContext 22 | assertEquals("io.joon.notitimer", appContext.packageName) 23 | } 24 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 14 | 15 | 16 | 17 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /app/src/main/java/io/joon/notitimer/MainActivity.kt: -------------------------------------------------------------------------------- 1 | package io.joon.notitimer 2 | 3 | import android.app.PendingIntent 4 | import android.content.Intent 5 | import androidx.appcompat.app.AppCompatActivity 6 | import android.os.Bundle 7 | import android.widget.Toast 8 | import androidx.core.app.NotificationCompat 9 | import io.joon.notificationtimer.NotificationTimer 10 | import kotlinx.android.synthetic.main.activity_main.* 11 | 12 | class MainActivity : AppCompatActivity() { 13 | 14 | lateinit var notiTimer: NotificationTimer.Builder 15 | 16 | override fun onCreate(savedInstanceState: Bundle?) { 17 | super.onCreate(savedInstanceState) 18 | setContentView(R.layout.activity_main) 19 | 20 | val pendingIntent = Intent(this, MainActivity::class.java).let{ 21 | PendingIntent.getActivity(this, 0, it, PendingIntent.FLAG_UPDATE_CURRENT) 22 | } 23 | 24 | build_btn.setOnClickListener { 25 | notiTimer = NotificationTimer.Builder(this) 26 | .setSmallIcon(R.drawable.ic_timer) 27 | .setPlayButtonIcon(R.drawable.ic_play_noti) 28 | .setPauseButtonIcon(R.drawable.ic_pause_noti) 29 | .setStopButtonIcon(R.drawable.ic_stop_noti) 30 | .setControlMode(true) 31 | .setColor(R.color.sexy_blue) 32 | .setShowWhen(false) 33 | .setAutoCancel(false) 34 | .setOnlyAlertOnce(true) 35 | .setPriority(NotificationCompat.PRIORITY_LOW) 36 | .setContentIntent(pendingIntent) 37 | .setOnTickListener { time_until_finish_text.text = it.toString() } 38 | .setOnFinishListener { Toast.makeText(this, "timer finished", Toast.LENGTH_SHORT).show() } 39 | .setContentTitle("Timer :)") 40 | } 41 | 42 | play_btn.setOnClickListener { 43 | notiTimer.play(time_editText.text.toString().toLong()) 44 | } 45 | 46 | pause_btn.setOnClickListener { 47 | notiTimer.pause() 48 | } 49 | 50 | stop_btn.setOnClickListener { 51 | notiTimer.stop() 52 | time_until_finish_text.text = null 53 | } 54 | 55 | terminate_btn.setOnClickListener { 56 | notiTimer.terminate() 57 | } 58 | } 59 | } -------------------------------------------------------------------------------- /app/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 7 | 8 | 9 | 15 | 18 | 21 | 22 | 23 | 24 | 30 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_launcher_background.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 10 | 15 | 20 | 25 | 30 | 35 | 40 | 45 | 50 | 55 | 60 | 65 | 70 | 75 | 80 | 85 | 90 | 95 | 100 | 105 | 110 | 115 | 120 | 125 | 130 | 135 | 140 | 145 | 150 | 155 | 160 | 165 | 170 | 171 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_pause_noti.xml: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_play_noti.xml: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_stop_noti.xml: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_timer.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Jun-Hub/NotificationTimer/1625c0ce12a94ce8b43b4b7583225366f53b7fd9/app/src/main/res/drawable/ic_timer.jpeg -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 21 | 22 |