Files
iOS/Sources/Extensions/Widgets/Controls/Button/IntentButtonEntity.swift
Bruno Pantaleão Gonçalves a4dbd8b843 Add entity context to configuration screens (#4777)
<!-- 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 -->
This PR adds device and area context to Watch, CarPlay, Widgets and App
Icon shortcuts configuration screens.
## 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-06-18 14:28:17 +02:00

106 lines
3.7 KiB
Swift

import AppIntents
import Foundation
import GRDB
import SFSafeSymbols
import Shared
@available(iOS 18.0, *)
struct IntentButtonEntity: AppEntity, EntityContextRepresentable {
static let typeDisplayRepresentation = TypeDisplayRepresentation(name: "Button")
static let defaultQuery = IntentButtonAppEntityQuery()
// UniqueID: serverId-entityId
var id: String
var entityId: String
var serverId: String
var areaName: String?
var deviceName: String?
var displayString: String
var iconName: String
var displayRepresentation: DisplayRepresentation {
DisplayRepresentation(
title: "\(displayString)",
subtitle: contextSubtitle.map { LocalizedStringResource(stringLiteral: $0) }
)
}
init(
id: String,
entityId: String,
serverId: String,
areaName: String? = nil,
deviceName: String? = nil,
displayString: String,
iconName: String
) {
self.id = id
self.entityId = entityId
self.serverId = serverId
self.areaName = areaName
self.deviceName = deviceName
self.displayString = displayString
self.iconName = iconName
}
}
@available(iOS 18.0, *)
struct IntentButtonAppEntityQuery: EntityQuery, EntityStringQuery {
#if WIDGET_EXTENSION
@IntentParameterDependency<ControlButtonConfiguration>(\.$server)
var config
#endif
func entities(for identifiers: [String]) async throws -> [IntentButtonEntity] {
await getButtonEntities().flatMap(\.1).filter { identifiers.contains($0.id) }
}
func entities(matching string: String) async throws -> IntentItemCollection<IntentButtonEntity> {
await collection(for: getButtonEntities(matching: string))
}
func suggestedEntities() async throws -> IntentItemCollection<IntentButtonEntity> {
await collection(for: getButtonEntities())
}
/// Scopes the list to the server picked in the configuration (flat list). When no server is
/// selected (e.g. a widget configured before this option existed), falls back to grouping
/// every server's entities into sections.
private func collection(
for entitiesPerServer: [(Server, [IntentButtonEntity])]
) -> IntentItemCollection<IntentButtonEntity> {
#if WIDGET_EXTENSION
if let server = config?.server {
let items = entitiesPerServer.first { $0.0.identifier.rawValue == server.id }?.1 ?? []
return .init(items: items)
}
#endif
return .init(sections: entitiesPerServer.map { server, items in
.init(.init(stringLiteral: server.info.name), items: items)
})
}
private func getButtonEntities(matching string: String? = nil) async -> [(Server, [IntentButtonEntity])] {
var buttonEntities: [(Server, [IntentButtonEntity])] = []
let entities = ControlEntityProvider(domains: [.button, .inputButton]).getEntities(matching: string)
for (server, values) in entities {
let deviceMap = values.devicesMap(for: server.identifier.rawValue)
let areasMap = values.areasMap(for: server.identifier.rawValue)
buttonEntities.append((server, values.map({ entity in
IntentButtonEntity(
id: entity.id,
entityId: entity.entityId,
serverId: entity.serverId,
areaName: areasMap[entity.entityId]?.name,
deviceName: deviceMap[entity.entityId]?.name,
displayString: entity.name,
iconName: entity.icon ?? SFSymbol.circleCircle.rawValue
)
})))
}
return buttonEntities
}
}