Skip to content

Commit

Permalink
feat: bitcoin price widget for android (GaloyMoney#3309)
Browse files Browse the repository at this point in the history
* chore: android data query works

* chore: chart works

* chore: wow graph

* chore: full UI

* chore: finish up

* chore: cleanup

* fix: show price

* chore: remove logs

* fix: minor adjustments
  • Loading branch information
sandipndev authored Jun 17, 2024
1 parent 3571b60 commit 2db08f7
Show file tree
Hide file tree
Showing 10 changed files with 340 additions and 1 deletion.
5 changes: 4 additions & 1 deletion android/app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -147,9 +147,12 @@ android {
dependencies {
// The version of react-native is set by the React Native Gradle Plugin
implementation("com.facebook.react:react-android")

implementation("com.google.mlkit:barcode-scanning:17.2.0")

// For widget
implementation("androidx.work:work-runtime-ktx:2.9.0")
implementation("com.github.PhilJay:MPAndroidChart:v3.1.0")

if (hermesEnabled.toBoolean()) {
implementation("com.facebook.react:hermes-android")
} else {
Expand Down
6 changes: 6 additions & 0 deletions android/app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,12 @@
android:usesCleartextTraffic="true"
android:networkSecurityConfig="@xml/network_security_config"
android:theme="@style/AppTheme"> <!-- Apply @style/AppTheme on .MainApplication -->
<receiver android:name=".BitcoinPriceWidget" android:exported="false">
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
</intent-filter>
<meta-data android:name="android.appwidget.provider" android:resource="@xml/bitcoin_price_widget_info" />
</receiver>
<activity
android:name=".MainActivity"
android:label="@string/app_name"
Expand Down
226 changes: 226 additions & 0 deletions android/app/src/main/java/com/galoyapp/BitcoinPriceWidget.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
package com.galoyapp

import android.annotation.SuppressLint
import android.app.PendingIntent
import android.appwidget.AppWidgetManager
import android.appwidget.AppWidgetProvider
import android.content.ComponentName
import android.content.Context
import android.content.Intent
import android.graphics.Bitmap
import android.graphics.Color
import android.net.Uri
import android.view.View
import android.widget.RemoteViews
import androidx.core.content.ContextCompat
import androidx.work.Data
import androidx.work.OneTimeWorkRequestBuilder
import androidx.work.OutOfQuotaPolicy
import androidx.work.Worker
import androidx.work.WorkerParameters
import java.io.BufferedReader
import java.io.InputStreamReader
import java.net.HttpURLConnection
import java.net.URL
import java.util.concurrent.TimeUnit
import androidx.work.PeriodicWorkRequestBuilder
import androidx.work.WorkManager
import com.github.mikephil.charting.charts.LineChart
import com.github.mikephil.charting.data.Entry
import com.github.mikephil.charting.data.LineData
import com.github.mikephil.charting.data.LineDataSet
import org.json.JSONArray
import org.json.JSONObject
import kotlin.math.pow

class BitcoinPriceWidget : AppWidgetProvider() {
override fun onUpdate(context: Context, appWidgetManager: AppWidgetManager, appWidgetIds: IntArray) {
for (appWidgetId in appWidgetIds) {
updateAppWidget(context, appWidgetManager, appWidgetId)
}
}

override fun onEnabled(context: Context) {
super.onEnabled(context)

val immediateWorkRequest = OneTimeWorkRequestBuilder<FetchPriceWorker>()
.setInputData(Data.Builder().putString("RANGE", "ONE_DAY").build())
.setExpedited(OutOfQuotaPolicy.RUN_AS_NON_EXPEDITED_WORK_REQUEST)
.build()

val periodicWorkRequest = PeriodicWorkRequestBuilder<FetchPriceWorker>(15, TimeUnit.MINUTES)
.setInputData(Data.Builder().putString("RANGE", "ONE_DAY").build())
.build()

WorkManager.getInstance(context).apply {
enqueue(immediateWorkRequest)
enqueue(periodicWorkRequest)
}

val appWidgetManager = AppWidgetManager.getInstance(context)
val appWidgetIds = appWidgetManager.getAppWidgetIds(ComponentName(context, BitcoinPriceWidget::class.java))
appWidgetIds.forEach { appWidgetId ->
updateAppWidget(context, appWidgetManager, appWidgetId)
}
}

override fun onDisabled(context: Context) {
WorkManager.getInstance(context).cancelAllWorkByTag("FETCH_PRICE_WORK")
}
}

private fun generateChartBitmap(context: Context, data: JSONArray, width: Int, height: Int): Bitmap {
val entries = ArrayList<Entry>()
var maxY = 0f
var minY = 9999999f
for (i in 0 until data.length()) {
val item = data.getJSONObject(i)
val price = item.getJSONObject("price").getString("formattedAmount").toFloat()
entries.add(Entry(i.toFloat(), price))
if (price > maxY) { maxY = price }
if (price < minY) { minY = price }
}

val chart = LineChart(context).apply {
setDrawGridBackground(false)
description.isEnabled = false
legend.isEnabled = false
axisLeft.isEnabled = false
axisRight.isEnabled = false
xAxis.isEnabled = false

setBackgroundColor(Color.BLACK)
setExtraOffsets(0f, 0f, 0f, 0f)
setViewPortOffsets(0f, 0f, 0f, 0f)

axisLeft.axisMaximum = maxY + ( (maxY - minY) * 0.3f )
}

val dataSet = LineDataSet(entries, "").apply {
setDrawValues(false)
setDrawCircles(false)
mode = LineDataSet.Mode.HORIZONTAL_BEZIER
cubicIntensity = 0.4f // Adjust this value to control the smoothness, default is 0.2
lineWidth = 1f // Adjust the line width for aesthetic appearance
color = ContextCompat.getColor(context, R.color.primary)
setDrawFilled(true)
fillDrawable = ContextCompat.getDrawable(context, R.drawable.bitcoin_price_widget_chart_gradient)
}

chart.data = LineData(dataSet)

chart.measure(
View.MeasureSpec.makeMeasureSpec(width, View.MeasureSpec.EXACTLY),
View.MeasureSpec.makeMeasureSpec(height, View.MeasureSpec.EXACTLY))
chart.layout(0, 0, chart.measuredWidth, chart.measuredHeight)

chart.invalidate()
return chart.getChartBitmap();
}

@SuppressLint("DefaultLocale")
internal fun updateAppWidget(context: Context, appWidgetManager: AppWidgetManager, appWidgetId: Int) {
val options = appWidgetManager.getAppWidgetOptions(appWidgetId)
val width = options.getInt(AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH)
val height = options.getInt(AppWidgetManager.OPTION_APPWIDGET_MAX_HEIGHT)

val minWidth = View.MeasureSpec.getSize(width)
val maxHeight = View.MeasureSpec.getSize(height)

val views = RemoteViews(context.packageName, R.layout.bitcoin_price_widget)

val prefs = context.getSharedPreferences("bitcoinPricePrefs", Context.MODE_PRIVATE)
val priceArray = JSONArray(prefs.getString("PRICE_ARRAY", "[]"))

val realtimePrice = JSONObject(prefs.getString("REALTIME_PRICE", "No Data") ?: "{}")
if (!realtimePrice.has("noData")) {

val bitmap = generateChartBitmap(context, priceArray, minWidth, maxHeight)
views.setImageViewBitmap(R.id.chart_image_view, bitmap)

val btcSatBase = realtimePrice.getJSONObject("btcSatPrice").getLong("base")
val btcSatOffset = realtimePrice.getJSONObject("btcSatPrice").getInt("offset")
val btcUsdPrice = (btcSatBase / 10.0.pow(btcSatOffset)) * 100000000 / 100

val formattedBtcUsdPrice = String.format("%.2f", btcUsdPrice)
views.setTextViewText(R.id.btc_price, "$$formattedBtcUsdPrice")

views.setViewVisibility(R.id.error_message, View.GONE)
views.setViewVisibility(R.id.btc_price, View.VISIBLE)
views.setViewVisibility(R.id.btc_price_label, View.VISIBLE)
views.setViewVisibility(R.id.btc_logo, View.VISIBLE)
} else {
views.setViewVisibility(R.id.error_message, View.VISIBLE)
views.setViewVisibility(R.id.btc_price, View.GONE)
views.setViewVisibility(R.id.btc_price_label, View.GONE)
views.setViewVisibility(R.id.btc_logo, View.GONE)
}

val intent = Intent(Intent.ACTION_VIEW, Uri.parse("blink://price"))
val pendingIntent = PendingIntent.getActivity(context, 0, intent, PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT)
views.setOnClickPendingIntent(R.id.bitcoin_price_widget, pendingIntent)

appWidgetManager.updateAppWidget(appWidgetId, views)
}

enum class BitcoinPriceRanges {
ONE_DAY,
ONE_WEEK,
ONE_MONTH,
ONE_YEAR,
FIVE_YEARS,
}

class FetchPriceWorker(context: Context, params: WorkerParameters) : Worker(context, params) {
override fun doWork(): Result {
val range = BitcoinPriceRanges.valueOf(inputData.getString("RANGE") ?: "ONE_DAY")
val jsonResponse = fetchBitcoinPrice(range)
val priceArray = jsonResponse.getJSONArray("btcPriceList")
val realtimePrice = jsonResponse.getJSONObject("realtimePrice")
val prefs =
applicationContext.getSharedPreferences("bitcoinPricePrefs", Context.MODE_PRIVATE)

with(prefs.edit()) {
putString("PRICE_ARRAY", priceArray.toString())
putString("REALTIME_PRICE", realtimePrice.toString())
apply()
}

updateWidgets(applicationContext)
return Result.success()
}

private fun updateWidgets(context: Context) {
val appWidgetManager = AppWidgetManager.getInstance(context)
val ids = appWidgetManager.getAppWidgetIds(ComponentName(context, BitcoinPriceWidget::class.java))
ids.forEach { id ->
val updateIntent = Intent(context, BitcoinPriceWidget::class.java).apply {
action = AppWidgetManager.ACTION_APPWIDGET_UPDATE
putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, intArrayOf(id))
}
context.sendBroadcast(updateIntent)
}
}

private fun fetchBitcoinPrice(range: BitcoinPriceRanges): JSONObject {
val url = URL("https://api.mainnet.galoy.io/graphql")
val query = "query BitcoinPriceForAppWidget(\$range: PriceGraphRange!) { btcPriceList(range: \$range) { timestamp price { base offset currencyUnit formattedAmount } } realtimePrice { btcSatPrice { base offset } timestamp usdCentPrice { base offset } } }"
val jsonInputString = """{ "query": "$query", "variables": { "range": "$range" } }"""

with(url.openConnection() as HttpURLConnection) {
requestMethod = "POST"
setRequestProperty("Content-Type", "application/json")
doOutput = true
outputStream.use { os ->
os.write(jsonInputString.toByteArray())
}
return if (responseCode == HttpURLConnection.HTTP_OK) {
BufferedReader(InputStreamReader(inputStream)).use { br ->
JSONObject(br.readText()).getJSONObject("data")
}
} else {
JSONObject("{\"btcPriceList\": [], \"realtimePrice\": { \"noData\": true } }")
}
}
}
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<gradient
android:startColor="#00000000"
android:endColor="#66FC5805"
android:angle="90" />
</shape>
64 changes: 64 additions & 0 deletions android/app/src/main/res/layout/bitcoin_price_widget.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/black"
android:id="@+id/bitcoin_price_widget"
android:padding="0dp">

<TextView
android:id="@+id/error_message"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="@string/blink_price_no_internet"
android:textColor="#FFFFFF"
android:textSize="16sp"
android:visibility="gone"
android:gravity="center"
android:padding="16dp" />

<ImageView
android:id="@+id/chart_image_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginTop="0dp"
android:adjustViewBounds="true"
android:contentDescription="@string/chart_content_description" />

<TextView
android:id="@+id/btc_price"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:layout_marginTop="8dp"
android:paddingHorizontal="8dp"
android:text="@string/loading"
android:textColor="#FFFFFF"
android:textStyle="bold"
android:background="#80000000"
android:textSize="18sp" />

<TextView
android:id="@+id/btc_price_label"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/btc_price"
android:layout_alignStart="@id/btc_price"
android:text="@string/btc_usd"
android:textSize="11sp"
android:textColor="#FFFFFF"
android:paddingHorizontal="8dp"
android:background="#80000000"
/>

<ImageView
android:id="@+id/btc_logo"
android:layout_width="86dp"
android:layout_height="86dp"
android:layout_alignParentBottom="true"
android:layout_alignParentEnd="true"
android:layout_marginBottom="-20dp"
android:src="@drawable/bootsplash_logo"
android:contentDescription="@string/blink_logo_description" />
</RelativeLayout>
1 change: 1 addition & 0 deletions android/app/src/main/res/values/colors.xml
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
<resources>
<color name="bootsplash_background">#000000</color>
<color name="windowBackgroundColor">#ffffff</color>
<color name="primary">#FC5805</color>
</resources>
8 changes: 8 additions & 0 deletions android/app/src/main/res/values/strings.xml
Original file line number Diff line number Diff line change
@@ -1,3 +1,11 @@
<resources>
<string name="app_name">Blink</string>
<string name="appwidget_text">Bitcoin Price</string>
<string name="add_widget">Add bitcoin price widget</string>
<string name="app_widget_description">USD/BTC price as reported by Blink Servers</string>
<string name="chart_content_description">Bitcoin Price USD/BTC Chart</string>
<string name="btc_usd">BTC/USD</string>
<string name="blink_logo_description">Blink Logo</string>
<string name="blink_price_no_internet">Need internet connection to show BTC/USD price data</string>
<string name="loading">Loading…</string>
</resources>
13 changes: 13 additions & 0 deletions android/app/src/main/res/xml/bitcoin_price_widget_info.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:description="@string/app_widget_description"
android:initialKeyguardLayout="@layout/bitcoin_price_widget"
android:initialLayout="@layout/bitcoin_price_widget"
android:minWidth="80dp"
android:minHeight="80dp"
android:previewImage="@drawable/bitcoin_price_widget"
android:targetCellWidth="2"
android:targetCellHeight="2"
android:resizeMode="horizontal|vertical"
android:updatePeriodMillis="86400000"
android:widgetCategory="home_screen" />
10 changes: 10 additions & 0 deletions android/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,16 @@ buildscript {
androidXAnnotation = "1.2.0"
androidXBrowser = "1.3.0"
}
subprojects { subproject ->
afterEvaluate{
if((subproject.plugins.hasPlugin('android') || subproject.plugins.hasPlugin('android-library'))) {
android {
compileSdkVersion rootProject.ext.compileSdkVersion
buildToolsVersion rootProject.ext.buildToolsVersion
}
}
}
}
repositories {
google()
jcenter()
Expand Down

0 comments on commit 2db08f7

Please sign in to comment.