fd62db1720
Enable audio-reactive lighting on the Android-TV build. A push-based AndroidAudioEngine captures system playback audio via AudioPlaybackCapture (API 29+), reusing the existing MediaProjection token, and feeds PCM into the unchanged AudioAnalyzer pipeline. No new Python deps; no Chaquopy/pip changes (numpy already bundled). - Python: android_audio_engine.py — module-level queue + configure/ push_samples/shutdown mirroring mediaprojection_engine; AndroidAudioEngine (priority 100) registered behind a guarded import. push_samples copies and defensively trims/clamps each block so the analyzer can't crash on variable-length or non-frame-divisible PCM. - Kotlin: AudioCapture.kt — AudioRecord + AudioPlaybackCaptureConfiguration, fixed chunk-size block framing, little-endian float32, mic fallback; reads back the actual negotiated channel/sample rate. PythonBridge gains configureAudio/pushAudio/shutdownAudio with a cached module handle. - Wiring: CaptureService starts/stops AudioCapture in the MediaProjection path (gated on API>=29 + RECORD_AUDIO + live projection); MainActivity requests RECORD_AUDIO; manifest declares it. Degrades gracefully when denied; root path stays audio-less by design. - Tests: 13 desktop-CI tests incl. an over-length/non-divisible regression guard that exercises the full read_chunk -> AudioAnalyzer.analyze path.
130 lines
6.6 KiB
XML
130 lines
6.6 KiB
XML
<?xml version="1.0" encoding="utf-8"?>
|
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
|
xmlns:tools="http://schemas.android.com/tools">
|
|
|
|
<!-- BLE scanning and connecting — API ≥31 uses granular permissions;
|
|
older releases need BLUETOOTH + ACCESS_FINE_LOCATION for scanning.
|
|
neverForLocation avoids the location permission dialog on API 31+. -->
|
|
<uses-permission android:name="android.permission.BLUETOOTH"
|
|
android:maxSdkVersion="30" />
|
|
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"
|
|
android:maxSdkVersion="30" />
|
|
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"
|
|
android:maxSdkVersion="30" />
|
|
<uses-permission android:name="android.permission.BLUETOOTH_SCAN"
|
|
android:usesPermissionFlags="neverForLocation"
|
|
tools:targetApi="s" />
|
|
<uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />
|
|
|
|
<!-- BLE hardware — required=false so non-BT boxes still install. -->
|
|
<uses-feature
|
|
android:name="android.hardware.bluetooth_le"
|
|
android:required="false" />
|
|
|
|
<!-- Network access for WLED HTTP/UDP, web UI, MQTT -->
|
|
<uses-permission android:name="android.permission.INTERNET" />
|
|
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
|
|
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
|
|
|
|
<!-- Foreground service permissions.
|
|
FOREGROUND_SERVICE_MEDIA_PROJECTION: required on API 34+ for the
|
|
MediaProjection capture path.
|
|
FOREGROUND_SERVICE_SPECIAL_USE: required on API 34+ for the root
|
|
screenrecord capture path (it doesn't use MediaProjection).
|
|
Both are declared because the service may run in either mode. -->
|
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
|
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION" />
|
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_SPECIAL_USE" />
|
|
|
|
<!-- POST_NOTIFICATIONS for Android 13+ foreground service notification -->
|
|
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
|
|
|
|
<!-- RECORD_AUDIO for on-device system-playback capture (AudioPlaybackCapture,
|
|
API 29+) feeding audio-reactive lighting. Runtime "dangerous" permission,
|
|
requested in MainActivity; capture degrades gracefully when denied.
|
|
Playback capture runs under the existing mediaProjection FGS type, so no
|
|
FOREGROUND_SERVICE_MICROPHONE / microphone FGS type is needed (that would
|
|
only be required if the mic-fallback path ran inside the service). -->
|
|
<uses-permission android:name="android.permission.RECORD_AUDIO" />
|
|
|
|
<!-- Autostart on boot — BootReceiver spawns CaptureService in root
|
|
mode so capture resumes without the user touching the remote. -->
|
|
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
|
<!-- Exempt from Doze/App Standby so the FG service isn't killed
|
|
overnight on phones; essentially a no-op on TV boxes. -->
|
|
<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS" />
|
|
<!-- Optional wake lock for sustained-performance boxes that aggressively
|
|
sleep the CPU with the display off. -->
|
|
<uses-permission android:name="android.permission.WAKE_LOCK" />
|
|
|
|
<!-- Android TV declarations -->
|
|
<uses-feature
|
|
android:name="android.software.leanback"
|
|
android:required="true" />
|
|
<uses-feature
|
|
android:name="android.hardware.touchscreen"
|
|
android:required="false" />
|
|
|
|
<!-- USB host — for USB-to-TTL adapters driving Adalight/AmbiLED
|
|
controllers. required=false so phones without USB host still install. -->
|
|
<uses-feature
|
|
android:name="android.hardware.usb.host"
|
|
android:required="false" />
|
|
|
|
<application
|
|
android:name=".LedGrabApp"
|
|
android:allowBackup="false"
|
|
android:enableOnBackInvokedCallback="true"
|
|
android:icon="@mipmap/ic_launcher"
|
|
android:label="@string/app_name"
|
|
android:banner="@drawable/banner_tv"
|
|
android:networkSecurityConfig="@xml/network_security_config"
|
|
android:theme="@style/Theme.LedGrab">
|
|
|
|
<!-- TV launcher activity. Boots through the SplashScreen theme so
|
|
the (sometimes multi-second) Chaquopy stdlib unpack doesn't
|
|
show as a black screen on first launch. -->
|
|
<activity
|
|
android:name=".MainActivity"
|
|
android:exported="true"
|
|
android:theme="@style/Theme.LedGrab.Splash">
|
|
<intent-filter>
|
|
<action android:name="android.intent.action.MAIN" />
|
|
<category android:name="android.intent.category.LEANBACK_LAUNCHER" />
|
|
</intent-filter>
|
|
</activity>
|
|
|
|
<!-- Foreground service for screen capture + Python server.
|
|
Declares BOTH mediaProjection AND specialUse: only one is
|
|
active at a time but Android needs to see the union of
|
|
possible types up-front so it doesn't kill the service when
|
|
we promote it with a different type at runtime.
|
|
FOREGROUND_SERVICE_TYPE_SPECIAL_USE on API 34+ requires the
|
|
PROPERTY_SPECIAL_USE_FGS_SUBTYPE rationale below. -->
|
|
<service
|
|
android:name=".CaptureService"
|
|
android:foregroundServiceType="mediaProjection|specialUse"
|
|
android:exported="false">
|
|
<property
|
|
android:name="android.app.PROPERTY_SPECIAL_USE_FGS_SUBTYPE"
|
|
android:value="Root-mode screen capture for ambient LED sync. Uses /system/bin/screenrecord on rooted devices to avoid MediaProjection's persistent capture indicator overlay, which is required for the always-on ambient-lighting use case." />
|
|
</service>
|
|
|
|
<!-- Autostart — fires on device boot (and package replace).
|
|
On rooted devices, launches CaptureService directly so capture
|
|
resumes without the user tapping Start. Unrooted devices are
|
|
no-op because MediaProjection consent cannot be bypassed. -->
|
|
<receiver
|
|
android:name=".BootReceiver"
|
|
android:exported="true"
|
|
android:enabled="true">
|
|
<intent-filter>
|
|
<action android:name="android.intent.action.BOOT_COMPLETED" />
|
|
<action android:name="android.intent.action.LOCKED_BOOT_COMPLETED" />
|
|
<action android:name="android.intent.action.MY_PACKAGE_REPLACED" />
|
|
</intent-filter>
|
|
</receiver>
|
|
</application>
|
|
|
|
</manifest>
|