2023-02-06 01:07:00 +00:00
import os
import cv2
import argparse
import shutil
import math
2023-02-08 01:58:35 +00:00
def resize_images ( src_img_folder , dst_img_folder , max_resolution = " 512x512 " , divisible_by = 1 ) :
2023-02-06 01:07:00 +00:00
# Split the max_resolution string by "," and strip any whitespaces
max_resolutions = [ res . strip ( ) for res in max_resolution . split ( ' , ' ) ]
# # Calculate max_pixels from max_resolution string
# max_pixels = int(max_resolution.split("x")[0]) * int(max_resolution.split("x")[1])
# Create destination folder if it does not exist
if not os . path . exists ( dst_img_folder ) :
os . makedirs ( dst_img_folder )
# Iterate through all files in src_img_folder
for filename in os . listdir ( src_img_folder ) :
# Check if the image is png, jpg or webp
if not filename . endswith ( ( ' .png ' , ' .jpg ' , ' .webp ' ) ) :
# Copy the file to the destination folder if not png, jpg or webp
shutil . copy ( os . path . join ( src_img_folder , filename ) , os . path . join ( dst_img_folder , filename ) )
continue
# Load image
img = cv2 . imread ( os . path . join ( src_img_folder , filename ) )
for max_resolution in max_resolutions :
# Calculate max_pixels from max_resolution string
max_pixels = int ( max_resolution . split ( " x " ) [ 0 ] ) * int ( max_resolution . split ( " x " ) [ 1 ] )
# Calculate current number of pixels
current_pixels = img . shape [ 0 ] * img . shape [ 1 ]
# Check if the image needs resizing
if current_pixels > max_pixels :
# Calculate scaling factor
scale_factor = max_pixels / current_pixels
# Calculate new dimensions
new_height = int ( img . shape [ 0 ] * math . sqrt ( scale_factor ) )
new_width = int ( img . shape [ 1 ] * math . sqrt ( scale_factor ) )
# Resize image
img = cv2 . resize ( img , ( new_width , new_height ) )
# Calculate the new height and width that are divisible by divisible_by
new_height = new_height if new_height % divisible_by == 0 else new_height - new_height % divisible_by
new_width = new_width if new_width % divisible_by == 0 else new_width - new_width % divisible_by
# Center crop the image to the calculated dimensions
y = int ( ( img . shape [ 0 ] - new_height ) / 2 )
x = int ( ( img . shape [ 1 ] - new_width ) / 2 )
img = img [ y : y + new_height , x : x + new_width ]
# Split filename into base and extension
base , ext = os . path . splitext ( filename )
new_filename = base + ' + ' + max_resolution + ' .jpg '
2023-02-08 01:58:35 +00:00
# copy caption file with right name if one exist
if os . path . exists ( os . path . join ( src_img_folder , base + ' .txt ' ) ) :
shutil . copy ( os . path . join ( src_img_folder , base + ' .txt ' ) , os . path . join ( dst_img_folder , new_filename + ' .txt ' ) )
2023-02-06 01:07:00 +00:00
# Save resized image in dst_img_folder
cv2 . imwrite ( os . path . join ( dst_img_folder , new_filename ) , img , [ cv2 . IMWRITE_JPEG_QUALITY , 100 ] )
print ( f " Resized image: { filename } with size { img . shape [ 0 ] } x { img . shape [ 1 ] } as { new_filename } " )
def main ( ) :
parser = argparse . ArgumentParser ( description = ' Resize images in a folder to a specified max resolution(s) ' )
parser . add_argument ( ' src_img_folder ' , type = str , help = ' Source folder containing the images ' )
parser . add_argument ( ' dst_img_folder ' , type = str , help = ' Destination folder to save the resized images ' )
* 2023/02/06 (v20.7.0)
- ``--bucket_reso_steps`` and ``--bucket_no_upscale`` options are added to training scripts (fine tuning, DreamBooth, LoRA and Textual Inversion) and ``prepare_buckets_latents.py``.
- ``--bucket_reso_steps`` takes the steps for buckets in aspect ratio bucketing. Default is 64, same as before.
- Any value greater than or equal to 1 can be specified; 64 is highly recommended and a value divisible by 8 is recommended.
- If less than 64 is specified, padding will occur within U-Net. The result is unknown.
- If you specify a value that is not divisible by 8, it will be truncated to divisible by 8 inside VAE, because the size of the latent is 1/8 of the image size.
- If ``--bucket_no_upscale`` option is specified, images smaller than the bucket size will be processed without upscaling.
- Internally, a bucket smaller than the image size is created (for example, if the image is 300x300 and ``bucket_reso_steps=64``, the bucket is 256x256). The image will be trimmed.
- Implementation of [#130](https://github.com/kohya-ss/sd-scripts/issues/130).
- Images with an area larger than the maximum size specified by ``--resolution`` are downsampled to the max bucket size.
- Now the number of data in each batch is limited to the number of actual images (not duplicated). Because a certain bucket may contain smaller number of actual images, so the batch may contain same (duplicated) images.
- ``--random_crop`` now also works with buckets enabled.
- Instead of always cropping the center of the image, the image is shifted left, right, up, and down to be used as the training data. This is expected to train to the edges of the image.
- Implementation of discussion [#34](https://github.com/kohya-ss/sd-scripts/discussions/34).
2023-02-06 16:04:07 +00:00
parser . add_argument ( ' --max_resolution ' , type = str , help = ' Maximum resolution(s) in the format " 512x512,448x448,384x384, etc, etc " ' , default = " 512x512,448x448,384x384 " )
2023-02-06 01:07:00 +00:00
parser . add_argument ( ' --divisible_by ' , type = int , help = ' Ensure new dimensions are divisible by this value ' , default = 1 )
args = parser . parse_args ( )
resize_images ( args . src_img_folder , args . dst_img_folder , args . max_resolution )
if __name__ == ' __main__ ' :
main ( )