{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "SyntaxError",
     "evalue": "invalid syntax (409617474.py, line 21)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;36m  Cell \u001b[0;32mIn[1], line 21\u001b[0;36m\u001b[0m\n\u001b[0;31m    ball_y = max_height * (1 - (1 - 2 * t) ** 2) - abs(max_height - 64) Adjusted for downward movement\u001b[0m\n\u001b[0m                                                                        ^\u001b[0m\n\u001b[0;31mSyntaxError\u001b[0m\u001b[0;31m:\u001b[0m invalid syntax\n"
     ]
    }
   ],
   "source": [
    "from PIL import Image, ImageDraw\n",
    "\n",
    "def create_frame_straight_down(ball_y, frame_size=(64, 64), ball_radius=5):\n",
    "    \"\"\"Create a single frame with the ball falling straight down along the y-axis.\"\"\"\n",
    "    frame = Image.new(\"RGB\", frame_size, \"white\")\n",
    "    draw = ImageDraw.Draw(frame)\n",
    "    # Draw the ball at a fixed x-coordinate\n",
    "    x_position = frame_size[0] // 2\n",
    "    top_left = (x_position - ball_radius, ball_y - ball_radius)\n",
    "    bottom_right = (x_position + ball_radius, ball_y + ball_radius)\n",
    "    draw.ellipse([top_left, bottom_right], fill=\"blue\")\n",
    "    return frame\n",
    "\n",
    "def generate_downward_fall_frames(num_frames=10, frame_size=(64, 64)):\n",
    "    \"\"\"Generate frames for a ball falling downwards and bouncing up.\"\"\"\n",
    "    frames = []\n",
    "    max_height = frame_size[1] // 2  # Maximum height of the bounce\n",
    "    for i in range(num_frames):\n",
    "        # Calculate y position of the ball (simple bounce logic)\n",
    "        t = i / (num_frames - 1)  # normalized time\n",
    "        ball_y = max_height * (1 - (1 - 2 * t) ** 2) - abs(max_height - 64) Adjusted for downward movement\n",
    "        frame = create_frame_straight_down(ball_y, frame_size)\n",
    "        frames.append(frame)\n",
    "    return frames\n",
    "\n",
    "\n",
    "# Generate frames for the animation\n",
    "animation_frames = generate_downward_fall_frames()\n",
    "\n",
    "# Display the first frame as an example\n",
    "animation_frames[0].show()\n",
    "animation_frames[9].show()\n",
    "\n",
    "animation_frames[0].save(\n",
    "    './bounce_animation.gif',\n",
    "    save_all=True,\n",
    "    append_images=animation_frames[1:],\n",
    "    duration=100,  # duration for each frame in milliseconds\n",
    "    loop=0  # loop forever\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10, 64, 64])\n"
     ]
    }
   ],
   "source": [
    "from torchvision.transforms import ToTensor\n",
    "import torch\n",
    "import random\n",
    "\n",
    "\n",
    "def create_bw_frame(ball_y, ball_x, frame_size=(64, 64), ball_radius=5):\n",
    "    \"\"\"Create a single black and white frame with the ball at the specified y position.\"\"\"\n",
    "    frame = Image.new(\"L\", frame_size, \"white\")  # 'L' mode for black and white (luminance)\n",
    "    draw = ImageDraw.Draw(frame)\n",
    "    x_position = ball_x\n",
    "    top_left = (x_position - ball_radius, ball_y - ball_radius)\n",
    "    bottom_right = (x_position + ball_radius, ball_y + ball_radius)\n",
    "    draw.ellipse([top_left, bottom_right], fill=\"black\")\n",
    "    return frame\n",
    "\n",
    "def generate_bw_animation_frames(num_frames=10, frame_size=(64, 64)):\n",
    "    \"\"\"Generate a series of black and white frames for the animation.\"\"\"\n",
    "    frames = []\n",
    "    ball_x = random.randrange(8, 56, 1)\n",
    "    max_height = random.randrange(20, 56, 1)\n",
    "    for i in range(num_frames):\n",
    "        t = i / (num_frames - 1)\n",
    "        ball_y = max_height * (1 - (1 - 2 * t) ** 2) +(64 - max_height)\n",
    "        frame = create_bw_frame(ball_y, ball_x, frame_size)\n",
    "        frames.append(frame)\n",
    "    return frames\n",
    "\n",
    "# Generate black and white frames\n",
    "bw_frames = generate_bw_animation_frames()\n",
    "\n",
    "# Convert frames to PyTorch tensors\n",
    "tensor_frames = torch.stack([ToTensor()(frame) for frame in bw_frames], dim=0).squeeze(1)\n",
    "\n",
    "# Example of one frame as a tensor\n",
    "print(tensor_frames.shape)  # Should be [1, 64, 64] as it's a single channel image\n",
    "\n",
    "bw_frames[0].save(\n",
    "    './bounce_animation.gif',\n",
    "    save_all=True,\n",
    "    append_images=bw_frames[1:],\n",
    "    duration=100,  # duration for each frame in milliseconds\n",
    "    loop=0  # loop forever\n",
    ")\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "mcvd",
   "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.8.18"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
