{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "5f351819-e848-4d02-888d-6b64b92637fa",
   "metadata": {},
   "source": [
    "### Rotation with cropping"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2a9b400c-838b-47cd-a99a-2fb00ffb725d",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6d197ebd-df2f-4b0d-b1a7-e9a794f88c72",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "from os.path import join, exists, expanduser\n",
    "from genericpath import isdir\n",
    "from glob import glob\n",
    "import numpy as np\n",
    "from PIL import Image\n",
    "from collections import defaultdict\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import math\n",
    "\n",
    "from lib.r2d2.extract import extract_keypoints_modified\n",
    "from relfm.utils.paths import REPO_PATH\n",
    "from relfm.utils.log import print_update, tqdm_iterator\n",
    "from relfm.utils.geometry import (\n",
    "    append_rotation_to_homography,\n",
    "    get_image_corners,\n",
    "    apply_homography_to_keypoints,\n",
    "    apply_clean_rotation,\n",
    "    center_crop, \n",
    "    resize,\n",
    ")\n",
    "from relfm.utils.visualize import (\n",
    "    show_images_with_keypoints,\n",
    "    set_latex_fonts, get_colors,\n",
    "    show_grid_of_images,\n",
    "    draw_kps_on_image,\n",
    ")\n",
    "from relfm.utils.matching import evaluate_matching_with_rotation, analyze_result\n",
    "from relfm.inference.r2d2_on_hpatches import configure_save_dir"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e700840-6259-456e-ba14-2ffa7e0ff30d",
   "metadata": {},
   "outputs": [],
   "source": [
    "set_latex_fonts(show_sample=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aa365bce-beff-4332-b3f9-21c082b05216",
   "metadata": {},
   "outputs": [],
   "source": [
    "def add_margin(pil_img, top, right, bottom, left, color=(0, 0, 0)):\n",
    "    \"\"\"Ref: https://note.nkmk.me/en/python-pillow-add-margin-expand-canvas/\"\"\"\n",
    "    width, height = pil_img.size\n",
    "    new_width = width + right + left\n",
    "    new_height = height + top + bottom\n",
    "    result = Image.new(pil_img.mode, (new_width, new_height), color)\n",
    "    result.paste(pil_img, (left, top))\n",
    "    return result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "847eead5-a42f-41ba-8ed6-49a87d285b0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "sequence = \"../data/hpatches-sequences-release/i_castle/\"\n",
    "\n",
    "source_img_path = join(sequence, \"1.ppm\")\n",
    "source_img = Image.open(source_img_path)\n",
    "\n",
    "target_img_path = join(sequence, \"2.ppm\")\n",
    "target_img = Image.open(target_img_path)\n",
    "\n",
    "rotation = 45\n",
    "target_img_rot = target_img.rotate(rotation)\n",
    "\n",
    "pad = 100\n",
    "target_img_pad = add_margin(target_img, pad, pad, pad, pad)\n",
    "target_img_pad_rot = target_img_pad.rotate(rotation)\n",
    "\n",
    "H = np.loadtxt(join(sequence, \"H_1_2\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4109b4b6-dea3-4fc3-ba44-0057d83902f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "show_grid_of_images(\n",
    "    [source_img, target_img, target_img_rot, target_img_pad, target_img_pad_rot],\n",
    "    subtitles=[\"Source\", \"Target\", \"Target rotated\", \"Padded\", \"Pad $\\\\rightarrow$ Rotate\"],\n",
    "    n_cols=5,\n",
    "    figsize=(15, 3)\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c2ed6610-8524-4ae2-bc39-4da0ea8ca210",
   "metadata": {},
   "outputs": [],
   "source": [
    "final_images = [target_img, target_img_rot]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fe8171e6-963a-4400-989e-ba0da9573834",
   "metadata": {},
   "source": [
    "### Final code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3be206cb-40a6-4e0e-a334-38378b43fb9d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# def center_crop(im, new_width, new_height):\n",
    "#     width, height = im.size\n",
    "\n",
    "#     left = (width - new_width)/2\n",
    "#     top = (height - new_height)/2\n",
    "#     right = (width + new_width)/2\n",
    "#     bottom = (height + new_height)/2\n",
    "\n",
    "#     # Crop the center of the image\n",
    "#     im = im.crop((left, top, right, bottom))\n",
    "    \n",
    "#     H_crop = np.eye(3)\n",
    "#     H_crop[0, 2] = -left\n",
    "#     H_crop[1, 2] = -top\n",
    "    \n",
    "#     return im, H_crop\n",
    "\n",
    "\n",
    "# def resize(im, new_width, new_height):\n",
    "#     width, height = im.size\n",
    "#     im = im.resize((new_width, new_height))\n",
    "\n",
    "#     sx = new_width / width\n",
    "#     sy = new_height / height\n",
    "#     H_resize = np.array([\n",
    "#         [sx, 0., 0.],\n",
    "#         [0., sy, 0.],\n",
    "#         [0., 0., 1.],\n",
    "#     ])\n",
    "    \n",
    "#     return im, H_resize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "039b4e7d-a4e5-4ba3-832c-969253440a99",
   "metadata": {},
   "outputs": [],
   "source": [
    "image = target_img.copy()\n",
    "degrees = rotation\n",
    "rotated = target_img.rotate(degrees)\n",
    "\n",
    "# apply rotation homography\n",
    "H_rot = append_rotation_to_homography(H=np.eye(3), rotation=rotation, width=image.size[0], height=image.size[1])\n",
    "\n",
    "# crop rotated image\n",
    "# credits: https://stackoverflow.com/questions/21346670/cropping-rotated-image-with-same-aspect-ratio\n",
    "aspect_ratio = float(image.size[0]) / image.size[1]\n",
    "rotated_aspect_ratio = float(rotated.size[0]) / rotated.size[1]\n",
    "angle = math.fabs(degrees) * math.pi / 180\n",
    "\n",
    "if aspect_ratio < 1:\n",
    "    total_height = float(image.size[0]) / rotated_aspect_ratio\n",
    "else:\n",
    "    total_height = float(image.size[1])\n",
    "\n",
    "h = total_height / (aspect_ratio * math.sin(angle) + math.cos(angle))\n",
    "w = h * aspect_ratio\n",
    "\n",
    "rotated_cropped, H_crop = center_crop(rotated, w, h)\n",
    "\n",
    "# resize rotated_cropped image to original dimensions\n",
    "# credits: https://stackoverflow.com/questions/16646183/crop-an-image-in-the-centre-using-pil\n",
    "W, H = image.size\n",
    "rotated_cropped_resized, H_resize = resize(rotated_cropped, W, H)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "48a825b8-0f25-4d97-9cc2-4c7053cf0a1d",
   "metadata": {},
   "outputs": [],
   "source": [
    "assert rotated_cropped_resized.size == image.size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fedef30c-775f-45f0-9a18-39eef403e7ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "show_grid_of_images(\n",
    "    [image, rotated, rotated_cropped, rotated_cropped_resized],\n",
    "    subtitles=[\"Original\", \"Rotated\", \"Rotated + Cropped\", \"R + C + Resized\"],\n",
    "    figsize=(15, 4),\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ca0e4d0a-5ace-4fde-94d8-43c876095c4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "kps = np.array([\n",
    "    [300., 200.], \n",
    "    [200., 300.],\n",
    "    [300., 400.],\n",
    "    [400., 400.],\n",
    "])\n",
    "\n",
    "kps_rotated = apply_homography_to_keypoints(kps, H_rot)\n",
    "\n",
    "H_combined = H_crop @ H_rot\n",
    "# kps_rotated_cropped = apply_homography_to_keypoints(kps_rotated, H_crop)\n",
    "kps_rotated_cropped = apply_homography_to_keypoints(kps, H_combined)\n",
    "\n",
    "H_combined = H_resize @ H_crop @ H_rot\n",
    "kps_rotated_cropped_resized = apply_homography_to_keypoints(kps, H_combined)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2dd648b6-5651-4259-a7ec-05456adca4c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "H_resize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fade7d89-ec71-43f9-9cf2-1042b6659a72",
   "metadata": {},
   "outputs": [],
   "source": [
    "kps_rotated_cropped"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "517e9cb4-67bb-445d-abe0-8918814c1f1c",
   "metadata": {},
   "outputs": [],
   "source": [
    "kps_rotated_cropped_resized"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7d895840-e439-4879-a034-87d40c6da7fe",
   "metadata": {},
   "outputs": [],
   "source": [
    "image_with_kps = draw_kps_on_image(image, kps, return_as=\"PIL\", radius=8)\n",
    "rotated_with_kps = draw_kps_on_image(rotated, kps_rotated, return_as=\"PIL\", radius=8)\n",
    "rotated_cropped_with_kps = draw_kps_on_image(rotated_cropped, kps_rotated_cropped, return_as=\"PIL\", radius=8)\n",
    "rotated_cropped_resized_with_kps = draw_kps_on_image(\n",
    "    rotated_cropped_resized, kps_rotated_cropped_resized, return_as=\"PIL\", radius=8)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a7b10b50-05c1-4395-91d9-ca59a62dbdcc",
   "metadata": {},
   "outputs": [],
   "source": [
    "show_grid_of_images(\n",
    "    [image_with_kps, rotated_with_kps, rotated_cropped_with_kps, rotated_cropped_resized_with_kps],\n",
    "    subtitles=[\"Original\", \"Rotated\", \"Rotated + Cropped\", \"R + C + Resized\"],\n",
    "    figsize=(15, 4),\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd86caed-1afb-457f-a6d8-287e866c6bfe",
   "metadata": {},
   "outputs": [],
   "source": [
    "show_grid_of_images(\n",
    "    images=[image_with_kps, rotated_with_kps],\n",
    "    subtitles=[\"Source\", \"Target (Transformation: $H$)\"],\n",
    "    save=False,\n",
    "    save_path=\"../Figures/rot_crop_3.pdf\"\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a007719d-eb00-4455-b16f-bd2379d581c5",
   "metadata": {},
   "source": [
    "### Combining it into one transformation function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "60e5736f-80f2-4753-ace6-fd6ac9b7a0a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "# def apply_clean_rotation(image, degrees, H=np.eye(3)):\n",
    "#     \"\"\"\n",
    "#     Applies rotation to an image followed by cropping to clean out black areas.\n",
    "    \n",
    "#     Args:\n",
    "#         image (np.ndarray): input image\n",
    "#         degrees (int): rotation in degrees\n",
    "#         H (np.ndarray): homography matrix, (optional) defaults to $I_{3}$\n",
    "    \n",
    "#     Returns:\n",
    "#         (tuple): transformed image and homography\n",
    "#     \"\"\"\n",
    "#     original_width, original_height = image.size\n",
    "    \n",
    "#     # rotate the image\n",
    "#     rotated = image.rotate(degrees)\n",
    "#     H_rot = append_rotation_to_homography(\n",
    "#         H=H,\n",
    "#         rotation=degrees,\n",
    "#         width=image.size[0],\n",
    "#         height=image.size[1],\n",
    "#     )\n",
    "    \n",
    "#     # crop rotated image to ignore black areas\n",
    "#     # credits: https://stackoverflow.com/questions/21346670/cropping-rotated-image-with-same-aspect-ratio\n",
    "#     aspect_ratio = float(image.size[0]) / image.size[1]\n",
    "#     rotated_aspect_ratio = float(rotated.size[0]) / rotated.size[1]\n",
    "#     angle = math.fabs(degrees) * math.pi / 180\n",
    "#     if aspect_ratio < 1:\n",
    "#         total_height = float(image.size[0]) / rotated_aspect_ratio\n",
    "#     else:\n",
    "#         total_height = float(image.size[1])\n",
    "#     h = total_height / (aspect_ratio * math.sin(angle) + math.cos(angle))\n",
    "#     w = h * aspect_ratio\n",
    "#     rotated_cropped, H_crop = center_crop(rotated, w, h)\n",
    "    \n",
    "#     # resize rotated_cropped image to original dimensions\n",
    "#     # credits: https://stackoverflow.com/questions/16646183/crop-an-image-in-the-centre-using-pil\n",
    "#     W, H = image.size\n",
    "#     rotated_cropped_resized, H_resize = resize(rotated_cropped, W, H)\n",
    "    \n",
    "#     # combine the homographies\n",
    "#     H_combined = H_resize @ H_crop @ H_rot\n",
    "    \n",
    "#     return rotated_cropped_resized, H_combined"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a474658e-65c0-46f3-a8c9-c155b92710ba",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load sample image\n",
    "image = Image.open(target_img_path)\n",
    "H = np.eye(3)\n",
    "\n",
    "# consider a rotation\n",
    "degrees = 45\n",
    "\n",
    "# consider sample keypoints\n",
    "kps = np.array([\n",
    "    [300., 200.],\n",
    "    [200., 300.],\n",
    "    [300., 400.],\n",
    "    [400., 400.],\n",
    "])\n",
    "\n",
    "# apply the transformation\n",
    "image_transformed, H_transformed, _, _ = apply_clean_rotation(\n",
    "    image, degrees, H=H,\n",
    ")\n",
    "\n",
    "# draw keypoints on images\n",
    "kps_on_image = apply_homography_to_keypoints(kps, H)\n",
    "image_with_kps = draw_kps_on_image(image, kps_on_image, radius=10)\n",
    "\n",
    "kps_on_image_transformed = apply_homography_to_keypoints(kps_on_image, H_transformed)\n",
    "image_transformed_with_kps = draw_kps_on_image(image_transformed, kps_on_image_transformed, radius=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dbba5f75-c3e5-4f82-8882-114e21e4001d",
   "metadata": {},
   "outputs": [],
   "source": [
    "show_grid_of_images(\n",
    "    images=[image_with_kps, image_transformed_with_kps],\n",
    "    subtitles=[\"Image\", \"Image Transformed\"],\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0a8f4685-4a49-47cb-acec-a311ddaba1ae",
   "metadata": {},
   "source": [
    "### Test it on a pair of images"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ebf9adfe-da29-49f3-bc6b-6b10a3e74152",
   "metadata": {},
   "outputs": [],
   "source": [
    "source = Image.open(source_img_path)\n",
    "target = Image.open(target_img_path)\n",
    "\n",
    "H = np.loadtxt(join(sequence, \"H_1_2\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cd9c40a2-3e2d-476c-852b-963d5d8f42f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def check_kps_with_homography(\n",
    "        img1, img2, H, kps=None,\n",
    "        num_kps=10, sample=\"random\", **draw_args,\n",
    "    ):\n",
    "    \"\"\"Checks if H correctly transforms keypoints kps from img1 to img2.\"\"\"\n",
    "    if kps is None:\n",
    "        width, height = img1.size\n",
    "        assert sample in [\"random\", \"uniform\"]\n",
    "        \n",
    "        if sample == \"random\":\n",
    "            np.random.seed(0)\n",
    "            kps = np.vstack(\n",
    "                [np.random.randint(0, width, num_kps), np.random.randint(0, height, num_kps)]\n",
    "            ).T\n",
    "        elif sample == \"uniform\":\n",
    "            x = np.linspace(0, width, num_kps)\n",
    "            y = np.linspace(0, height, num_kps)\n",
    "            X, Y = np.meshgrid(x, y)\n",
    "\n",
    "            kps = []\n",
    "            for i in range(num_kps):\n",
    "                for j in range(num_kps):\n",
    "                    kps.append([X[i, j], Y[i, j]])\n",
    "            kps = np.array(kps)\n",
    "    \n",
    "    img1_with_kps = draw_kps_on_image(img1, kps, **draw_args)\n",
    "    \n",
    "    kps_transformed = apply_homography_to_keypoints(kps, H)\n",
    "    img2_with_kps = draw_kps_on_image(img2, kps_transformed, **draw_args)\n",
    "\n",
    "    show_grid_of_images(\n",
    "        images=[img1_with_kps, img2_with_kps],\n",
    "        subtitles=[\"Source\", \"Target (Transformation: $H$)\"],\n",
    "        save=False,\n",
    "        save_path=\"../Figures/rot_crop_2.pdf\"\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f5e1428d-1dd4-46f7-9ecd-ac546b9c90eb",
   "metadata": {},
   "outputs": [],
   "source": [
    "check_kps_with_homography(source, target, H, radius=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ca39dec8-f05b-4071-a47c-82292124a29f",
   "metadata": {},
   "outputs": [],
   "source": [
    "degrees = 45"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eeec85b4-5183-4b30-95d0-9939f72e7a73",
   "metadata": {},
   "source": [
    "#### Apply on source and sanity-check"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "27b4539e-f019-4768-8ff9-e9407cd814d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "_, _, source_transformed, H_source = apply_clean_rotation(image=source, degrees=degrees, H=np.eye(3))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fbaa4a87-daaa-4309-a0db-b82eb019421f",
   "metadata": {},
   "outputs": [],
   "source": [
    "check_kps_with_homography(source, source_transformed, H_source, radius=10)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d5c62bef-2406-44ca-82eb-e7c50b8ca39b",
   "metadata": {},
   "source": [
    "#### Apply on target and sanity-check"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd888bbc-7e4f-4926-b8dd-170b62a277f8",
   "metadata": {},
   "outputs": [],
   "source": [
    "target_transformed, H_target, _, _  = apply_clean_rotation(image=target, degrees=degrees, H=np.eye(3))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "14ee2af5-d44d-4970-9f10-b3f24473de45",
   "metadata": {},
   "outputs": [],
   "source": [
    "check_kps_with_homography(target, target_transformed, H_target, radius=10)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "288e401c-e113-4d11-8c3d-3ae1304d2b3b",
   "metadata": {},
   "source": [
    "#### Combine"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3782d30c-5305-448f-898f-989ca1a63dd4",
   "metadata": {},
   "outputs": [],
   "source": [
    "H_together = H_target @ H @ np.linalg.inv(H_source)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "53f659e2-34cb-40cd-ade2-660bb59ec1ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "check_kps_with_homography(source_transformed, target_transformed, H_together, radius=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "38b9c844-e75a-45db-b75e-ca4336ba3e8d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
