Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update safe area calculations #76

Merged
merged 3 commits into from
Oct 24, 2023
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,11 @@ import android.net.Network
import android.system.Os
import android.view.View
import android.view.ViewGroup
import android.view.WindowInsets
import android.webkit.*
import androidx.activity.ComponentActivity
import androidx.appcompat.app.AppCompatDelegate
import androidx.core.view.WindowInsetsCompat
import androidx.lifecycle.DefaultLifecycleObserver
import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.MutableLiveData
Expand All @@ -29,7 +31,7 @@ import com.github.itwin.mobilesdk.jsonvalue.optStringOrNull
import com.github.itwin.mobilesdk.jsonvalue.toMap
import kotlinx.coroutines.*
import org.json.JSONObject
import java.lang.Float.max
import java.lang.Integer.max
import java.net.URLEncoder
import java.util.concurrent.atomic.AtomicBoolean

Expand Down Expand Up @@ -592,38 +594,55 @@ open class ITMApplication(
webView?.let { webView ->
(webView.parent as? ViewGroup)?.removeView(webView)
container.addView(webView)
webView.setOnApplyWindowInsetsListener { v, insets ->
updateSafeAreas(v)
v.onApplyWindowInsets(insets)
webView.setOnApplyWindowInsetsListener { view, insets ->
updateSafeAreas(view, insets)
view.onApplyWindowInsets(insets)
}
}
}

private fun updateSafeAreas(view: View) {
val activity = ((view.parent as? ViewGroup)?.context as? Activity) ?: return
val window = activity.window ?: return
val message = mutableMapOf(
"left" to 0.0f,
"right" to 0.0f,
"top" to 0.0f,
"bottom" to 0.0f
)
window.decorView.rootWindowInsets?.displayCutout?.let { displayCutoutInsets ->
private data class Quint<out A, out B, out C, out D, out E>(
val first: A,
val second: B,
val third: C,
val fourth: D,
val fifth: E
) {
/**
* Returns string representation of the [Quint] including its [first], [second], [third], [fourth], and [fifth] values.
*/
override fun toString(): String = "($first, $second, $third, $fourth, $fifth)"
}

private fun getSafeAreas(activity: Activity, view: View, insets: WindowInsets): Quint<Float, Float, Float, Float, Float> {
fun toDp(value: Int): Float {
val density = activity.resources.displayMetrics.density
val top = displayCutoutInsets.safeInsetTop / density
val bottom = displayCutoutInsets.safeInsetBottom / density
val left = displayCutoutInsets.safeInsetLeft / density
val right = displayCutoutInsets.safeInsetRight / density
// Make both sides have the same safe area.
val sides = max(left, right)
message["left"] = sides
message["right"] = sides
// Include the actual left/right insets so developers can use them if desired.
message["minLeft"] = left
message["minRight"] = right
message["top"] = top
message["bottom"] = bottom
return value / density
}
var insetsType = WindowInsetsCompat.Type.displayCutout()
// Ideally we'd always get system bars insets but before API 30 getInsets() returns bars insets even if they are overlaid on top.
if (activity.isInMultiWindowMode)
insetsType = insetsType or WindowInsetsCompat.Type.systemBars()

return with(WindowInsetsCompat.toWindowInsetsCompat(insets, view).getInsets(insetsType)) {
Quint(toDp(max(left, right)), toDp(left), toDp(right), toDp(top), toDp(bottom))
}
}

@Suppress("DestructuringDeclarationWithTooManyEntries")
private fun updateSafeAreas(view: View, insets: WindowInsets) {
val activity = ((view.parent as? ViewGroup)?.context as? Activity) ?: return
val (sides, left, right, top, bottom) = getSafeAreas(activity, view, insets)
val message = mapOf(
// We want both sides to have the same safe area.
"left" to sides,
"right" to sides,
// Include the actual left/right insets so developers can use them if desired.
"minLeft" to left,
"minRight" to right,
"top" to top,
"bottom" to bottom,
)
messenger.send("Bentley_ITM_muiUpdateSafeAreas", message)
}

Expand Down