Files
iOS/Sources/App/Frontend/WebView/WebViewController+EmptyState.swift
Bruno Pantaleão Gonçalves 0bb2d6296f Fix Safe area empty state (#4572)
<!-- Thank you for submitting a Pull Request and helping to improve Home
Assistant. Please complete the following sections to help the processing
and review of your changes. Please do not delete anything from this
template. -->

## Summary
<!-- Provide a brief summary of the changes you have made and most
importantly what they aim to achieve -->

## Screenshots
<!-- If this is a user-facing change not in the frontend, please include
screenshots in light and dark mode. -->

## Link to pull request in Documentation repository
<!-- Pull requests that add, change or remove functionality must have a
corresponding pull request in the Companion App Documentation repository
(https://github.com/home-assistant/companion.home-assistant). Please add
the number of this pull request after the "#" -->
Documentation: home-assistant/companion.home-assistant#

## Any other notes
<!-- If there is any other information of note, like if this Pull
Request is part of a bigger change, please include it here. -->
2026-04-28 10:30:52 +00:00

126 lines
4.3 KiB
Swift

import Shared
import SwiftUI
import UIKit
// MARK: - Empty State
extension WebViewController {
func setupEmptyState() {
let emptyState = WebViewEmptyStateWrapperView(
style: emptyStateStyle(for: connectionState),
server: server,
showsErrorDetailsButton: shouldShowErrorDetailsButton,
retryAction: { [weak self] in
self?.hideEmptyState()
self?.refresh()
},
settingsAction: { [weak self] in
self?.showSettingsViewController()
},
errorDetailsAction: { [weak self] in
self?.presentLatestLoadErrorDetails()
},
reauthAction: { [weak self] urlType in
self?.performReauthentication(using: urlType)
},
dismissAction: { [weak self] in
self?.hideEmptyState()
}
)
addChild(emptyState.hostingViewController)
view.addSubview(emptyState)
emptyState.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
emptyState.leftAnchor.constraint(equalTo: view.leftAnchor),
emptyState.rightAnchor.constraint(equalTo: view.rightAnchor),
emptyState.topAnchor.constraint(equalTo: view.topAnchor),
emptyState.bottomAnchor.constraint(equalTo: view.bottomAnchor),
])
emptyState.alpha = 0
emptyStateView = emptyState
emptyState.hostingViewController.didMove(toParent: self)
}
func emptyStateStyle(for connectionState: FrontEndConnectionState) -> WebViewEmptyStateStyle {
switch connectionState {
case .authInvalid:
.unauthenticated
case .connected, .disconnected, .unknown:
.disconnected
}
}
func showEmptyState() {
emptyStateView?.update(
style: emptyStateStyle(for: connectionState),
showsErrorDetailsButton: shouldShowErrorDetailsButton
)
UIView.animate(withDuration: emptyStateTransitionDuration, delay: 0, options: .curveEaseInOut, animations: {
self.emptyStateView?.alpha = 1
}, completion: nil)
}
var shouldShowErrorDetailsButton: Bool {
connectionState == .disconnected && latestLoadError != nil
}
func presentLatestLoadErrorDetails() {
guard let latestLoadError else { return }
presentOverlayController(
controller: UIHostingController(rootView: ConnectionErrorDetailsView(
server: server,
error: latestLoadError
)),
animated: true
)
}
@objc func hideEmptyState() {
UIView.animate(withDuration: emptyStateTransitionDuration, delay: 0, options: .curveEaseInOut, animations: {
self.emptyStateView?.alpha = 0
}, completion: nil)
}
// To avoid keeping the empty state on screen when user is disconnected in background
// due to inactivity, we reset the empty state timer
@objc func resetEmptyStateTimerWithLatestConnectedState() {
let state: FrontEndConnectionState = if connectionState == .authInvalid {
.authInvalid
} else {
isConnected ? .connected : .disconnected
}
updateFrontendConnectionState(state: state.rawValue)
}
func emptyStateObservations() {
// Hide empty state when enter background
NotificationCenter.default.addObserver(
self,
selector: #selector(hideEmptyState),
name: UIApplication.didEnterBackgroundNotification,
object: nil
)
// Show empty state again if after entering foreground it is not connected
NotificationCenter.default.addObserver(
self,
selector: #selector(resetEmptyStateTimerWithLatestConnectedState),
name: UIApplication.willEnterForegroundNotification,
object: nil
)
}
func removeEmptyStateObservations() {
NotificationCenter.default.removeObserver(self, name: UIApplication.didEnterBackgroundNotification, object: nil)
NotificationCenter.default.removeObserver(
self,
name: UIApplication.willEnterForegroundNotification,
object: nil
)
}
}