guardia-messenger/node_modules/expo-modules-core/ios/Fabric/ExpoFabricView.swift
DESKTOP-TKLFCPRython f29f525c77 refactor: 101.79.17.164 → zioinfo.co.kr 전체 도메인 변환 + Manager UI 배포
- 37개 파일 IP → zioinfo.co.kr 치환 (소스/매뉴얼/설정/하네스)
- Manager DrConsole/NetworkConsole/CsapConsole 빌드 + /var/www/manager/ 배포
- 테스트: Manager HTTP 200, ITSM 신규 API 7개 전체 200

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-31 10:09:17 +09:00

178 lines
7.4 KiB
Swift

// Copyright 2022-present 650 Industries. All rights reserved.
@objc(ExpoFabricView)
open class ExpoFabricView: ExpoFabricViewObjC, AnyExpoView {
/**
A weak reference to the app context associated with this view.
The app context is injected into the class after the context is initialized.
see the `makeClass` static function.
*/
public weak var appContext: AppContext? { ExpoFabricView.appContextFromClass() }
/**
The view definition that setup from `ExpoFabricView.create()`.
*/
private var viewDefinition: AnyViewDefinition?
/**
A dictionary of prop objects that contain prop setters.
*/
lazy var viewManagerPropDict: [String: AnyViewProp]? = viewDefinition?.propsDict()
// MARK: - Initializers
// swiftlint:disable unavailable_function
@objc
public init() {
// For derived views, their initializer should be replaced by the 'class_replaceMethod'.
fatalError("Unsupported direct init() call for ExpoFabricView.")
}
// swiftlint:enable unavailable_function
@objc
public override init(frame: CGRect) {
super.init(frame: frame)
}
required public init(appContext: AppContext? = nil) {
super.init(frame: .zero)
}
/**
The view creator expected to be called for derived ExpoFabricView, the `viewDefinition` and event dispatchers will be setup from here.
NOTE: We swizzle the initializers, e.g. `ViewManagerAdapter_ExpoImage.new()` to `ImageView.init(appContext:)`
and we also need viewDefintion (or moduleName) for the `installEventDispatchers()`.
Swizzling ExpoFabricView doesn't give us chance to inject iMethod or iVar of ImageView and pass the moduleName.
Alternatively, we try to add a dedicated `ExpoFabricView.create()` and passing viewDefinition into the class.
That's not a perfect implementation but turns out to be the only way to get the viewDefinition (or moduleName).
The example call flow would be:
`ViewManagerAdapter_ExpoImage.new()` -> `ViewDefinition.createView()` -> `ExpoFabricView.create()` ->
`ImageView.init(appContext:)` -> `ExpoFabricView.init(appContext:)` -> `view.viewDefinition = viewDefinition` here
*/
internal static func create(viewType: ExpoFabricView.Type, viewDefinition: AnyViewDefinition, appContext: AppContext) -> ExpoFabricView {
let view = viewType.init(appContext: appContext)
view.viewDefinition = viewDefinition
assert(appContext == view.appContext)
view.installEventDispatchers()
return view
}
// Mark the required init as unavailable so that subclasses can avoid overriding it.
@available(*, unavailable)
public required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
// MARK: - ExpoFabricViewInterface
public override func updateProps(_ props: [String: Any]) {
guard let context = appContext, let propsDict = viewManagerPropDict else {
return
}
for (key, prop) in propsDict {
let newValue = props[key] as Any
// TODO: @tsapeta: Figure out better way to rethrow errors from here.
// Adding `throws` keyword to the function results in different
// method signature in Objective-C. Maybe just call `RCTLogError`?
try? prop.set(value: Conversions.fromNSObject(newValue), onView: self, appContext: context)
}
}
/**
Calls lifecycle methods registered by `OnViewDidUpdateProps` definition component.
*/
public override func viewDidUpdateProps() {
guard let viewDefinition else {
return
}
viewDefinition.callLifecycleMethods(withType: .didUpdateProps, forView: self)
}
/**
Returns a bool value whether the view supports prop with the given name.
*/
public override func supportsProp(withName name: String) -> Bool {
return viewManagerPropDict?.index(forKey: name) != nil
}
// MARK: - Privates
/**
Installs convenient event dispatchers for declared events, so the view can just invoke the block to dispatch the proper event.
*/
private func installEventDispatchers() {
viewDefinition?.eventNames.forEach { eventName in
installEventDispatcher(forEvent: eventName, onView: self) { [weak self] (body: [String: Any]) in
if let self = self {
self.dispatchEvent(eventName, payload: body)
} else {
log.error("Cannot dispatch an event while the managing ExpoFabricView is deallocated")
}
}
}
}
// MARK: - Statics
internal static var viewClassesRegistry = [String: AnyClass]()
/**
Dynamically creates a subclass of the `ExpoFabricView` class with injected app context and name of the associated module.
The new subclass is saved in the registry, so when asked for the next time, it's returned from cache with the updated app context.
- Note: Apple's documentation says that classes created with `objc_allocateClassPair` should then be registered using `objc_registerClassPair`,
but we can't do that as there might be more than one class with the same name (Expo Go) and allocating another one would return `nil`.
*/
@objc
public static func makeViewClass(forAppContext appContext: AppContext, className: String) -> AnyClass? {
let moduleName = String(className.dropFirst(ViewModuleWrapper.viewManagerAdapterPrefix.count))
if let viewClass = viewClassesRegistry[className] {
inject(appContext: appContext)
injectInitializer(appContext: appContext, moduleName: moduleName, toViewClass: viewClass)
return viewClass
}
guard let viewClass = objc_allocateClassPair(ExpoFabricView.self, className, 0) else {
fatalError("Cannot allocate a Fabric view class for '\(className)'")
}
inject(appContext: appContext)
injectInitializer(appContext: appContext, moduleName: moduleName, toViewClass: viewClass)
// Save the allocated view class in the registry for the later use (e.g. when the app is reloaded).
viewClassesRegistry[className] = viewClass
return viewClass
}
internal static func inject(appContext: AppContext) {
// Keep it weak so we don't leak the app context.
weak var weakAppContext = appContext
let appContextBlock: @convention(block) () -> AppContext? = { weakAppContext }
let appContextBlockImp: IMP = imp_implementationWithBlock(appContextBlock)
class_replaceMethod(object_getClass(ExpoFabricView.self), #selector(appContextFromClass), appContextBlockImp, "@@:")
}
internal static func injectInitializer(appContext: AppContext, moduleName: String, toViewClass viewClass: AnyClass) {
// The default initializer for native views. It will be called by Fabric.
let newBlock: @convention(block) () -> Any = {[weak appContext] in
guard let appContext, let moduleHolder = appContext.moduleRegistry.get(moduleHolderForName: moduleName) else {
fatalError(Exceptions.AppContextLost().reason)
}
guard let view = moduleHolder.definition.view?.createView(appContext: appContext) else {
fatalError("Cannot create a view from module '\(moduleName)'")
}
_ = Unmanaged.passRetained(view) // retain the view given this is an initializer
return view
}
let newBlockImp: IMP = imp_implementationWithBlock(newBlock)
class_replaceMethod(object_getClass(viewClass), Selector("new"), newBlockImp, "@@:")
}
// swiftlint:disable unavailable_function
@objc
private dynamic static func appContextFromClass() -> AppContext? {
fatalError("The AppContext must be injected in the 'ExpoFabricView' class")
}
// swiftlint:enable unavailable_function
}