///
/// HybridNitroImageViewComponent.mm
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
/// https://github.com/mrousavy/nitro
/// Copyright © Marc Rousavy @ Margelo
///

#import "HybridNitroImageViewComponent.hpp"
#import <memory>
#import <react/renderer/componentregistry/ComponentDescriptorProvider.h>
#import <React/RCTViewComponentView.h>
#import <React/RCTComponentViewFactory.h>
#import <React/UIView+ComponentViewProtocol.h>
#import <NitroModules/NitroDefines.hpp>
#import <UIKit/UIKit.h>

#import "HybridNitroImageViewSpecSwift.hpp"
#import "NitroImage-Swift-Cxx-Umbrella.hpp"

#if __has_include(<cxxreact/ReactNativeVersion.h>)
#include <cxxreact/ReactNativeVersion.h>
#if REACT_NATIVE_VERSION_MINOR >= 82
#define ENABLE_RCT_COMPONENT_VIEW_INVALIDATE
#endif
#endif

using namespace facebook;
using namespace margelo::nitro::image;
using namespace margelo::nitro::image::views;

/**
 * Represents the React Native View holder for the Nitro "NitroImageView" HybridView.
 */
@interface HybridNitroImageViewComponent: RCTViewComponentView
+ (BOOL)shouldBeRecycled;
@end

@implementation HybridNitroImageViewComponent {
  std::shared_ptr<HybridNitroImageViewSpecSwift> _hybridView;
}

+ (void) load {
  [super load];
  [RCTComponentViewFactory.currentComponentViewFactory registerComponentViewClass:[HybridNitroImageViewComponent class]];
}

+ (react::ComponentDescriptorProvider) componentDescriptorProvider {
  return react::concreteComponentDescriptorProvider<HybridNitroImageViewComponentDescriptor>();
}

- (instancetype) init {
  if (self = [super init]) {
    std::shared_ptr<HybridNitroImageViewSpec> hybridView = NitroImage::NitroImageAutolinking::createNitroImageView();
    _hybridView = std::dynamic_pointer_cast<HybridNitroImageViewSpecSwift>(hybridView);
    [self updateView];
  }
  return self;
}

- (void) updateView {
  // 1. Get Swift part
  NitroImage::HybridNitroImageViewSpec_cxx& swiftPart = _hybridView->getSwiftPart();

  // 2. Get UIView*
  void* viewUnsafe = swiftPart.getView();
  UIView* view = (__bridge_transfer UIView*) viewUnsafe;

  // 3. Update RCTViewComponentView's [contentView]
  [self setContentView:view];
}

- (void) updateProps:(const std::shared_ptr<const react::Props>&)props
            oldProps:(const std::shared_ptr<const react::Props>&)oldProps {
  // 1. Downcast props
  const auto& newViewPropsConst = *std::static_pointer_cast<HybridNitroImageViewProps const>(props);
  auto& newViewProps = const_cast<HybridNitroImageViewProps&>(newViewPropsConst);
  NitroImage::HybridNitroImageViewSpec_cxx& swiftPart = _hybridView->getSwiftPart();

  // 2. Update each prop individually
  swiftPart.beforeUpdate();

  // image: optional
  if (newViewProps.image.isDirty) {
    swiftPart.setImage(newViewProps.image.value);
    newViewProps.image.isDirty = false;
  }
  // resizeMode: optional
  if (newViewProps.resizeMode.isDirty) {
    swiftPart.setResizeMode(newViewProps.resizeMode.value);
    newViewProps.resizeMode.isDirty = false;
  }
  // recyclingKey: optional
  if (newViewProps.recyclingKey.isDirty) {
    swiftPart.setRecyclingKey(newViewProps.recyclingKey.value);
    newViewProps.recyclingKey.isDirty = false;
  }

  swiftPart.afterUpdate();

  // 3. Update hybridRef if it changed
  if (newViewProps.hybridRef.isDirty) {
    // hybridRef changed - call it with new this
    const auto& maybeFunc = newViewProps.hybridRef.value;
    if (maybeFunc.has_value()) {
      maybeFunc.value()(_hybridView);
    }
    newViewProps.hybridRef.isDirty = false;
  }

  // 4. Continue in base class
  [super updateProps:props oldProps:oldProps];
}

+ (BOOL)shouldBeRecycled {
  return NitroImage::NitroImageAutolinking::isNitroImageViewRecyclable();
}

- (void)prepareForRecycle {
  [super prepareForRecycle];
  NitroImage::HybridNitroImageViewSpec_cxx& swiftPart = _hybridView->getSwiftPart();
  swiftPart.maybePrepareForRecycle();
}

#ifdef ENABLE_RCT_COMPONENT_VIEW_INVALIDATE
- (void)invalidate {
  NitroImage::HybridNitroImageViewSpec_cxx& swiftPart = _hybridView->getSwiftPart();
  swiftPart.onDropView();
  [super invalidate];
}
#endif

@end
