diff --git a/posthog-android/src/main/java/com/posthog/android/internal/PostHogAndroidUtils.kt b/posthog-android/src/main/java/com/posthog/android/internal/PostHogAndroidUtils.kt index 63ae20ad..e3c1e214 100644 --- a/posthog-android/src/main/java/com/posthog/android/internal/PostHogAndroidUtils.kt +++ b/posthog-android/src/main/java/com/posthog/android/internal/PostHogAndroidUtils.kt @@ -9,16 +9,19 @@ import android.content.pm.ApplicationInfo import android.content.pm.PackageInfo import android.content.pm.PackageManager import android.content.pm.PackageManager.GET_META_DATA +import android.graphics.Bitmap import android.graphics.Point import android.net.ConnectivityManager import android.net.Uri import android.os.Build import android.os.Process import android.telephony.TelephonyManager +import android.util.Base64 import android.util.DisplayMetrics import android.view.WindowManager import com.posthog.PostHogInternal import com.posthog.android.PostHogAndroidConfig +import java.io.ByteArrayOutputStream @Suppress("DEPRECATION") internal fun getPackageInfo( @@ -205,3 +208,52 @@ public fun getApplicationInfo(context: Context): ApplicationInfo = .packageManager .getApplicationInfo(context.packageName, GET_META_DATA) } + +private fun Bitmap.isValid(): Boolean { + return !isRecycled && + width > 0 && + height > 0 +} + +@PostHogInternal +@Suppress("DEPRECATION") +public fun Bitmap.base64( + format: Bitmap.CompressFormat = Bitmap.CompressFormat.JPEG, + quality: Int = 30, +): String? { + if (!isValid()) { + return null + } + + val lossyFormat = + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { + Bitmap.CompressFormat.WEBP_LOSSY + } else { + Bitmap.CompressFormat.WEBP + } + + val losslessFormat = + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { + Bitmap.CompressFormat.WEBP_LOSSLESS + } else { + Bitmap.CompressFormat.WEBP + } + + val htmlFormat = + when (format) { + Bitmap.CompressFormat.JPEG -> "jpeg" + Bitmap.CompressFormat.PNG -> "png" + Bitmap.CompressFormat.WEBP -> "webp" + lossyFormat -> "webp" + losslessFormat -> "webp" + else -> "jpeg" + } + + ByteArrayOutputStream(allocationByteCount).use { + // we can make format and type configurable + compress(format, quality, it) + val byteArray = it.toByteArray() + val encoded = Base64.encodeToString(byteArray, Base64.DEFAULT) ?: return null + return "data:image/$htmlFormat;base64,$encoded" + } +} diff --git a/posthog-android/src/main/java/com/posthog/android/replay/PostHogReplayIntegration.kt b/posthog-android/src/main/java/com/posthog/android/replay/PostHogReplayIntegration.kt index f90c6a64..29848ebc 100644 --- a/posthog-android/src/main/java/com/posthog/android/replay/PostHogReplayIntegration.kt +++ b/posthog-android/src/main/java/com/posthog/android/replay/PostHogReplayIntegration.kt @@ -24,7 +24,6 @@ import android.os.Build import android.os.Handler import android.os.HandlerThread import android.text.InputType -import android.util.Base64 import android.util.TypedValue import android.view.Gravity import android.view.MotionEvent @@ -54,6 +53,7 @@ import com.posthog.PostHog import com.posthog.PostHogIntegration import com.posthog.android.PostHogAndroidConfig import com.posthog.android.internal.MainHandler +import com.posthog.android.internal.base64 import com.posthog.android.internal.densityValue import com.posthog.android.internal.displayMetrics import com.posthog.android.internal.screenSize @@ -83,7 +83,6 @@ import curtains.onDecorViewReady import curtains.phoneWindow import curtains.touchEventInterceptors import curtains.windowAttachCount -import java.io.ByteArrayOutputStream import java.lang.ref.WeakReference import java.util.WeakHashMap import java.util.concurrent.CountDownLatch @@ -1100,26 +1099,6 @@ public class PostHogReplayIntegration( return null } - private fun Bitmap.isValid(): Boolean { - return !isRecycled && - width > 0 && - height > 0 - } - - private fun Bitmap.base64(): String? { - if (!isValid()) { - return null - } - - ByteArrayOutputStream(allocationByteCount).use { - // we can make format and type configurable - compress(Bitmap.CompressFormat.JPEG, 30, it) - val byteArray = it.toByteArray() - val encoded = Base64.encodeToString(byteArray, Base64.DEFAULT) ?: return null - return "data:image/jpeg;base64,$encoded" - } - } - private fun Drawable.base64( width: Int, height: Int,