Loading app/src/main/java/io/heckel/ntfy/db/Repository.kt +14 −0 Original line number Diff line number Diff line Loading @@ -301,6 +301,16 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas .apply() } fun getJsonStreamRemindTime(): Long { return sharedPrefs.getLong(SHARED_PREFS_JSON_STREAM_REMIND_TIME, JSON_STREAM_REMIND_TIME_ALWAYS) } fun setJsonStreamRemindTime(timeMillis: Long) { sharedPrefs.edit() .putLong(SHARED_PREFS_JSON_STREAM_REMIND_TIME, timeMillis) .apply() } fun getDefaultBaseUrl(): String? { return sharedPrefs.getString(SHARED_PREFS_DEFAULT_BASE_URL, null) ?: sharedPrefs.getString(SHARED_PREFS_UNIFIED_PUSH_BASE_URL, null) // Fall back to UP URL, removed when default is set! Loading Loading @@ -434,6 +444,7 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas const val SHARED_PREFS_BROADCAST_ENABLED = "BroadcastEnabled" const val SHARED_PREFS_RECORD_LOGS_ENABLED = "RecordLogs" const val SHARED_PREFS_BATTERY_OPTIMIZATIONS_REMIND_TIME = "BatteryOptimizationsRemindTime" const val SHARED_PREFS_JSON_STREAM_REMIND_TIME = "JsonStreamRemindTime" // Deprecation of JSON stream const val SHARED_PREFS_UNIFIED_PUSH_BASE_URL = "UnifiedPushBaseURL" // Legacy key required for migration to DefaultBaseURL const val SHARED_PREFS_DEFAULT_BASE_URL = "DefaultBaseURL" const val SHARED_PREFS_LAST_TOPICS = "LastTopics" Loading Loading @@ -464,6 +475,9 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas const val BATTERY_OPTIMIZATIONS_REMIND_TIME_ALWAYS = 1L const val BATTERY_OPTIMIZATIONS_REMIND_TIME_NEVER = Long.MAX_VALUE const val JSON_STREAM_REMIND_TIME_ALWAYS = 1L const val JSON_STREAM_REMIND_TIME_NEVER = Long.MAX_VALUE private const val TAG = "NtfyRepository" private var instance: Repository? = null Loading app/src/main/java/io/heckel/ntfy/ui/MainActivity.kt +28 −1 Original line number Diff line number Diff line Loading @@ -119,8 +119,9 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc Log.addScrubTerm(s.topic) } // Update battery banner // Update banner + JSON stream banner showHideBatteryBanner(subscriptions) showHideJsonStreamBanner(subscriptions) } } Loading Loading @@ -168,6 +169,23 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc } } // JSON stream banner val jsonStreamBanner = findViewById<View>(R.id.main_banner_json_stream) // Banner visibility is toggled in onResume() val jsonStreamDismissButton = findViewById<Button>(R.id.main_banner_json_stream_dontaskagain) val jsonStreamRemindButton = findViewById<Button>(R.id.main_banner_json_stream_remind_later) val jsonStreamLearnMoreButton = findViewById<Button>(R.id.main_banner_json_stream_learn_mode) jsonStreamDismissButton.setOnClickListener { jsonStreamBanner.visibility = View.GONE repository.setJsonStreamRemindTime(Repository.JSON_STREAM_REMIND_TIME_NEVER) } jsonStreamRemindButton.setOnClickListener { jsonStreamBanner.visibility = View.GONE repository.setJsonStreamRemindTime(System.currentTimeMillis() + ONE_DAY_MILLIS) } jsonStreamLearnMoreButton.setOnClickListener { startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(getString(R.string.main_banner_json_stream_button_learn_more_url)))) } // Create notification channels right away, so we can configure them immediately after installing the app dispatcher?.init() Loading Loading @@ -199,6 +217,15 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc Log.d(TAG, "Battery: ignoring optimizations = $ignoringOptimizations (we want this to be true); instant subscriptions = $hasInstantSubscriptions; remind time reached = $batteryRemindTimeReached; banner = $showBanner") } private fun showHideJsonStreamBanner(subscriptions: List<Subscription>) { val hasSelfhostedSubscriptions = subscriptions.count { it.baseUrl != appBaseUrl } > 0 val usingWebSockets = repository.getConnectionProtocol() == Repository.CONNECTION_PROTOCOL_WS val jsonStreamRemindTimeReached = repository.getJsonStreamRemindTime() < System.currentTimeMillis() val showBanner = hasSelfhostedSubscriptions && jsonStreamRemindTimeReached && !usingWebSockets val jsonStreamBanner = findViewById<View>(R.id.main_banner_json_stream) jsonStreamBanner.visibility = if (showBanner) View.VISIBLE else View.GONE } private fun schedulePeriodicPollWorker() { val workerVersion = repository.getPollWorkerVersion() val workPolicy = if (workerVersion == PollWorker.VERSION) { Loading app/src/main/res/drawable/ic_announcement_orange_24dp.xml 0 → 100644 +9 −0 Original line number Diff line number Diff line <vector xmlns:android="http://schemas.android.com/apk/res/android" android:width="24dp" android:height="24dp" android:viewportWidth="24" android:viewportHeight="24"> <path android:pathData="M20,2L4,2c-1.1,0 -2,0.9 -2,2v18l4,-4h14c1.1,0 2,-0.9 2,-2L22,4c0,-1.1 -0.9,-2 -2,-2zM20,16L5.17,16L4,17.17L4,4h16v12zM11,5h2v6h-2zM11,13h2v2h-2z" android:fillColor="#FF9800"/> </vector> app/src/main/res/layout/activity_main.xml +74 −3 Original line number Diff line number Diff line <?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" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent"> <com.google.android.material.card.MaterialCardView Loading Loading @@ -44,7 +44,7 @@ android:layout_height="wrap_content" android:layout_marginTop="5dp" android:layout_marginEnd="5dp" android:text="@string/main_banner_battery_button_ask_later" android:text="@string/main_banner_battery_button_remind_later" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toStartOf="@id/main_banner_battery_fix_now" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery_text" Loading Loading @@ -77,6 +77,77 @@ </androidx.constraintlayout.widget.ConstraintLayout> </com.google.android.material.card.MaterialCardView> <com.google.android.material.card.MaterialCardView android:layout_width="match_parent" android:layout_height="wrap_content" app:shapeAppearance="?shapeAppearanceLargeComponent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery" android:id="@+id/main_banner_json_stream" android:visibility="visible"> <androidx.constraintlayout.widget.ConstraintLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:id="@+id/main_banner_json_stream_constraint" android:elevation="5dp"> <ImageView android:layout_width="28dp" android:layout_height="28dp" app:srcCompat="@drawable/ic_announcement_orange_24dp" android:id="@+id/main_banner_json_stream_image" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="@+id/main_banner_json_stream_text" app:layout_constraintEnd_toStartOf="@id/main_banner_json_stream_text" app:layout_constraintBottom_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginStart="15dp"/> <TextView android:id="@+id/main_banner_json_stream_text" android:layout_width="0dp" android:layout_height="wrap_content" android:text="@string/main_banner_json_stream_text" android:textAppearance="@style/TextAppearance.AppCompat.Small" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toTopOf="parent" android:layout_marginEnd="15dp" android:layout_marginTop="15dp" app:layout_constraintStart_toEndOf="@+id/main_banner_json_stream_image" android:layout_marginStart="10dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_remind_later" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:text="@string/main_banner_json_stream_button_remind_later" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp" app:layout_constraintEnd_toStartOf="@+id/main_banner_json_stream_learn_mode" android:layout_marginEnd="5dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_dontaskagain" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:layout_marginEnd="5dp" android:text="@string/main_banner_json_stream_button_dismiss" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toStartOf="@id/main_banner_json_stream_remind_later" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_learn_mode" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:text="@string/main_banner_json_stream_button_learn_more" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp" app:layout_constraintEnd_toEndOf="parent" android:layout_marginEnd="15dp"/> </androidx.constraintlayout.widget.ConstraintLayout> </com.google.android.material.card.MaterialCardView> <androidx.swiperefreshlayout.widget.SwipeRefreshLayout android:id="@+id/main_subscriptions_list_container" android:layout_width="match_parent" Loading @@ -85,7 +156,7 @@ android:visibility="visible" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery"> app:layout_constraintTop_toBottomOf="@id/main_banner_json_stream"> <androidx.recyclerview.widget.RecyclerView android:id="@+id/main_subscriptions_list" android:layout_width="match_parent" Loading app/src/main/res/values/strings.xml +9 −2 Original line number Diff line number Diff line Loading @@ -69,10 +69,17 @@ <!-- Main activity: Battery banner --> <string name="main_banner_battery_text">Battery optimization should be disabled to avoid issues with notification delivery.</string> <string name="main_banner_battery_button_ask_later">Ask later</string> <string name="main_banner_battery_button_dismiss">Don\'t ask again</string> <string name="main_banner_battery_button_remind_later">Remind later</string> <string name="main_banner_battery_button_dismiss">Dismiss</string> <string name="main_banner_battery_button_fix_now">Fix now</string> <!-- Main activity: JSON stream banner --> <string name="main_banner_json_stream_text">Starting June 2022, WebSockets will be used to communicate with the server. Be sure get your selfhosted server ready.</string> <string name="main_banner_json_stream_button_remind_later">Remind later</string> <string name="main_banner_json_stream_button_dismiss">Dismiss</string> <string name="main_banner_json_stream_button_learn_more">Learn more</string> <string name="main_banner_json_stream_button_learn_more_url">https://ntfy.sh/docs/deprecations</string> <!-- Add dialog --> <string name="add_dialog_title">Subscribe to topic</string> <string name="add_dialog_description_below"> Loading Loading
app/src/main/java/io/heckel/ntfy/db/Repository.kt +14 −0 Original line number Diff line number Diff line Loading @@ -301,6 +301,16 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas .apply() } fun getJsonStreamRemindTime(): Long { return sharedPrefs.getLong(SHARED_PREFS_JSON_STREAM_REMIND_TIME, JSON_STREAM_REMIND_TIME_ALWAYS) } fun setJsonStreamRemindTime(timeMillis: Long) { sharedPrefs.edit() .putLong(SHARED_PREFS_JSON_STREAM_REMIND_TIME, timeMillis) .apply() } fun getDefaultBaseUrl(): String? { return sharedPrefs.getString(SHARED_PREFS_DEFAULT_BASE_URL, null) ?: sharedPrefs.getString(SHARED_PREFS_UNIFIED_PUSH_BASE_URL, null) // Fall back to UP URL, removed when default is set! Loading Loading @@ -434,6 +444,7 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas const val SHARED_PREFS_BROADCAST_ENABLED = "BroadcastEnabled" const val SHARED_PREFS_RECORD_LOGS_ENABLED = "RecordLogs" const val SHARED_PREFS_BATTERY_OPTIMIZATIONS_REMIND_TIME = "BatteryOptimizationsRemindTime" const val SHARED_PREFS_JSON_STREAM_REMIND_TIME = "JsonStreamRemindTime" // Deprecation of JSON stream const val SHARED_PREFS_UNIFIED_PUSH_BASE_URL = "UnifiedPushBaseURL" // Legacy key required for migration to DefaultBaseURL const val SHARED_PREFS_DEFAULT_BASE_URL = "DefaultBaseURL" const val SHARED_PREFS_LAST_TOPICS = "LastTopics" Loading Loading @@ -464,6 +475,9 @@ class Repository(private val sharedPrefs: SharedPreferences, private val databas const val BATTERY_OPTIMIZATIONS_REMIND_TIME_ALWAYS = 1L const val BATTERY_OPTIMIZATIONS_REMIND_TIME_NEVER = Long.MAX_VALUE const val JSON_STREAM_REMIND_TIME_ALWAYS = 1L const val JSON_STREAM_REMIND_TIME_NEVER = Long.MAX_VALUE private const val TAG = "NtfyRepository" private var instance: Repository? = null Loading
app/src/main/java/io/heckel/ntfy/ui/MainActivity.kt +28 −1 Original line number Diff line number Diff line Loading @@ -119,8 +119,9 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc Log.addScrubTerm(s.topic) } // Update battery banner // Update banner + JSON stream banner showHideBatteryBanner(subscriptions) showHideJsonStreamBanner(subscriptions) } } Loading Loading @@ -168,6 +169,23 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc } } // JSON stream banner val jsonStreamBanner = findViewById<View>(R.id.main_banner_json_stream) // Banner visibility is toggled in onResume() val jsonStreamDismissButton = findViewById<Button>(R.id.main_banner_json_stream_dontaskagain) val jsonStreamRemindButton = findViewById<Button>(R.id.main_banner_json_stream_remind_later) val jsonStreamLearnMoreButton = findViewById<Button>(R.id.main_banner_json_stream_learn_mode) jsonStreamDismissButton.setOnClickListener { jsonStreamBanner.visibility = View.GONE repository.setJsonStreamRemindTime(Repository.JSON_STREAM_REMIND_TIME_NEVER) } jsonStreamRemindButton.setOnClickListener { jsonStreamBanner.visibility = View.GONE repository.setJsonStreamRemindTime(System.currentTimeMillis() + ONE_DAY_MILLIS) } jsonStreamLearnMoreButton.setOnClickListener { startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(getString(R.string.main_banner_json_stream_button_learn_more_url)))) } // Create notification channels right away, so we can configure them immediately after installing the app dispatcher?.init() Loading Loading @@ -199,6 +217,15 @@ class MainActivity : AppCompatActivity(), ActionMode.Callback, AddFragment.Subsc Log.d(TAG, "Battery: ignoring optimizations = $ignoringOptimizations (we want this to be true); instant subscriptions = $hasInstantSubscriptions; remind time reached = $batteryRemindTimeReached; banner = $showBanner") } private fun showHideJsonStreamBanner(subscriptions: List<Subscription>) { val hasSelfhostedSubscriptions = subscriptions.count { it.baseUrl != appBaseUrl } > 0 val usingWebSockets = repository.getConnectionProtocol() == Repository.CONNECTION_PROTOCOL_WS val jsonStreamRemindTimeReached = repository.getJsonStreamRemindTime() < System.currentTimeMillis() val showBanner = hasSelfhostedSubscriptions && jsonStreamRemindTimeReached && !usingWebSockets val jsonStreamBanner = findViewById<View>(R.id.main_banner_json_stream) jsonStreamBanner.visibility = if (showBanner) View.VISIBLE else View.GONE } private fun schedulePeriodicPollWorker() { val workerVersion = repository.getPollWorkerVersion() val workPolicy = if (workerVersion == PollWorker.VERSION) { Loading
app/src/main/res/drawable/ic_announcement_orange_24dp.xml 0 → 100644 +9 −0 Original line number Diff line number Diff line <vector xmlns:android="http://schemas.android.com/apk/res/android" android:width="24dp" android:height="24dp" android:viewportWidth="24" android:viewportHeight="24"> <path android:pathData="M20,2L4,2c-1.1,0 -2,0.9 -2,2v18l4,-4h14c1.1,0 2,-0.9 2,-2L22,4c0,-1.1 -0.9,-2 -2,-2zM20,16L5.17,16L4,17.17L4,4h16v12zM11,5h2v6h-2zM11,13h2v2h-2z" android:fillColor="#FF9800"/> </vector>
app/src/main/res/layout/activity_main.xml +74 −3 Original line number Diff line number Diff line <?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" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent"> <com.google.android.material.card.MaterialCardView Loading Loading @@ -44,7 +44,7 @@ android:layout_height="wrap_content" android:layout_marginTop="5dp" android:layout_marginEnd="5dp" android:text="@string/main_banner_battery_button_ask_later" android:text="@string/main_banner_battery_button_remind_later" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toStartOf="@id/main_banner_battery_fix_now" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery_text" Loading Loading @@ -77,6 +77,77 @@ </androidx.constraintlayout.widget.ConstraintLayout> </com.google.android.material.card.MaterialCardView> <com.google.android.material.card.MaterialCardView android:layout_width="match_parent" android:layout_height="wrap_content" app:shapeAppearance="?shapeAppearanceLargeComponent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery" android:id="@+id/main_banner_json_stream" android:visibility="visible"> <androidx.constraintlayout.widget.ConstraintLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:id="@+id/main_banner_json_stream_constraint" android:elevation="5dp"> <ImageView android:layout_width="28dp" android:layout_height="28dp" app:srcCompat="@drawable/ic_announcement_orange_24dp" android:id="@+id/main_banner_json_stream_image" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="@+id/main_banner_json_stream_text" app:layout_constraintEnd_toStartOf="@id/main_banner_json_stream_text" app:layout_constraintBottom_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginStart="15dp"/> <TextView android:id="@+id/main_banner_json_stream_text" android:layout_width="0dp" android:layout_height="wrap_content" android:text="@string/main_banner_json_stream_text" android:textAppearance="@style/TextAppearance.AppCompat.Small" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toTopOf="parent" android:layout_marginEnd="15dp" android:layout_marginTop="15dp" app:layout_constraintStart_toEndOf="@+id/main_banner_json_stream_image" android:layout_marginStart="10dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_remind_later" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:text="@string/main_banner_json_stream_button_remind_later" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp" app:layout_constraintEnd_toStartOf="@+id/main_banner_json_stream_learn_mode" android:layout_marginEnd="5dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_dontaskagain" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:layout_marginEnd="5dp" android:text="@string/main_banner_json_stream_button_dismiss" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toStartOf="@id/main_banner_json_stream_remind_later" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp"/> <com.google.android.material.button.MaterialButton android:id="@+id/main_banner_json_stream_learn_mode" style="@style/Widget.MaterialComponents.Button.TextButton" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginTop="5dp" android:text="@string/main_banner_json_stream_button_learn_more" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_json_stream_text" android:layout_marginBottom="5dp" app:layout_constraintEnd_toEndOf="parent" android:layout_marginEnd="15dp"/> </androidx.constraintlayout.widget.ConstraintLayout> </com.google.android.material.card.MaterialCardView> <androidx.swiperefreshlayout.widget.SwipeRefreshLayout android:id="@+id/main_subscriptions_list_container" android:layout_width="match_parent" Loading @@ -85,7 +156,7 @@ android:visibility="visible" app:layout_constraintStart_toStartOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintTop_toBottomOf="@+id/main_banner_battery"> app:layout_constraintTop_toBottomOf="@id/main_banner_json_stream"> <androidx.recyclerview.widget.RecyclerView android:id="@+id/main_subscriptions_list" android:layout_width="match_parent" Loading
app/src/main/res/values/strings.xml +9 −2 Original line number Diff line number Diff line Loading @@ -69,10 +69,17 @@ <!-- Main activity: Battery banner --> <string name="main_banner_battery_text">Battery optimization should be disabled to avoid issues with notification delivery.</string> <string name="main_banner_battery_button_ask_later">Ask later</string> <string name="main_banner_battery_button_dismiss">Don\'t ask again</string> <string name="main_banner_battery_button_remind_later">Remind later</string> <string name="main_banner_battery_button_dismiss">Dismiss</string> <string name="main_banner_battery_button_fix_now">Fix now</string> <!-- Main activity: JSON stream banner --> <string name="main_banner_json_stream_text">Starting June 2022, WebSockets will be used to communicate with the server. Be sure get your selfhosted server ready.</string> <string name="main_banner_json_stream_button_remind_later">Remind later</string> <string name="main_banner_json_stream_button_dismiss">Dismiss</string> <string name="main_banner_json_stream_button_learn_more">Learn more</string> <string name="main_banner_json_stream_button_learn_more_url">https://ntfy.sh/docs/deprecations</string> <!-- Add dialog --> <string name="add_dialog_title">Subscribe to topic</string> <string name="add_dialog_description_below"> Loading