copyRectify function

Image copyRectify(
  1. Image src, {
  2. required Point topLeft,
  3. required Point topRight,
  4. required Point bottomLeft,
  5. required Point bottomRight,
  6. Interpolation interpolation = Interpolation.nearest,
  7. Image? toImage,
})

Returns a copy of the src image, where the given rectangle has been mapped to the full image.

Implementation

Image copyRectify(Image src,
    {required Point topLeft,
    required Point topRight,
    required Point bottomLeft,
    required Point bottomRight,
    Interpolation interpolation = Interpolation.nearest,
    Image? toImage}) {
  // You can't interpolate index pixels, so we need to convert the image
  // to a non-palette image if non-nearest interpolation is used.
  if (interpolation != Interpolation.nearest && src.hasPalette) {
    src = src.convert(numChannels: src.numChannels);
  }

  Image? firstFrame;
  for (final frame in src.frames) {
    final dst = firstFrame?.addFrame() ??
        toImage ??
        Image.from(frame, noAnimation: true);
    firstFrame ??= dst;
    for (var y = 0; y < dst.height; ++y) {
      final v = y / (dst.height - 1);
      for (var x = 0; x < dst.width; ++x) {
        final u = x / (dst.width - 1);
        // bilinear interpolation
        final srcPixelCoord = topLeft * (1 - u) * (1 - v) +
            topRight * u * (1 - v) +
            bottomLeft * (1 - u) * v +
            bottomRight * u * v;

        final srcPixel = interpolation == Interpolation.nearest
            ? frame.getPixel(srcPixelCoord.xi, srcPixelCoord.yi)
            : frame.getPixelInterpolate(srcPixelCoord.x, srcPixelCoord.y,
                interpolation: interpolation);

        dst.setPixel(x, y, srcPixel);
      }
    }
  }

  return firstFrame!;
}