add iOS style battery widget

This commit is contained in:
Kavish Devar
2025-01-26 19:14:53 +05:30
parent 66d7adf22c
commit f67e5defcf
17 changed files with 505 additions and 121 deletions

View File

@@ -11,8 +11,9 @@
<uses-permission
android:name="android.permission.BLUETOOTH_PRIVILEGED"
tools:ignore="ProtectedPermissions" />
<uses-permission
android:name="android.permission.BLUETOOTH_ADMIN"
<uses-permission android:name="android.permission.BATTERY_STATS"
tools:ignore="ProtectedPermissions"/>
<uses-permission android:name="android.permission.UPDATE_DEVICE_STATS"
tools:ignore="ProtectedPermissions" />
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_SCAN"
@@ -32,7 +33,7 @@
tools:ignore="UnusedAttribute"
tools:targetApi="31">
<receiver
android:name=".BatteryWidget"
android:name=".widgets.BatteryWidget"
android:exported="false">
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />

View File

@@ -321,8 +321,8 @@ fun NoiseControlSettings(service: AirPodsService) {
Box(
modifier = Modifier
.fillMaxSize()
.padding(4.dp)
.background(selectedBackground, RoundedCornerShape(11.dp))
.padding(3.dp)
.background(selectedBackground, RoundedCornerShape(12.dp))
)
}

View File

@@ -69,7 +69,9 @@ import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import androidx.navigation.NavController
import me.kavishdevar.aln.R
import me.kavishdevar.aln.composables.IndependentToggle
import me.kavishdevar.aln.composables.StyledSwitch
import me.kavishdevar.aln.services.ServiceManager
import kotlin.math.roundToInt
@OptIn(ExperimentalMaterial3Api::class)
@@ -130,6 +132,8 @@ fun AppSettingsScreen(navController: NavController) {
val backgroundColor = if (isDarkTheme) Color(0xFF1C1C1E) else Color(0xFFFFFFFF)
val textColor = if (isDarkTheme) Color.White else Color.Black
IndependentToggle("Show phone battery in widget", ServiceManager.getService()!!, "setPhoneBatteryInWidget", sharedPreferences)
Column (
modifier = Modifier
.fillMaxWidth()

View File

@@ -28,6 +28,7 @@ import android.app.Service
import android.appwidget.AppWidgetManager
import android.bluetooth.BluetoothAdapter
import android.bluetooth.BluetoothDevice
import android.bluetooth.BluetoothHeadset
import android.bluetooth.BluetoothManager
import android.bluetooth.BluetoothProfile
import android.bluetooth.BluetoothSocket
@@ -42,6 +43,7 @@ import android.content.Intent
import android.content.IntentFilter
import android.content.SharedPreferences
import android.media.AudioManager
import android.os.BatteryManager
import android.os.Binder
import android.os.Build
import android.os.Handler
@@ -49,10 +51,12 @@ import android.os.IBinder
import android.os.Looper
import android.os.ParcelUuid
import android.util.Log
import android.view.View
import android.widget.RemoteViews
import androidx.annotation.RequiresPermission
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.core.app.NotificationCompat
import androidx.core.content.edit
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.channels.awaitClose
@@ -62,7 +66,6 @@ import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.callbackFlow
import kotlinx.coroutines.launch
import me.kavishdevar.aln.BatteryWidget
import me.kavishdevar.aln.MainActivity
import me.kavishdevar.aln.R
import me.kavishdevar.aln.utils.AirPodsNotifications
@@ -75,6 +78,7 @@ import me.kavishdevar.aln.utils.Enums
import me.kavishdevar.aln.utils.LongPressPackets
import me.kavishdevar.aln.utils.MediaController
import me.kavishdevar.aln.utils.Window
import me.kavishdevar.aln.widgets.BatteryWidget
import org.lsposed.hiddenapibypass.HiddenApiBypass
object ServiceManager {
@@ -103,7 +107,7 @@ object ServiceManager {
}
}
//@Suppress("unused")
// @Suppress("unused")
class AirPodsService: Service() {
private var macAddress = ""
inner class LocalBinder : Binder() {
@@ -115,7 +119,6 @@ class AirPodsService: Service() {
private val _packetLogsFlow = MutableStateFlow<Set<String>>(emptySet())
val packetLogsFlow: StateFlow<Set<String>> get() = _packetLogsFlow
override fun onCreate() {
super.onCreate()
sharedPreferences = getSharedPreferences("packet_logs", MODE_PRIVATE)
@@ -127,7 +130,7 @@ class AirPodsService: Service() {
val logs = sharedPreferences.getStringSet(packetLogKey, mutableSetOf())?.toMutableSet() ?: mutableSetOf()
logs.add(logEntry)
_packetLogsFlow.value = logs
sharedPreferences.edit().putStringSet(packetLogKey, logs).apply()
sharedPreferences.edit { putStringSet(packetLogKey, logs) }
}
fun getPacketLogs(): Set<String> {
@@ -135,7 +138,8 @@ class AirPodsService: Service() {
}
private fun clearPacketLogs() {
sharedPreferences.edit().remove(packetLogKey).apply()
sharedPreferences.edit { remove(packetLogKey).apply() }
}
fun clearLogs() {
@@ -198,7 +202,148 @@ class AirPodsService: Service() {
var device: BluetoothDevice? = null
private lateinit var earReceiver: BroadcastReceiver
var widgetMobileBatteryEnabled = false
val METADATA_UNTETHERED_LEFT_CHARGING = 13
val METADATA_UNTETHERED_LEFT_BATTERY = 10
val METADATA_UNTETHERED_RIGHT_CHARGING = 14
val METADATA_UNTETHERED_RIGHT_BATTERY = 11
val METADATA_UNTETHERED_CASE_CHARGING = 15
val METADATA_UNTETHERED_CASE_BATTERY = 12
@SuppressLint("MissingPermission")
fun setBatteryLevels(
leftStatus: Boolean, leftLevel: Int,
rightStatus: Boolean, rightLevel: Int,
caseStatus: Boolean, caseLevel: Int,
device: BluetoothDevice
) {
HiddenApiBypass.addHiddenApiExemptions("Landroid/bluetooth/BluetoothDevice;")
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_LEFT_CHARGING,
leftStatus.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_LEFT_BATTERY,
leftLevel.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_RIGHT_CHARGING,
rightStatus.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_RIGHT_BATTERY,
rightLevel.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_CASE_CHARGING,
caseStatus.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"setMetadata",
METADATA_UNTETHERED_CASE_BATTERY,
caseLevel.toString().toByteArray()
)
HiddenApiBypass.invoke(
BluetoothDevice::class.java,
device,
"sendVendorSpecificHeadsetEvent",
"+IPHONEACCEV",
BluetoothHeadset.AT_CMD_TYPE_SET,
1,
leftLevel,
2,
rightLevel,
3,
caseLevel
)
// Prepare the intent to broadcast vendor-specific headset event
val intent = Intent(BluetoothHeadset.ACTION_VENDOR_SPECIFIC_HEADSET_EVENT).apply {
putExtra(BluetoothHeadset.EXTRA_VENDOR_SPECIFIC_HEADSET_EVENT_CMD, "+IPHONEACCEV")
putExtra(BluetoothHeadset.EXTRA_VENDOR_SPECIFIC_HEADSET_EVENT_CMD_TYPE, BluetoothHeadset.AT_CMD_TYPE_SET)
putExtra(BluetoothHeadset.EXTRA_VENDOR_SPECIFIC_HEADSET_EVENT_ARGS, arrayOf(
1, leftLevel,
2, rightLevel,
3, caseLevel
))
putExtra(BluetoothDevice.EXTRA_DEVICE, device)
putExtra(BluetoothDevice.EXTRA_NAME, device.name)
addCategory(BluetoothHeadset.VENDOR_SPECIFIC_HEADSET_EVENT_COMPANY_ID_CATEGORY + "." + 76)
}
// Send the broadcast to update the battery levels
sendBroadcast(intent)
// Broadcast battery level changes
val batteryIntent = Intent("android.bluet9ooth.device.action.BATTERY_LEVEL_CHANGED").apply {
putExtra(BluetoothDevice.EXTRA_DEVICE, device)
putExtra("android.bluetooth.device.extra.BATTERY_LEVEL", leftLevel) // Update with appropriate levels
}
sendBroadcast(batteryIntent)
}
object PhoneBatteryReceiver: BroadcastReceiver() {
override fun onReceive(context: Context?, intent: Intent) {
if (intent.action == Intent.ACTION_BATTERY_CHANGED) {
ServiceManager.getService()?.updateBatteryWidget()
}
else if (intent.action == AirPodsNotifications.DISCONNECT_RECEIVERS) {
try {
context?.unregisterReceiver(this)
} catch (e: Exception) {
e.printStackTrace()
}
}
}
}
val phoneBatteryIntentFilter = IntentFilter().apply {
addAction(Intent.ACTION_BATTERY_CHANGED)
addAction(AirPodsNotifications.DISCONNECT_RECEIVERS)
}
fun setPhoneBatteryInWidget(enabled: Boolean) {
widgetMobileBatteryEnabled = enabled
if (enabled) {
try {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
registerReceiver(
PhoneBatteryReceiver,
phoneBatteryIntentFilter,
RECEIVER_EXPORTED
)
} else {
registerReceiver(PhoneBatteryReceiver, phoneBatteryIntentFilter)
}
} catch (e: Exception) {
e.printStackTrace()
}
} else {
try {
unregisterReceiver(PhoneBatteryReceiver)
} catch (e: Exception) {
e.printStackTrace()
}
}
updateBatteryWidget()
}
@SuppressLint("MissingPermission")
fun scanForAirPods(bluetoothAdapter: BluetoothAdapter): Flow<List<ScanResult>> = callbackFlow {
val bluetoothLeScanner = bluetoothAdapter.bluetoothLeScanner
@@ -295,44 +440,85 @@ class AirPodsService: Service() {
val widgetIds = appWidgetManager.getAppWidgetIds(componentName)
val remoteViews = RemoteViews(packageName, R.layout.battery_widget).also {
val leftBattery = batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT }
val rightBattery = batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT }
val caseBattery = batteryNotification.getBattery().find { it.component == BatteryComponent.CASE }
it.setTextViewText(
R.id.left_battery_widget,
batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT }?.let {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
leftBattery?.let {
"${it.level}%"
} ?: ""
)
it.setProgressBar(
R.id.left_battery_progress,
100,
batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT }?.level ?: 0,
leftBattery?.level ?: 0,
false
)
it.setViewVisibility(
R.id.left_charging_icon,
if (leftBattery?.status == BatteryStatus.CHARGING) View.VISIBLE else View.GONE
)
it.setTextViewText(
R.id.right_battery_widget,
batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT }?.let {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
rightBattery?.let {
"${it.level}%"
} ?: ""
)
it.setProgressBar(
R.id.right_battery_progress,
100,
batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT }?.level ?: 0,
rightBattery?.level ?: 0,
false
)
it.setViewVisibility(
R.id.right_charging_icon,
if (rightBattery?.status == BatteryStatus.CHARGING) View.VISIBLE else View.GONE
)
it.setTextViewText(
R.id.case_battery_widget,
batteryNotification.getBattery().find { it.component == BatteryComponent.CASE }?.let {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
caseBattery?.let {
"${it.level}%"
} ?: ""
)
it.setProgressBar(
R.id.case_battery_progress,
100,
batteryNotification.getBattery().find { it.component == BatteryComponent.CASE }?.level ?: 0,
caseBattery?.level ?: 0,
false
)
it.setViewVisibility(
R.id.case_charging_icon,
if (caseBattery?.status == BatteryStatus.CHARGING) View.VISIBLE else View.GONE
)
it.setViewVisibility(
R.id.phone_battery_widget_container,
if (widgetMobileBatteryEnabled) View.VISIBLE else View.GONE
)
if (widgetMobileBatteryEnabled) {
val batteryManager = getSystemService<BatteryManager>(BatteryManager::class.java)
val batteryLevel = batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY)
val charging = batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_STATUS) == BatteryManager.BATTERY_STATUS_CHARGING
it.setTextViewText(
R.id.phone_battery_widget,
"$batteryLevel%"
)
it.setViewVisibility(
R.id.phone_charging_icon,
if (charging) View.VISIBLE else View.GONE
)
it.setProgressBar(
R.id.phone_battery_progress,
100,
batteryLevel,
false
)
}
}
Log.d("AirPodsService", "Updating battery widget")
appWidgetManager.updateAppWidget(widgetIds, remoteViews)
}
@@ -397,7 +583,7 @@ class AirPodsService: Service() {
Log.d("AirPodsService", "Service started")
ServiceManager.setService(this)
startForegroundNotification()
Log.d("AirPodsService", "Initializing CrossDevice")
CrossDevice.init(this)
Log.d("AirPodsService", "CrossDevice initialized")
@@ -415,12 +601,6 @@ class AirPodsService: Service() {
addAction("android.bluetooth.a2dp.profile.action.PLAYING_STATE_CHANGED")
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
registerReceiver(bluetoothReceiver, serviceIntentFilter, RECEIVER_EXPORTED)
} else {
registerReceiver(bluetoothReceiver, serviceIntentFilter)
}
connectionReceiver = object: BroadcastReceiver() {
override fun onReceive(context: Context?, intent: Intent?) {
if (intent?.action == AirPodsNotifications.Companion.AIRPODS_CONNECTION_DETECTED) {
@@ -432,8 +612,8 @@ class AirPodsService: Service() {
val name = this@AirPodsService.getSharedPreferences("settings", MODE_PRIVATE)
.getString("name", device?.name)
if (this@AirPodsService.getSharedPreferences("settings", MODE_PRIVATE).getString("name", null) == null) {
this@AirPodsService.getSharedPreferences("settings", MODE_PRIVATE).edit()
.putString("name", name).apply()
this@AirPodsService.getSharedPreferences("settings", MODE_PRIVATE).edit {
putString("name", name)}
}
Log.d("AirPodsQuickSwitchServices", CrossDevice.isAvailable.toString())
if (!CrossDevice.checkAirPodsConnectionStatus()) {
@@ -444,8 +624,7 @@ class AirPodsService: Service() {
macAddress = device!!.address
updateNotificationContent(true, name.toString(), batteryNotification.getBattery())
}
}
else if (intent?.action == AirPodsNotifications.Companion.AIRPODS_DISCONNECTED) {
} else if (intent?.action == AirPodsNotifications.Companion.AIRPODS_DISCONNECTED) {
device = null
isConnectedLocally = false
popupShown = false
@@ -454,16 +633,28 @@ class AirPodsService: Service() {
}
}
val deviceIntentFilter = IntentFilter().apply {
addAction(AirPodsNotifications.Companion.AIRPODS_CONNECTION_DETECTED)
addAction(AirPodsNotifications.Companion.AIRPODS_DISCONNECTED)
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
registerReceiver(connectionReceiver, deviceIntentFilter, RECEIVER_EXPORTED)
registerReceiver(bluetoothReceiver, serviceIntentFilter, RECEIVER_EXPORTED)
} else {
registerReceiver(connectionReceiver, deviceIntentFilter)
registerReceiver(bluetoothReceiver, serviceIntentFilter)
}
widgetMobileBatteryEnabled = getSharedPreferences("settings", MODE_PRIVATE).getBoolean("show_phone_battery_in_widget", true)
if (widgetMobileBatteryEnabled) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
registerReceiver(PhoneBatteryReceiver, phoneBatteryIntentFilter, RECEIVER_EXPORTED)
} else {
registerReceiver(PhoneBatteryReceiver, phoneBatteryIntentFilter)
}
}
val bluetoothAdapter = getSystemService(BluetoothManager::class.java).adapter
if (bluetoothAdapter.isEnabled) {
CoroutineScope(Dispatchers.IO).launch {
@@ -801,6 +992,15 @@ class AirPodsService: Service() {
} else {
connectAudio(this@AirPodsService, device)
}
// setBatteryLevels(
// batteryNotification.getBattery()[0].status == 1,
// batteryNotification.getBattery()[0].level,
// batteryNotification.getBattery()[1].status == 1,
// batteryNotification.getBattery()[1].level,
// batteryNotification.getBattery()[2].status == 1,
// batteryNotification.getBattery()[2].level,
// device
// )
} else if (conversationAwarenessNotification.isConversationalAwarenessData(
data
)

View File

@@ -17,15 +17,21 @@
*/
package me.kavishdevar.aln
package me.kavishdevar.aln.widgets
import android.app.PendingIntent
import android.appwidget.AppWidgetManager
import android.appwidget.AppWidgetProvider
import android.content.Context
import android.content.Intent
import android.graphics.Canvas
import android.util.Log
import android.widget.RemoteViews
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.core.graphics.createBitmap
import me.kavishdevar.aln.MainActivity
import me.kavishdevar.aln.R
import me.kavishdevar.aln.services.ServiceManager
import me.kavishdevar.aln.utils.BatteryComponent
import me.kavishdevar.aln.utils.BatteryStatus
class BatteryWidget : AppWidgetProvider() {
override fun onUpdate(
@@ -33,7 +39,6 @@ class BatteryWidget : AppWidgetProvider() {
appWidgetManager: AppWidgetManager,
appWidgetIds: IntArray
) {
// There may be multiple widgets active, so update all of them
for (appWidgetId in appWidgetIds) {
updateAppWidget(context, appWidgetManager, appWidgetId)
}
@@ -44,40 +49,19 @@ class BatteryWidget : AppWidgetProvider() {
}
}
@OptIn(ExperimentalMaterial3Api::class)
internal fun updateAppWidget(
context: Context,
appWidgetManager: AppWidgetManager,
appWidgetId: Int
) {
val service = ServiceManager.getService()
val batteryList = service?.batteryNotification?.getBattery()
val views = RemoteViews(context.packageName, R.layout.battery_widget)
views.setTextViewText(R.id.left_battery_widget,
batteryList?.find { it.component == BatteryComponent.LEFT }?.let {
// if (it.status != BatteryStatus.DISCONNECTED) {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
// } else {
// ""
// }
} ?: "")
views.setTextViewText(R.id.right_battery_widget,
batteryList?.find { it.component == BatteryComponent.RIGHT }?.let {
// if (it.status != BatteryStatus.DISCONNECTED) {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
// } else {
// ""
// }
} ?: "")
views.setTextViewText(R.id.case_battery_widget,
batteryList?.find { it.component == BatteryComponent.CASE }?.let {
// if (it.status != BatteryStatus.DISCONNECTED) {
"${if (it.status == BatteryStatus.CHARGING) "⚡" else ""} ${it.level}%"
// } else {
// ""
// }
} ?: "")
service?.updateBatteryWidget()
val openActivityIntent = PendingIntent.getActivity(context, 0, Intent(context, MainActivity::class.java), PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE)
views.setOnClickPendingIntent(R.id.battery_widget, openActivityIntent)
appWidgetManager.updateAppWidget(appWidgetId, views)
}

View File

@@ -1,7 +0,0 @@
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item>
<shape android:shape="rectangle">
<solid android:color="#DA000000" />
</shape>
</item>
</layer-list>

View File

@@ -0,0 +1,14 @@
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_pressed="true">
<shape android:shape="rectangle">
<solid android:color="#121212" />
<corners android:radius="16dp" />
</shape>
</item>
<item>
<shape android:shape="rectangle">
<solid android:color="#404040" />
<corners android:radius="12dp" />
</shape>
</item>
</selector>

View File

@@ -1,15 +1,13 @@
<rotate xmlns:android="http://schemas.android.com/apk/res/android"
android:fromDegrees="-90"
<rotate
xmlns:android="http://schemas.android.com/apk/res/android"
android:fromDegrees="270"
android:toDegrees="270">
<shape
android:shape="ring"
android:innerRadiusRatio="3"
android:thicknessRatio="8"
android:innerRadiusRatio="3.0"
android:thickness="6dp"
android:useLevel="true">
<gradient
android:type="sweep"
android:useLevel="true"
android:startColor="#00ff00"
android:endColor="#00ff00" />
<solid android:color="#00D85B" />
<corners android:radius="10dp" />
</shape>
</rotate>
</rotate>

View File

@@ -0,0 +1,15 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="88.06dp"
android:height="140.66dp"
android:viewportWidth="88.06"
android:viewportHeight="140.66">
<path
android:fillColor="#FF000000"
android:pathData="M0,0h88.06v140.66h-88.06z"
android:strokeAlpha="0"
android:fillAlpha="0"/>
<path
android:pathData="M0,77.55C0,79.99 1.88,81.8 4.5,81.8L39.81,81.8L21.19,132.42C18.75,138.86 25.44,142.3 29.63,137.05L86.44,66.05C87.5,64.74 88.06,63.49 88.06,62.05C88.06,59.67 86.19,57.8 83.56,57.8L48.25,57.8L66.88,7.17C69.31,0.74 62.63,-2.7 58.44,2.61L1.63,73.55C0.56,74.92 0,76.17 0,77.55Z"
android:fillColor="#ffffff"
android:fillAlpha="0.85"/>
</vector>

View File

@@ -1,10 +1,10 @@
<?xml version="1.0" encoding="utf-8"?>
<shape
xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle" >
android:shape="rectangle">
<solid
android:color="@color/popup_background" >
android:color="@color/popup_background">
</solid>
<padding

View File

@@ -0,0 +1,11 @@
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item>
<shape
android:shape="ring"
android:innerRadiusRatio="3.0"
android:thickness="6dp">
<solid android:color="#49474E" />
</shape>
</item>
</selector>

View File

@@ -0,0 +1,19 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="80.31dp"
android:height="132.44dp"
android:viewportWidth="80.31"
android:viewportHeight="132.44">
<path
android:fillColor="#FF000000"
android:pathData="M0,0h80.31v132.44h-80.31z"
android:strokeAlpha="0"
android:fillAlpha="0"/>
<path
android:pathData="M12.44,114.31C11.06,114.31 10.06,113.31 10.06,111.94L10.06,12.44C10.06,11 11,10.06 12.44,10.06L67.88,10.06C69.31,10.06 70.25,11 70.25,12.44L70.25,111.94C70.25,113.31 69.31,114.31 67.88,114.31Z"
android:fillColor="#ffffff"
android:fillAlpha="0.2125"/>
<path
android:pathData="M7.75,132.31L72.56,132.31C77.13,132.31 80.31,129.13 80.31,124.56L80.31,7.75C80.31,3.19 77.13,0 72.56,0L7.75,0C3.19,0 0,3.19 0,7.75L0,124.56C0,129.13 3.25,132.31 7.75,132.31ZM12.44,114.31C11.06,114.31 10.06,113.31 10.06,111.94L10.06,12.44C10.06,11 11,10.06 12.44,10.06L67.88,10.06C69.31,10.06 70.25,11 70.25,12.44L70.25,111.94C70.25,113.31 69.31,114.31 67.88,114.31Z"
android:fillColor="#ffffff"
android:fillAlpha="0.85"/>
</vector>

View File

@@ -0,0 +1,9 @@
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item>
<shape android:shape="rectangle">
<padding android:bottom="0dp" android:left="0dp" android:right="0dp" android:top="0dp" />
<solid android:color="#222222" />
<corners android:radius="32dp" />
</shape>
</item>
</layer-list>

View File

@@ -2,28 +2,106 @@
xmlns:tools="http://schemas.android.com/tools"
style="@style/Widget.ALN.AppWidget.Container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_height="wrap_content"
android:layout_margin="0dp"
android:padding="0dp"
android:id="@+id/battery_widget"
android:theme="@style/Theme.ALN.AppWidgetContainer"
android:background="@drawable/blur_background">
android:background="@drawable/widget_background">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:baselineAligned="false"
android:orientation="horizontal"
android:layout_margin="0dp"
android:gravity="center"
android:orientation="horizontal">
android:id="@android:id/background"
android:layout_gravity="center">
<LinearLayout
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:layout_height="match_parent"
android:gravity="center"
android:id="@+id/phone_battery_widget_container"
android:orientation="vertical">
<FrameLayout
android:layout_width="90dp"
android:layout_height="90dp"
android:layout_margin="0dp"
android:padding="6dp"
android:layout_gravity="center">
<ProgressBar
android:id="@+id/phone_battery_progress_background"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="100"
android:progressDrawable="@drawable/progress_bar_background" />
<ProgressBar
android:id="@+id/phone_battery_progress"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="50"
android:progressDrawable="@drawable/circular_progress_bar" />
<ImageView
android:id="@+id/phone_charging_icon"
android:layout_width="20dp"
android:layout_height="20dp"
android:layout_gravity="top|center_horizontal"
android:importantForAccessibility="no"
android:src="@drawable/ic_power"
android:visibility="gone"
android:tint="@color/white"
tools:ignore="HardcodedText" />
<ImageView
android:layout_width="28dp"
android:layout_height="28dp"
android:src="@drawable/smartphone"
android:tint="@color/white"
android:layout_gravity="center"
android:importantForAccessibility="no"
tools:ignore="HardcodedText" />
</FrameLayout>
<TextView
android:id="@+id/phone_battery_widget"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="24sp"
android:textColor="@color/white"
android:fontFamily="@font/sf_pro"
android:gravity="center"
android:textFontWeight="300"
android:text="Phone"
tools:ignore="HardcodedText" />
</LinearLayout>
<LinearLayout
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="match_parent"
android:gravity="center"
android:orientation="vertical">
<FrameLayout
android:layout_width="64dp"
android:layout_height="64dp"
android:layout_width="90dp"
android:layout_height="90dp"
android:layout_margin="0dp"
android:padding="6dp"
android:layout_gravity="center">
<ProgressBar
android:id="@+id/left_battery_progress_background"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="100"
android:progressDrawable="@drawable/progress_bar_background" />
<ProgressBar
android:id="@+id/left_battery_progress"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
@@ -31,22 +109,36 @@
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="50"
android:progressDrawable="@drawable/circular_progress_bar" />
<ImageView
android:layout_width="32dp"
android:layout_height="32dp"
android:id="@+id/left_charging_icon"
android:layout_width="20dp"
android:layout_height="20dp"
android:layout_gravity="top|center_horizontal"
android:importantForAccessibility="no"
android:src="@drawable/ic_power"
android:visibility="gone"
android:tint="@color/white"
tools:ignore="HardcodedText" />
<ImageView
android:layout_width="28dp"
android:layout_height="28dp"
android:src="@drawable/airpods_pro_left_notification"
android:tint="@color/popup_text"
android:tint="@color/white"
android:layout_gravity="center"
android:importantForAccessibility="no"
tools:ignore="HardcodedText" />
</FrameLayout>
<TextView
android:id="@+id/left_battery_widget"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:textColor="@color/popup_text"
android:textSize="24sp"
android:textColor="@color/white"
android:fontFamily="@font/sf_pro"
android:gravity="center"
android:textFontWeight="300"
android:text="Left"
tools:ignore="HardcodedText" />
</LinearLayout>
@@ -54,13 +146,24 @@
<LinearLayout
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:layout_height="match_parent"
android:gravity="center"
android:orientation="vertical">
<FrameLayout
android:layout_width="64dp"
android:layout_height="64dp"
android:layout_width="90dp"
android:layout_height="90dp"
android:layout_margin="0dp"
android:padding="6dp"
android:layout_gravity="center">
<ProgressBar
android:id="@+id/right_battery_progress_background"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="100"
android:progressDrawable="@drawable/progress_bar_background" />
<ProgressBar
android:id="@+id/right_battery_progress"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
@@ -70,10 +173,21 @@
android:max="100"
android:progressDrawable="@drawable/circular_progress_bar" />
<ImageView
android:layout_width="32dp"
android:layout_height="32dp"
android:id="@+id/right_charging_icon"
android:layout_width="20dp"
android:layout_height="20dp"
android:layout_gravity="top|center_horizontal"
android:importantForAccessibility="no"
android:src="@drawable/ic_power"
android:visibility="gone"
android:tint="@color/white"
tools:ignore="HardcodedText" />
<ImageView
android:layout_width="28dp"
android:layout_height="28dp"
android:importantForAccessibility="no"
android:src="@drawable/airpods_pro_right_notification"
android:tint="@color/popup_text"
android:tint="@color/white"
android:layout_gravity="center"
tools:ignore="HardcodedText" />
</FrameLayout>
@@ -81,8 +195,10 @@
android:id="@+id/right_battery_widget"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:textColor="@color/popup_text"
android:textSize="24sp"
android:textFontWeight="300"
android:textColor="@color/white"
android:fontFamily="@font/sf_pro"
android:gravity="center"
android:text="Right"
tools:ignore="HardcodedText" />
@@ -91,13 +207,24 @@
<LinearLayout
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:layout_height="match_parent"
android:gravity="center"
android:orientation="vertical">
<FrameLayout
android:layout_width="64dp"
android:layout_height="64dp"
android:layout_width="90dp"
android:layout_height="90dp"
android:layout_margin="0dp"
android:padding="6dp"
android:layout_gravity="center">
<ProgressBar
android:id="@+id/case_battery_progress_background"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:indeterminate="false"
android:max="100"
android:progress="100"
android:progressDrawable="@drawable/progress_bar_background" />
<ProgressBar
android:id="@+id/case_battery_progress"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
@@ -107,10 +234,21 @@
android:max="100"
android:progressDrawable="@drawable/circular_progress_bar" />
<ImageView
android:layout_width="32dp"
android:layout_height="32dp"
android:id="@+id/case_charging_icon"
android:layout_width="20dp"
android:layout_height="20dp"
android:importantForAccessibility="no"
android:layout_gravity="top|center_horizontal"
android:src="@drawable/ic_power"
android:visibility="gone"
android:tint="@color/white"
tools:ignore="HardcodedText" />
<ImageView
android:layout_width="28dp"
android:layout_height="28dp"
android:importantForAccessibility="no"
android:src="@drawable/airpods_pro_case_notification"
android:tint="@color/popup_text"
android:tint="@color/white"
android:layout_gravity="center"
tools:ignore="HardcodedText" />
</FrameLayout>
@@ -118,9 +256,11 @@
android:id="@+id/case_battery_widget"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:textColor="@color/popup_text"
android:textSize="24sp"
android:textColor="@color/white"
android:gravity="center"
android:fontFamily="@font/sf_pro"
android:textFontWeight="300"
android:text="Case"
tools:ignore="HardcodedText" />
</LinearLayout>

View File

@@ -7,7 +7,7 @@
android:layout_margin="16.dp"
android:id="@+id/linear_layout"
android:orientation="vertical"
android:background="@drawable/shape">
android:background="@drawable/popup_shape">
<androidx.constraintlayout.widget.ConstraintLayout
android:layout_width="match_parent"

View File

@@ -4,17 +4,11 @@
<style name="Theme.ALN" parent="android:Theme.Material.Light.NoActionBar" />
<style name="Theme.ALN.AppWidgetContainerParent" parent="@android:style/Theme.DeviceDefault">
<!-- Radius of the outer bound of widgets to make the rounded corners -->
<item name="appWidgetRadius">24dp</item>
<!--
Radius of the inner view's bound of widgets to make the rounded corners.
It needs to be 8dp or less than the value of appWidgetRadius
-->
<item name="appWidgetInnerRadius">24dp</item>
<item name="appWidgetRadius">32dp</item>
<item name="appWidgetPadding">0dp</item>
</style>
<style name="Theme.ALN.AppWidgetContainer" parent="Theme.ALN.AppWidgetContainerParent">
<!-- Apply padding to avoid the content of the widget colliding with the rounded corners -->
<item name="appWidgetPadding">16dp</item>
<item name="appWidgetPadding">0dp</item>
</style>
</resources>

View File

@@ -1,14 +1,16 @@
<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:description="@string/app_widget_description"
android:initialKeyguardLayout="@layout/battery_widget"
android:initialLayout="@layout/battery_widget"
android:minWidth="40dp"
android:minWidth="180dp"
android:minHeight="40dp"
android:previewImage="@drawable/example_appwidget_preview"
android:previewLayout="@layout/battery_widget"
android:resizeMode="horizontal|vertical"
android:targetCellWidth="1"
android:targetCellWidth="3"
android:targetCellHeight="1"
android:updatePeriodMillis="300000"
android:widgetCategory="home_screen|keyguard" />
android:widgetCategory="home_screen|keyguard"
tools:ignore="UnusedAttribute" />