2021-10-25 15:01:10 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2020 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
* http://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.
|
|
|
|
*/
|
|
|
|
|
2021-10-26 20:35:51 +02:00
|
|
|
package io.heckel.ntfy
|
2021-10-25 15:01:10 +02:00
|
|
|
|
|
|
|
import android.app.Activity
|
2021-10-26 02:25:54 +02:00
|
|
|
import android.app.NotificationChannel
|
|
|
|
import android.app.NotificationManager
|
|
|
|
import android.content.Context
|
2021-10-25 15:01:10 +02:00
|
|
|
import android.content.Intent
|
2021-10-26 02:25:54 +02:00
|
|
|
import android.os.Build
|
2021-10-25 15:01:10 +02:00
|
|
|
import android.os.Bundle
|
|
|
|
import android.view.View
|
|
|
|
import androidx.activity.viewModels
|
2021-10-25 20:24:44 +02:00
|
|
|
import androidx.appcompat.app.AppCompatActivity
|
2021-10-26 02:25:54 +02:00
|
|
|
import androidx.core.app.NotificationCompat
|
|
|
|
import androidx.core.app.NotificationManagerCompat
|
2021-10-25 22:16:23 +02:00
|
|
|
import androidx.lifecycle.lifecycleScope
|
2021-10-25 15:01:10 +02:00
|
|
|
import androidx.recyclerview.widget.RecyclerView
|
2021-10-26 18:23:41 +02:00
|
|
|
import com.google.gson.GsonBuilder
|
|
|
|
import com.google.gson.JsonObject
|
|
|
|
import com.google.gson.JsonSyntaxException
|
2021-10-25 20:24:44 +02:00
|
|
|
import io.heckel.ntfy.add.AddTopicActivity
|
2021-10-25 15:01:10 +02:00
|
|
|
import io.heckel.ntfy.data.Topic
|
2021-10-26 20:40:52 +02:00
|
|
|
import io.heckel.ntfy.detail.DetailActivity
|
2021-10-26 20:35:51 +02:00
|
|
|
import io.heckel.ntfy.list.TopicsAdapter
|
|
|
|
import io.heckel.ntfy.list.TopicsViewModel
|
2021-10-26 20:40:52 +02:00
|
|
|
import io.heckel.ntfy.list.TopicsViewModelFactory
|
2021-10-26 18:23:41 +02:00
|
|
|
import kotlinx.coroutines.*
|
|
|
|
import java.io.IOException
|
|
|
|
import java.net.HttpURLConnection
|
|
|
|
import java.net.URL
|
2021-10-26 02:25:54 +02:00
|
|
|
import kotlin.random.Random
|
2021-10-25 15:01:10 +02:00
|
|
|
|
2021-10-25 19:45:56 +02:00
|
|
|
const val TOPIC_ID = "topic id"
|
2021-10-26 20:40:52 +02:00
|
|
|
const val TOPIC_URL = "url"
|
2021-10-25 15:01:10 +02:00
|
|
|
|
2021-10-26 20:35:51 +02:00
|
|
|
class MainActivity : AppCompatActivity() {
|
2021-10-26 19:46:49 +02:00
|
|
|
private val gson = GsonBuilder().create()
|
2021-10-26 18:23:41 +02:00
|
|
|
private val jobs = mutableMapOf<Long, Job>()
|
2021-10-25 15:01:10 +02:00
|
|
|
private val newTopicActivityRequestCode = 1
|
2021-10-26 20:35:51 +02:00
|
|
|
private val topicsListViewModel by viewModels<TopicsViewModel> {
|
|
|
|
TopicsViewModelFactory(this)
|
2021-10-25 15:01:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
|
|
super.onCreate(savedInstanceState)
|
|
|
|
setContentView(R.layout.activity_main)
|
|
|
|
|
2021-10-25 19:45:56 +02:00
|
|
|
val adapter = TopicsAdapter { topic -> adapterOnClick(topic) }
|
2021-10-25 15:01:10 +02:00
|
|
|
val recyclerView: RecyclerView = findViewById(R.id.recycler_view)
|
2021-10-25 19:45:56 +02:00
|
|
|
recyclerView.adapter = adapter
|
2021-10-25 15:01:10 +02:00
|
|
|
|
2021-10-26 03:14:09 +02:00
|
|
|
topicsListViewModel.topics.observe(this) {
|
2021-10-25 15:01:10 +02:00
|
|
|
it?.let {
|
2021-10-25 19:45:56 +02:00
|
|
|
adapter.submitList(it as MutableList<Topic>)
|
2021-10-25 15:01:10 +02:00
|
|
|
}
|
2021-10-25 19:45:56 +02:00
|
|
|
}
|
2021-10-25 15:01:10 +02:00
|
|
|
|
|
|
|
val fab: View = findViewById(R.id.fab)
|
|
|
|
fab.setOnClickListener {
|
|
|
|
fabOnClick()
|
|
|
|
}
|
2021-10-25 22:16:23 +02:00
|
|
|
|
2021-10-26 02:25:54 +02:00
|
|
|
createNotificationChannel()
|
2021-10-26 03:14:09 +02:00
|
|
|
}
|
2021-10-26 02:25:54 +02:00
|
|
|
|
2021-10-25 19:45:56 +02:00
|
|
|
/* Opens TopicDetailActivity when RecyclerView item is clicked. */
|
|
|
|
private fun adapterOnClick(topic: Topic) {
|
2021-10-26 20:40:52 +02:00
|
|
|
val intent = Intent(this, DetailActivity()::class.java)
|
2021-10-25 19:45:56 +02:00
|
|
|
intent.putExtra(TOPIC_ID, topic.id)
|
2021-10-25 15:01:10 +02:00
|
|
|
startActivity(intent)
|
|
|
|
}
|
|
|
|
|
2021-10-25 19:45:56 +02:00
|
|
|
/* Adds topic to topicList when FAB is clicked. */
|
2021-10-25 15:01:10 +02:00
|
|
|
private fun fabOnClick() {
|
|
|
|
val intent = Intent(this, AddTopicActivity::class.java)
|
|
|
|
startActivityForResult(intent, newTopicActivityRequestCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun onActivityResult(requestCode: Int, resultCode: Int, intentData: Intent?) {
|
|
|
|
super.onActivityResult(requestCode, resultCode, intentData)
|
|
|
|
|
2021-10-25 19:45:56 +02:00
|
|
|
/* Inserts topic into viewModel. */
|
2021-10-25 15:01:10 +02:00
|
|
|
if (requestCode == newTopicActivityRequestCode && resultCode == Activity.RESULT_OK) {
|
|
|
|
intentData?.let { data ->
|
2021-10-26 18:23:41 +02:00
|
|
|
val topicId = Random.nextLong()
|
2021-10-26 03:14:09 +02:00
|
|
|
val topicUrl = data.getStringExtra(TOPIC_URL) ?: return
|
2021-10-26 18:23:41 +02:00
|
|
|
val topic = Topic(topicId, topicUrl)
|
|
|
|
|
2021-10-26 19:46:49 +02:00
|
|
|
jobs[topicId] = subscribeTopic(topicUrl)
|
2021-10-26 18:23:41 +02:00
|
|
|
topicsListViewModel.add(topic)
|
2021-10-25 15:01:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-26 02:25:54 +02:00
|
|
|
|
2021-10-26 19:46:49 +02:00
|
|
|
private fun subscribeTopic(url: String): Job {
|
|
|
|
return this.lifecycleScope.launch(Dispatchers.IO) {
|
|
|
|
while (isActive) {
|
|
|
|
openURL(this, url)
|
|
|
|
delay(5000) // TODO exponential back-off
|
2021-10-26 02:25:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-26 19:46:49 +02:00
|
|
|
private fun openURL(scope: CoroutineScope, url: String) {
|
|
|
|
println("Connecting to $url ...")
|
|
|
|
val conn = (URL(url).openConnection() as HttpURLConnection).also {
|
|
|
|
it.doInput = true
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
val input = conn.inputStream.bufferedReader()
|
|
|
|
while (scope.isActive) {
|
2021-10-26 20:40:52 +02:00
|
|
|
val line = input.readLine() ?: break // Exit if null
|
2021-10-26 18:23:41 +02:00
|
|
|
try {
|
2021-10-26 20:40:52 +02:00
|
|
|
val json = gson.fromJson(line, JsonObject::class.java) ?: break // Exit if null
|
|
|
|
displayNotification(json)
|
2021-10-26 19:46:49 +02:00
|
|
|
} catch (e: JsonSyntaxException) {
|
|
|
|
// Ignore invalid JSON
|
2021-10-26 18:23:41 +02:00
|
|
|
}
|
|
|
|
}
|
2021-10-26 19:46:49 +02:00
|
|
|
} catch (e: IOException) {
|
|
|
|
println("PHIL: " + e.message)
|
|
|
|
} finally {
|
|
|
|
conn.disconnect()
|
2021-10-26 18:23:41 +02:00
|
|
|
}
|
2021-10-26 19:46:49 +02:00
|
|
|
println("Connection terminated: $url")
|
2021-10-26 18:23:41 +02:00
|
|
|
}
|
|
|
|
|
2021-10-26 19:46:49 +02:00
|
|
|
private fun displayNotification(json: JsonObject) {
|
|
|
|
if (json.isJsonNull || !json.has("message")) {
|
2021-10-26 18:23:41 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
val channelId = getString(R.string.notification_channel_id)
|
|
|
|
val notification = NotificationCompat.Builder(this, channelId)
|
|
|
|
.setSmallIcon(R.drawable.ntfy)
|
|
|
|
.setContentTitle("ntfy")
|
2021-10-26 19:46:49 +02:00
|
|
|
.setContentText(json.get("message").asString)
|
2021-10-26 18:23:41 +02:00
|
|
|
.setPriority(NotificationCompat.PRIORITY_DEFAULT)
|
|
|
|
.build()
|
|
|
|
with(NotificationManagerCompat.from(this)) {
|
|
|
|
notify(Random.nextInt(), notification)
|
|
|
|
}
|
|
|
|
}
|
2021-10-26 19:46:49 +02:00
|
|
|
|
|
|
|
private fun createNotificationChannel() {
|
|
|
|
// Create the NotificationChannel, but only on API 26+ because
|
|
|
|
// the NotificationChannel class is new and not in the support library
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
|
|
|
val channelId = getString(R.string.notification_channel_id)
|
|
|
|
val name = getString(R.string.notification_channel_name)
|
|
|
|
val descriptionText = getString(R.string.notification_channel_name)
|
|
|
|
val importance = NotificationManager.IMPORTANCE_DEFAULT
|
|
|
|
val channel = NotificationChannel(channelId, name, importance).apply {
|
|
|
|
description = descriptionText
|
|
|
|
}
|
|
|
|
// Register the channel with the system
|
|
|
|
val notificationManager: NotificationManager =
|
|
|
|
getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
|
|
|
|
notificationManager.createNotificationChannel(channel)
|
|
|
|
}
|
|
|
|
}
|
2021-10-25 15:01:10 +02:00
|
|
|
}
|