Files
smart-city-digital-twin-mar…/smart-app-city/frontend/node_modules/expo-camera/ios/Common/ExpoCameraUtils.swift
Eric FELIXINE e30ae8ed09 feat(smart-app): implement complete mobile app MVP
- App.tsx: full navigation (Auth stack + Main tabs with 5 screens)
- Auth: LoginScreen, RegisterScreen, ForgotPasswordScreen
- HomeScreen: dashboard with IoT metrics, weather widget, alerts, quick actions, sensors
- MapScreen: interactive map with layer toggles (6 layers)
- MarketplaceScreen: categories (6), products (5), search
- ChatScreen: AI chat with quick prompts (4), bot responses
- ProfileScreen: user info, stats, menu (9 items), logout
- AlertsScreen: alert list with severity, acknowledge
- SensorsScreen: sensor list with type filters (6 types), search
- ZonesScreen: zone cards with stats
- SettingsScreen: language picker (FR/EN/ES/DE), privacy, about
- Stores: iotStore (sensors, zones, alerts), notificationStore, uiStore + i18n
- Hooks: useSensors, useAlerts, useNotifications, useLocation
- Components: Card, Button, LoadingSpinner, ErrorBoundary, Header
- Services: iotService, notificationService (with axios API client)
- Utils: formatters (temp, AQI, noise, dates), validators (email, password, IBAN)
- Theme: colors.ts with full design system (Blue Ocean palette)
- Ditto: fixed MongoDB connection, new JWT secrets, official gateway image
2026-06-01 18:00:35 -04:00

228 lines
6.4 KiB
Swift

import AVFoundation
import CoreMotion
struct ExpoCameraUtils {
static func device(with mediaType: AVMediaType, preferring position: AVCaptureDevice.Position) -> AVCaptureDevice? {
return AVCaptureDevice.default(.builtInWideAngleCamera, for: mediaType, position: position)
}
static func deviceOrientation(
for accelerometerData: CMAccelerometerData,
default orientation: UIDeviceOrientation
) -> UIDeviceOrientation {
if accelerometerData.acceleration.x >= 0.75 {
return .landscapeRight
}
if accelerometerData.acceleration.x <= -0.75 {
return .landscapeLeft
}
if accelerometerData.acceleration.y <= -0.75 {
return .portrait
}
if accelerometerData.acceleration.y >= 0.75 {
return .portraitUpsideDown
}
return orientation
}
// .landscapeRight and .landscapeLeft of UIInterfaceOrientation are reversed when mapped to UIDeviceOrientation
static func physicalOrientation(
for orientation: UIInterfaceOrientation
) -> UIDeviceOrientation {
switch orientation {
case .portrait:
return .portrait
case .landscapeLeft:
return .landscapeRight
case .landscapeRight:
return .landscapeLeft
case .portraitUpsideDown:
return .portraitUpsideDown
case .unknown:
return .unknown
default:
return .unknown
}
}
static func videoOrientation(for interfaceOrientation: UIInterfaceOrientation) -> AVCaptureVideoOrientation {
switch interfaceOrientation {
case .portrait:
return .portrait
case .landscapeLeft:
return .landscapeLeft
case .landscapeRight:
return .landscapeRight
case .portraitUpsideDown:
return .portraitUpsideDown
default:
return .portrait
}
}
// .landscapeRight and .landscapeLeft need to be reversed when mapped back to AVCaptureVideoOrientation
static func videoOrientation(for deviceOrientation: UIDeviceOrientation) -> AVCaptureVideoOrientation {
switch deviceOrientation {
case .portrait:
return .portrait
case .portraitUpsideDown:
return .portraitUpsideDown
case .landscapeLeft:
return .landscapeRight
case .landscapeRight:
return .landscapeLeft
default:
return .portrait
}
}
static func toOrientationString(orientation: UIDeviceOrientation) -> String {
switch orientation {
case .portrait:
return "portrait"
case .landscapeLeft:
return "landscapeLeft"
case .landscapeRight:
return "landscapeRight"
case .portraitUpsideDown:
return "portraitUpsideDown"
case .faceDown:
return "faceDown"
case .faceUp:
return "faceUp"
case .unknown:
return "unknown"
@unknown default:
return "unknown"
}
}
static func toExifOrientation(orientation: UIImage.Orientation) -> Int {
switch orientation {
case .up:
return 1
case .down:
return 3
case .left:
return 8
case .right:
return 6
case .upMirrored:
return 2
case .downMirrored:
return 4
case .leftMirrored:
return 5
case .rightMirrored:
return 7
@unknown default:
return 1
}
}
static func exportImage(orientation: UIImage.Orientation) -> Int {
switch orientation {
case .left:
return 90
case .right:
return -90
case .down:
return 180
default:
return 0
}
}
static func generatePhoto(of size: CGSize) -> UIImage {
let rect = CGRect(x: 0, y: 0, width: size.width, height: size.height)
let renderer = UIGraphicsImageRenderer(size: size)
return renderer.image { ctx in
UIColor.black.setFill()
ctx.fill(rect)
let currentDate = Date()
let dateFormatter = DateFormatter()
dateFormatter.dateFormat = "dd.MM.YY HH:mm:ss"
let text = dateFormatter.string(from: currentDate)
text.draw(
with: CGRect(
x: size.width * 0.1,
y: size.height * 0.9,
width: size.width,
height: size.height
),
attributes: [.font: UIFont.systemFont(ofSize: 18), .foregroundColor: UIColor.orange],
context: nil
)
}
}
static func crop(image: UIImage, to rect: CGRect) -> UIImage {
let cgImage = image.cgImage
guard let croppedCgImage = cgImage?.cropping(to: rect) else {
return image
}
return UIImage(cgImage: croppedCgImage, scale: image.scale, orientation: image.imageOrientation)
}
static func write(data: Data, to path: String) -> String? {
let url = URL(fileURLWithPath: path)
do {
try data.write(to: url, options: .atomic)
return url.absoluteString
} catch {
return nil
}
}
static func data(from image: UIImage, with metadata: [String: Any], quality: Float) -> Data? {
guard let sourceCGImageRef = image.cgImage,
let sourceData = image.jpegData(compressionQuality: 1.0) as CFData?,
let sourceCGImageSourceRef = CGImageSourceCreateWithData(sourceData, nil),
let sourceMetadata = CGImageSourceCopyPropertiesAtIndex(sourceCGImageSourceRef, 0, nil) as? NSDictionary else {
return nil
}
let updatedMetadata = NSMutableDictionary(dictionary: sourceMetadata)
for (key, value) in metadata {
updatedMetadata[key] = value
}
updatedMetadata.setObject(NSNumber(value: quality), forKey: kCGImageDestinationLossyCompressionQuality as NSString)
let processedImageData = NSMutableData()
guard let sourceType = CGImageSourceGetType(sourceCGImageSourceRef) else {
return nil
}
guard let destinationCGImageRef =
CGImageDestinationCreateWithData(processedImageData, sourceType, 1, nil) else {
return nil
}
CGImageDestinationAddImage(destinationCGImageRef, sourceCGImageRef, updatedMetadata)
if CGImageDestinationFinalize(destinationCGImageRef) {
return processedImageData as Data
}
CGImageDestinationAddImage(destinationCGImageRef, sourceCGImageRef, updatedMetadata as CFDictionary)
return CGImageDestinationFinalize(destinationCGImageRef) ? processedImageData as Data : nil
}
static func updateExif(metadata: NSDictionary, with additionalData: [String: Any]) -> NSMutableDictionary {
let mutableMetadata = NSMutableDictionary(dictionary: metadata)
mutableMetadata.addEntries(from: additionalData)
if let gps = mutableMetadata[kCGImagePropertyGPSDictionary as String] as? [String: Any] {
for (gpsKey, gpsValue) in gps {
mutableMetadata["GPS" + gpsKey] = gpsValue
}
}
return mutableMetadata
}
}