import {
  BufferLoader,
  clone,
  CompressedTextureBuffer,
  GltfBufferViewLoader,
  GltfImageLoader,
  FeatureDetection,
  Resource,
  ResourceCache,
  when,
} from "../../Source/Cesium.js";
import createContext from "../createContext.js";
import dataUriToBuffer from "../dataUriToBuffer.js";
import pollToPromise from "../pollToPromise.js";

describe("Scene/GltfImageLoader", function () {
  var image = new Image();
  image.src =
    "";

  var pngBuffer = dataUriToBuffer(
    ""
  );
  var jpgBuffer = dataUriToBuffer(
    ""
  );

  var webpBuffer = dataUriToBuffer(
    ""
  );

  var gifBuffer = dataUriToBuffer(
    ""
  );

  var ktx2BasisBuffer;
  var ktx2BasisMipmapBuffer;

  var gltfUri = "https://example.com/model.glb";
  var gltfResource = new Resource({
    url: gltfUri,
  });

  var gltf = {
    buffers: [
      {
        uri: "external.bin",
        byteLength: 0, // updated in getGltf
      },
    ],
    bufferViews: [
      {
        buffer: 0,
        byteOffset: 0,
        byteLength: 0, // updated in getGltf
      },
    ],
    images: [
      {
        mimeType: "image/png",
        bufferView: 0,
      },
      {
        uri: "image.png",
      },
      {
        mimeType: "image/ktx2",
        bufferView: 0,
      },
      {
        uri: "image.ktx2",
      },
    ],
  };

  var context;

  function getGltf(imageBuffer) {
    var clonedGltf = clone(gltf, true);
    clonedGltf.buffers[0].byteLength = imageBuffer.byteLength;
    clonedGltf.bufferViews[0].byteLength = imageBuffer.byteLength;
    return clonedGltf;
  }

  beforeAll(function () {
    context = createContext();
    var ktx2BasisBufferPromise = Resource.fetchArrayBuffer({
      url: "./Data/Images/Green4x4_ETC1S.ktx2",
    }).then(function (arrayBuffer) {
      ktx2BasisBuffer = new Uint8Array(arrayBuffer);
    });
    var ktx2BasisMipmapBufferPromise = Resource.fetchArrayBuffer({
      url: "./Data/Images/Green4x4Mipmap_ETC1S.ktx2",
    }).then(function (arrayBuffer) {
      ktx2BasisMipmapBuffer = new Uint8Array(arrayBuffer);
    });

    return when.all([ktx2BasisBufferPromise, ktx2BasisMipmapBufferPromise]);
  });

  afterAll(function () {
    context.destroyForSpecs();
  });

  afterEach(function () {
    ResourceCache.clearForSpecs();
  });

  it("throws if resourceCache is undefined", function () {
    expect(function () {
      return new GltfImageLoader({
        resourceCache: undefined,
        gltf: gltf,
        imageId: 0,
        gltfResource: gltfResource,
        baseResource: gltfResource,
      });
    }).toThrowDeveloperError();
  });

  it("throws if gltf is undefined", function () {
    expect(function () {
      return new GltfImageLoader({
        resourceCache: ResourceCache,
        gltf: undefined,
        imageId: 0,
        gltfResource: gltfResource,
        baseResource: gltfResource,
      });
    }).toThrowDeveloperError();
  });

  it("throws if imageId is undefined", function () {
    expect(function () {
      return new GltfImageLoader({
        resourceCache: ResourceCache,
        gltf: gltf,
        imageId: undefined,
        gltfResource: gltfResource,
        baseResource: gltfResource,
      });
    }).toThrowDeveloperError();
  });

  it("throws if gltfResource is undefined", function () {
    expect(function () {
      return new GltfImageLoader({
        resourceCache: ResourceCache,
        gltf: gltf,
        imageId: 0,
        gltfResource: undefined,
        baseResource: gltfResource,
      });
    }).toThrowDeveloperError();
  });

  it("throws if baseResource is undefined", function () {
    expect(function () {
      return new GltfImageLoader({
        resourceCache: ResourceCache,
        gltf: gltf,
        imageId: 0,
        gltfResource: gltfResource,
        baseResource: undefined,
      });
    }).toThrowDeveloperError();
  });

  it("rejects promise if buffer view fails to load", function () {
    var error = new Error("404 Not Found");
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      when.reject(error)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise
      .then(function (imageLoader) {
        fail();
      })
      .otherwise(function (runtimeError) {
        expect(runtimeError.message).toBe(
          "Failed to load embedded image\nFailed to load buffer view\nFailed to load external buffer: https://example.com/external.bin\n404 Not Found"
        );
      });
  });

  it("rejects promise if image format is not recognized", function () {
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      when.resolve(gifBuffer)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(gifBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise
      .then(function (imageLoader) {
        fail();
      })
      .otherwise(function (runtimeError) {
        expect(runtimeError.message).toBe(
          "Failed to load embedded image\nImage format is not recognized"
        );
      });
  });

  it("rejects promise if uri fails to load", function () {
    var error = new Error("404 Not Found");
    spyOn(Resource.prototype, "fetchImage").and.returnValue(when.reject(error));

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 1,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise
      .then(function (imageLoader) {
        fail();
      })
      .otherwise(function (runtimeError) {
        expect(runtimeError.message).toBe(
          "Failed to load image: image.png\n404 Not Found"
        );
      });
  });

  function loadsFromBufferView(imageBuffer) {
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      when.resolve(imageBuffer)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(imageBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image.width).toBe(1);
      expect(imageLoader.image.height).toBe(1);
    });
  }

  it("loads PNG from buffer view", function () {
    return loadsFromBufferView(pngBuffer);
  });

  it("loads JPEG from buffer view", function () {
    return loadsFromBufferView(jpgBuffer);
  });

  it("loads WebP from buffer view", function () {
    return pollToPromise(function () {
      FeatureDetection.supportsWebP.initialize();
      return FeatureDetection.supportsWebP.initialized;
    }).then(function () {
      if (!FeatureDetection.supportsWebP()) {
        return;
      }
      return loadsFromBufferView(webpBuffer);
    });
  });

  it("loads KTX2/Basis from buffer view", function () {
    if (!context.supportsBasis) {
      return;
    }

    spyOn(BufferLoader, "_fetchArrayBuffer").and.returnValue(
      when.resolve(ktx2BasisBuffer)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(ktx2BasisBuffer),
      imageId: 2,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true);
      expect(imageLoader.image.width).toBe(4);
      expect(imageLoader.image.height).toBe(4);
      expect(imageLoader.mipLevels).toBeUndefined();
    });
  });

  it("loads KTX2/Basis with mipmap from buffer view", function () {
    if (!context.supportsBasis) {
      return;
    }

    spyOn(BufferLoader, "_fetchArrayBuffer").and.returnValue(
      when.resolve(ktx2BasisMipmapBuffer)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(ktx2BasisMipmapBuffer),
      imageId: 2,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true);
      expect(imageLoader.image.width).toBe(4);
      expect(imageLoader.image.height).toBe(4);
      expect(imageLoader.mipLevels.length).toBe(2);
    });
  });

  it("loads from uri", function () {
    spyOn(Resource.prototype, "fetchImage").and.returnValue(
      when.resolve(image)
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: clone(gltf, true),
      imageId: 1,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image.width).toBe(1);
      expect(imageLoader.image.height).toBe(1);
    });
  });

  it("loads KTX2/Basis from uri ", function () {
    if (!context.supportsBasis) {
      return;
    }

    var baseResource = new Resource({
      url: "./Data/Images/",
    });

    var clonedGltf = clone(gltf, true);
    clonedGltf.images[3].uri = "Green4x4_ETC1S.ktx2";

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: clonedGltf,
      imageId: 3,
      gltfResource: gltfResource,
      baseResource: baseResource,
    });

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true);
      expect(imageLoader.image.width).toBe(4);
      expect(imageLoader.image.height).toBe(4);
      expect(imageLoader.mipLevels).toBeUndefined();
    });
  });

  it("destroys image loader", function () {
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      when.resolve(pngBuffer)
    );

    var unloadBufferView = spyOn(
      GltfBufferViewLoader.prototype,
      "unload"
    ).and.callThrough();

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });
    expect(imageLoader.image).not.toBeDefined();

    imageLoader.load();

    return imageLoader.promise.then(function (imageLoader) {
      expect(imageLoader.image).toBeDefined();
      expect(imageLoader.isDestroyed()).toBe(false);

      imageLoader.destroy();

      expect(imageLoader.image).not.toBeDefined();
      expect(imageLoader.isDestroyed()).toBe(true);
      expect(unloadBufferView).toHaveBeenCalled();
    });
  });

  function resolveBufferViewAfterDestroy(reject) {
    var deferredPromise = when.defer();
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      deferredPromise.promise
    );

    // Load a copy of the buffer view into the cache so that the buffer view
    // promise resolves even if the image loader is destroyed
    var bufferViewLoaderCopy = ResourceCache.loadBufferView({
      gltf: getGltf(pngBuffer),
      bufferViewId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    expect(imageLoader.image).not.toBeDefined();

    imageLoader.load();
    imageLoader.destroy();

    if (reject) {
      deferredPromise.reject(new Error());
    } else {
      deferredPromise.resolve(pngBuffer);
    }

    expect(imageLoader.image).not.toBeDefined();
    expect(imageLoader.isDestroyed()).toBe(true);

    ResourceCache.unload(bufferViewLoaderCopy);
  }

  it("handles resolving buffer view after destroy", function () {
    resolveBufferViewAfterDestroy(false);
  });

  it("handles rejecting buffer view after destroy", function () {
    resolveBufferViewAfterDestroy(true);
  });

  function resolveImageFromTypedArrayAfterDestroy(reject) {
    spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue(
      when.resolve(pngBuffer)
    );

    var deferredPromise = when.defer();
    spyOn(GltfImageLoader, "_loadImageFromTypedArray").and.returnValue(
      deferredPromise.promise
    );

    // Load a copy of the buffer view into the cache so that the buffer view
    // promise resolves even if the image loader is destroyed
    var bufferViewLoaderCopy = ResourceCache.loadBufferView({
      gltf: getGltf(pngBuffer),
      bufferViewId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 0,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    expect(imageLoader.image).not.toBeDefined();

    imageLoader.load();
    imageLoader.destroy();

    if (reject) {
      deferredPromise.reject(new Error());
    } else {
      deferredPromise.resolve(image);
    }

    expect(imageLoader.image).not.toBeDefined();
    expect(imageLoader.isDestroyed()).toBe(true);

    ResourceCache.unload(bufferViewLoaderCopy);
  }

  it("handles resolving image from typed array after destroy", function () {
    resolveImageFromTypedArrayAfterDestroy(false);
  });

  it("handles rejecting image from typed array after destroy", function () {
    resolveImageFromTypedArrayAfterDestroy(true);
  });

  function resolveUriAfterDestroy(reject) {
    var deferredPromise = when.defer();
    spyOn(Resource.prototype, "fetchImage").and.returnValue(
      deferredPromise.promise
    );

    var imageLoader = new GltfImageLoader({
      resourceCache: ResourceCache,
      gltf: getGltf(pngBuffer),
      imageId: 1,
      gltfResource: gltfResource,
      baseResource: gltfResource,
    });

    expect(imageLoader.image).not.toBeDefined();

    imageLoader.load();
    imageLoader.destroy();

    if (reject) {
      deferredPromise.reject(new Error());
    } else {
      deferredPromise.resolve(image);
    }

    expect(imageLoader.image).not.toBeDefined();
    expect(imageLoader.isDestroyed()).toBe(true);
  }

  it("handles resolving uri after destroy", function () {
    resolveUriAfterDestroy(false);
  });

  it("handles rejecting uri after destroy", function () {
    resolveUriAfterDestroy(true);
  });
});