/**
 * Copyright © 2023-present 650 Industries, Inc. (aka Expo)
 * Copyright © Meta Platforms, Inc. and affiliates.
 *
 * This source code is licensed under the MIT license found in the
 * LICENSE file in the root directory of this source tree.
 */

// This file should not import `react-native` in order to remain self-contained.

/// <reference path="../ts-declarations/whatwg-url-without-unicode.d.ts" />
import { URL, URLSearchParams } from 'whatwg-url-without-unicode';

declare namespace globalThis {
  const RN$Bridgeless: undefined | boolean;
  const nativeModuleProxy: Record<string, unknown>;
  function __turboModuleProxy(name: string): unknown;
}

// TODO(@kitten): Provide BlobModule types matching native module
interface NativeBlobModule {
  BLOB_URI_SCHEME: string;
  BLOB_URI_HOST: string;
}
interface TurboBlobModule {
  getConstants(): {
    BLOB_URI_SCHEME: string;
    BLOB_URI_HOST: string;
  };
}

let isSetup = false;
let BLOB_URL_PREFIX: string | null = null;

function getBlobUrlPrefix() {
  if (isSetup) return BLOB_URL_PREFIX;
  isSetup = true;
  // if iOS: let BLOB_URL_PREFIX = 'blob:'

  // Pull the blob module without importing React Native.
  const BlobModule =
    globalThis.RN$Bridgeless !== true
      ? // Legacy RN implementation
        (globalThis.nativeModuleProxy['BlobModule'] as NativeBlobModule)
      : // Newer RN implementation
        (globalThis.__turboModuleProxy('BlobModule') as TurboBlobModule);

  const constants = 'BLOB_URI_SCHEME' in BlobModule ? BlobModule : BlobModule.getConstants();

  if (constants && typeof constants.BLOB_URI_SCHEME === 'string') {
    BLOB_URL_PREFIX = encodeURIComponent(constants.BLOB_URI_SCHEME) + ':';
    if (typeof constants.BLOB_URI_HOST === 'string') {
      BLOB_URL_PREFIX += `//${encodeURIComponent(constants.BLOB_URI_HOST)}/`;
    }
  }
  return BLOB_URL_PREFIX;
}

declare module 'whatwg-url-without-unicode' {
  // TODO(@kitten): Clarify where this came from
  type BlobLike = Blob & { data?: { blobId: string; offset: number } };

  interface URLConstructor {
    createObjectURL(blob: BlobLike): string;
    revokeObjectURL(url: URL): void;
    canParse(url: string, base?: string): boolean;
  }
}

/**
 * To allow Blobs be accessed via `content://` URIs,
 * you need to register `BlobProvider` as a ContentProvider in your app's `AndroidManifest.xml`:
 *
 * ```xml
 * <manifest>
 *   <application>
 *     <provider
 *       android:name="com.facebook.react.modules.blob.BlobProvider"
 *       android:authorities="@string/blob_provider_authority"
 *       android:exported="false"
 *     />
 *   </application>
 * </manifest>
 * ```
 * And then define the `blob_provider_authority` string in `res/values/strings.xml`.
 * Use a dotted name that's entirely unique to your app:
 *
 * ```xml
 * <resources>
 *   <string name="blob_provider_authority">your.app.package.blobs</string>
 * </resources>
 * ```
 */
URL.createObjectURL = function createObjectURL(blob) {
  if (getBlobUrlPrefix() == null) {
    throw new Error('Cannot create URL for blob');
  }
  return `${getBlobUrlPrefix()}${encodeURIComponent(blob.data!.blobId)}?offset=${encodeURIComponent(
    blob.data!.offset
  )}&size=${encodeURIComponent(blob.size)}`;
};

URL.revokeObjectURL = function revokeObjectURL(_url) {
  // Do nothing.
};

URL.canParse = function canParse(url: string, base?: string): boolean {
  try {
    // eslint-disable-next-line no-new
    new URL(url, base);
    return true;
  } catch {
    return false;
  }
};

export { URL, URLSearchParams };
