Files
iOS/Sources/App/Frontend/WebView/WebViewController+EmptyState.swift
Bruno Pantaleão Gonçalves 6e22a09338 Prevent duplicate error message when HA unreachable (#4546)
<!-- 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. -->
<img width="1206" height="2622" alt="Simulator Screenshot - Daily tester
2 - 2026-04-22 at 14 43 08"
src="https://github.com/user-attachments/assets/66e0e557-5b69-4f8d-82c1-53dbf7fb4b9b"
/>

## 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. -->

---------

Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>
2026-04-22 15:40:52 +02:00

124 lines
4.1 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()
}
)
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
}
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
)
}
}