iOS/Sources/App/BarcodeScanner/Camera/BarcodeScannerDataModel.swift
Bruno Pantaleão Gonçalves 9ea7951315
Fix barcode scanner camera leak (#3603)
<!-- 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. -->

---------

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2025-05-19 11:48:59 +02:00

64 lines
1.6 KiB
Swift

import AVFoundation
import os.log
import SwiftUI
protocol BarcodeScannerDataModelDelegate: AnyObject {
func didDetectBarcode(_ code: String, format: String)
}
final class BarcodeScannerDataModel: ObservableObject {
let camera = BarcodeScannerCamera()
@Published var viewfinderImage: Image?
weak var delegate: BarcodeScannerDataModelDelegate?
private var handleCameraPreviewsTask: Task<Void, Never>?
init() {
camera.delegate = self
self.handleCameraPreviewsTask = Task {
await handleCameraPreviews()
}
}
func stop() {
handleCameraPreviewsTask?.cancel()
camera.stop()
}
func handleCameraPreviews() async {
guard let previewStream = camera.previewStream else {
return
}
let imageStream = previewStream.map(\.image)
for await image in imageStream {
await MainActor.run {
viewfinderImage = image
}
}
}
func toggleFlashlight() {
camera.toggleFlashlight()
}
func turnOffFlashlight() {
camera.turnOffFlashlight()
}
}
extension BarcodeScannerDataModel: BarcodeScannerCameraDelegate {
func didDetectBarcode(_ code: String, format: String) {
delegate?.didDetectBarcode(code, format: format)
}
}
private extension CIImage {
var image: Image? {
let ciContext = CIContext()
guard let cgImage = ciContext.createCGImage(self, from: extent) else { return nil }
return Image(decorative: cgImage, scale: 1, orientation: .up)
}
}