@ajuarezso/capacitor-liquid-glass 0.3.4 → 0.3.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -152,7 +152,12 @@ public class LiquidGlassPlugin: CAPPlugin, CAPBridgedPlugin {
|
|
|
152
152
|
}
|
|
153
153
|
|
|
154
154
|
private func presentTabBar(items: [LiquidGlassTabItem], selectedIndex: Int, tintHex: String?, styleRaw: String) {
|
|
155
|
-
|
|
155
|
+
// CRÍTICO: usar `bridge?.viewController` (el VC que contiene el
|
|
156
|
+
// WKWebView de Capacitor) en lugar del `rootViewController` del
|
|
157
|
+
// window. iOS 26 aplica Liquid Glass automáticamente al UITabBar
|
|
158
|
+
// solo cuando está en la jerarquía del VC del webview. El rootVC
|
|
159
|
+
// del window puede ser un container distinto y romper el adopt.
|
|
160
|
+
guard let hostVC = bridge?.viewController else { return }
|
|
156
161
|
|
|
157
162
|
if tabBarOverlay == nil {
|
|
158
163
|
let overlay = LiquidGlassTabBarOverlay()
|
|
@@ -169,7 +174,7 @@ public class LiquidGlassPlugin: CAPPlugin, CAPBridgedPlugin {
|
|
|
169
174
|
}
|
|
170
175
|
|
|
171
176
|
let style = LiquidGlassTabBarStyle(rawValue: styleRaw) ?? .default
|
|
172
|
-
tabBarOverlay?.attach(to:
|
|
177
|
+
tabBarOverlay?.attach(to: hostVC)
|
|
173
178
|
tabBarOverlay?.configure(items: items, selectedIndex: selectedIndex, tintHex: tintHex, style: style)
|
|
174
179
|
tabBarOverlay?.show()
|
|
175
180
|
}
|
|
@@ -42,14 +42,18 @@ enum LiquidGlassTabBarStyle: String {
|
|
|
42
42
|
/// app se compila contra el SDK iOS 26.
|
|
43
43
|
///
|
|
44
44
|
/// **Decisión arquitectónica crítica**: este view controller se adjunta como
|
|
45
|
-
/// CHILD del
|
|
46
|
-
///
|
|
47
|
-
///
|
|
45
|
+
/// CHILD del view controller que contiene el WKWebView de Capacitor
|
|
46
|
+
/// (`bridge?.viewController`), y el `UITabBar` vive como subview de
|
|
47
|
+
/// `self.view`. Esto crea el view controller hierarchy completo que iOS 26
|
|
48
|
+
/// necesita para aplicar el material Liquid Glass automáticamente al
|
|
49
|
+
/// UITabBar.
|
|
48
50
|
///
|
|
49
|
-
///
|
|
50
|
-
///
|
|
51
|
-
///
|
|
52
|
-
///
|
|
51
|
+
/// Por qué NO usar `window.rootViewController`: en Capacitor el rootVC del
|
|
52
|
+
/// window puede ser un container distinto al VC del webview. iOS 26 solo
|
|
53
|
+
/// aplica Liquid Glass auto cuando el UITabBar vive en la jerarquía del VC
|
|
54
|
+
/// que tiene el webview activo. Agregarlo al rootVC del window puede
|
|
55
|
+
/// romper el auto-adopt y dejarlo con material translúcido genérico opaco
|
|
56
|
+
/// (no el real de Music / App Store).
|
|
53
57
|
///
|
|
54
58
|
/// Patrón inspirado en `stay-liquid` (alistairheath/stay-liquid GitHub).
|
|
55
59
|
final class LiquidGlassTabBarOverlay: UIViewController {
|
|
@@ -66,7 +70,7 @@ final class LiquidGlassTabBarOverlay: UIViewController {
|
|
|
66
70
|
override func viewDidLoad() {
|
|
67
71
|
super.viewDidLoad()
|
|
68
72
|
// Background transparente. El `self.view` está sizado SOLO al rect
|
|
69
|
-
// del tab bar (leading/trailing/bottom al
|
|
73
|
+
// del tab bar (leading/trailing/bottom al hostVC, sin top — la
|
|
70
74
|
// altura la determina el intrinsic content size del UITabBar dentro).
|
|
71
75
|
// Por eso NO necesitamos `hitTest` override: el view en sí solo
|
|
72
76
|
// existe sobre el área de la pill, los taps fuera no llegan.
|
|
@@ -75,9 +79,13 @@ final class LiquidGlassTabBarOverlay: UIViewController {
|
|
|
75
79
|
tabBar.translatesAutoresizingMaskIntoConstraints = false
|
|
76
80
|
tabBar.delegate = self
|
|
77
81
|
|
|
78
|
-
//
|
|
79
|
-
//
|
|
80
|
-
|
|
82
|
+
// CRÍTICO: NO setear appearance acá. iOS 26 adopta Liquid Glass
|
|
83
|
+
// automáticamente SOLO si el UITabBar mantiene su appearance default
|
|
84
|
+
// intacto. Tocar `standardAppearance` / `scrollEdgeAppearance` en
|
|
85
|
+
// CUALQUIER forma — incluso con `configureWithDefaultBackground()` —
|
|
86
|
+
// anula el adopt y deja un material translúcido genérico opaco.
|
|
87
|
+
// Solo los overrides intencionales (.ultraThin, .transparent) tocan
|
|
88
|
+
// el appearance, vía `applyAppearance(...)` desde `configure(...)`.
|
|
81
89
|
|
|
82
90
|
view.addSubview(tabBar)
|
|
83
91
|
|
|
@@ -91,12 +99,13 @@ final class LiquidGlassTabBarOverlay: UIViewController {
|
|
|
91
99
|
])
|
|
92
100
|
}
|
|
93
101
|
|
|
94
|
-
/// Adjunta este view controller como child del
|
|
95
|
-
///
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
102
|
+
/// Adjunta este view controller como child del view controller que
|
|
103
|
+
/// contiene el WKWebView de Capacitor (`bridge?.viewController`).
|
|
104
|
+
/// Idempotente: si ya está adjuntado al mismo host, no-op.
|
|
105
|
+
func attach(to hostVC: UIViewController?) {
|
|
106
|
+
guard let hostVC else { return }
|
|
107
|
+
if self.parent === hostVC { return }
|
|
108
|
+
// Si está adjuntado a otro VC (cambio de host entre sesiones),
|
|
100
109
|
// detachar primero antes de re-adjuntar.
|
|
101
110
|
if self.parent != nil {
|
|
102
111
|
self.willMove(toParent: nil)
|
|
@@ -104,37 +113,44 @@ final class LiquidGlassTabBarOverlay: UIViewController {
|
|
|
104
113
|
self.removeFromParent()
|
|
105
114
|
}
|
|
106
115
|
|
|
107
|
-
|
|
116
|
+
hostVC.addChild(self)
|
|
108
117
|
self.view.translatesAutoresizingMaskIntoConstraints = false
|
|
109
|
-
|
|
118
|
+
hostVC.view.addSubview(self.view)
|
|
110
119
|
// SIN top constraint — la altura del view se deriva del intrinsic
|
|
111
120
|
// content size del UITabBar dentro (~50pt + safe-area-bottom).
|
|
112
121
|
// Replica del patrón stay-liquid (validado en producción) que
|
|
113
122
|
// permite a iOS 26 detectar este UITabBar como "floating tab bar"
|
|
114
123
|
// y aplicarle Liquid Glass automáticamente.
|
|
115
124
|
NSLayoutConstraint.activate([
|
|
116
|
-
self.view.leadingAnchor.constraint(equalTo:
|
|
117
|
-
self.view.trailingAnchor.constraint(equalTo:
|
|
118
|
-
self.view.bottomAnchor.constraint(equalTo:
|
|
125
|
+
self.view.leadingAnchor.constraint(equalTo: hostVC.view.leadingAnchor),
|
|
126
|
+
self.view.trailingAnchor.constraint(equalTo: hostVC.view.trailingAnchor),
|
|
127
|
+
self.view.bottomAnchor.constraint(equalTo: hostVC.view.bottomAnchor),
|
|
119
128
|
])
|
|
120
|
-
self.didMove(toParent:
|
|
121
|
-
self.hostVC =
|
|
129
|
+
self.didMove(toParent: hostVC)
|
|
130
|
+
self.hostVC = hostVC
|
|
122
131
|
emitLayout()
|
|
123
132
|
}
|
|
124
133
|
|
|
125
134
|
/// Aplica el appearance correspondiente al estilo solicitado.
|
|
126
|
-
/// - `default` / `liquidGlass`:
|
|
127
|
-
///
|
|
135
|
+
/// - `default` / `liquidGlass`: **NO-OP** — el UITabBar mantiene su
|
|
136
|
+
/// appearance default sin tocar para que iOS 26 aplique Liquid Glass
|
|
137
|
+
/// automáticamente. Cualquier touch al appearance (incluso
|
|
138
|
+
/// `configureWithDefaultBackground()`) rompe el auto-adopt.
|
|
128
139
|
/// - `ultraThin`: blur mínimo `systemUltraThinMaterial` — más ver-through.
|
|
140
|
+
/// Override intencional que CANCELA Liquid Glass.
|
|
129
141
|
/// - `transparent`: sin background ni blur — content behind se ve completo.
|
|
142
|
+
/// Override intencional que CANCELA Liquid Glass.
|
|
130
143
|
private func applyAppearance(_ tabBar: UITabBar, style: LiquidGlassTabBarStyle) {
|
|
144
|
+
// Early-return para los estilos que confían en el adopt automático
|
|
145
|
+
// del sistema. Tocar appearance acá anula Liquid Glass.
|
|
146
|
+
if style == .default || style == .liquidGlass { return }
|
|
147
|
+
|
|
131
148
|
let appearance = UITabBarAppearance()
|
|
132
149
|
switch style {
|
|
133
150
|
case .default, .liquidGlass:
|
|
134
|
-
//
|
|
135
|
-
//
|
|
136
|
-
|
|
137
|
-
appearance.configureWithDefaultBackground()
|
|
151
|
+
// Unreachable — early-return arriba. Mantenido por exhaustividad
|
|
152
|
+
// del switch sobre el enum.
|
|
153
|
+
return
|
|
138
154
|
case .ultraThin:
|
|
139
155
|
appearance.configureWithDefaultBackground()
|
|
140
156
|
appearance.backgroundEffect = UIBlurEffect(style: .systemUltraThinMaterial)
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ajuarezso/capacitor-liquid-glass",
|
|
3
|
-
"version": "0.3.
|
|
3
|
+
"version": "0.3.5",
|
|
4
4
|
"description": "iOS 26 Liquid Glass native chrome (TabBar, NavigationBar, Alerts, Sheets) for Capacitor apps. Falls back gracefully on iOS < 26 and Android.",
|
|
5
5
|
"main": "dist/plugin.cjs.js",
|
|
6
6
|
"module": "dist/esm/index.js",
|