2022-08-26 07:15:42 +00:00
|
|
|
"""
|
2022-08-26 02:49:15 +00:00
|
|
|
Two helper classes for dealing with PNG images and their path names.
|
|
|
|
PngWriter -- Converts Images generated by T2I into PNGs, finds
|
|
|
|
appropriate names for them, and writes prompt metadata
|
2022-08-31 04:36:38 +00:00
|
|
|
into the PNG.
|
2022-08-26 02:49:15 +00:00
|
|
|
PromptFormatter -- Utility for converting a Namespace of prompt parameters
|
|
|
|
back into a formatted prompt string with command-line switches.
|
2022-08-26 07:15:42 +00:00
|
|
|
"""
|
2022-08-24 21:52:34 +00:00
|
|
|
import os
|
2022-08-24 23:47:59 +00:00
|
|
|
import re
|
2022-08-31 04:36:38 +00:00
|
|
|
from PIL import PngImagePlugin
|
2022-08-24 21:52:34 +00:00
|
|
|
|
2022-08-25 04:42:37 +00:00
|
|
|
# -------------------image generation utils-----
|
2022-08-28 20:14:29 +00:00
|
|
|
|
|
|
|
|
2022-08-25 04:42:37 +00:00
|
|
|
class PngWriter:
|
2022-08-31 04:21:04 +00:00
|
|
|
def __init__(self, outdir):
|
2022-08-26 07:15:42 +00:00
|
|
|
self.outdir = outdir
|
2022-08-25 04:42:37 +00:00
|
|
|
os.makedirs(outdir, exist_ok=True)
|
|
|
|
|
2022-08-31 04:21:04 +00:00
|
|
|
# gives the next unique prefix in outdir
|
|
|
|
def unique_prefix(self):
|
|
|
|
# sort reverse alphabetically until we find max+1
|
|
|
|
dirlist = sorted(os.listdir(self.outdir), reverse=True)
|
|
|
|
# find the first filename that matches our pattern or return 000000.0.png
|
|
|
|
existing_name = next(
|
|
|
|
(f for f in dirlist if re.match('^(\d+)\..*\.png', f)),
|
|
|
|
'0000000.0.png',
|
|
|
|
)
|
|
|
|
basecount = int(existing_name.split('.', 1)[0]) + 1
|
|
|
|
return f'{basecount:06}'
|
2022-08-25 04:42:37 +00:00
|
|
|
|
2022-08-31 04:21:04 +00:00
|
|
|
# saves image named _image_ to outdir/name, writing metadata from prompt
|
|
|
|
# returns full path of output
|
|
|
|
def save_image_and_prompt_to_png(self, image, prompt, name):
|
|
|
|
path = os.path.join(self.outdir, name)
|
2022-08-25 04:42:37 +00:00
|
|
|
info = PngImagePlugin.PngInfo()
|
2022-08-26 07:15:42 +00:00
|
|
|
info.add_text('Dream', prompt)
|
|
|
|
image.save(path, 'PNG', pnginfo=info)
|
2022-08-31 04:21:04 +00:00
|
|
|
return path
|
2022-08-25 21:26:48 +00:00
|
|
|
|
2022-08-26 07:15:42 +00:00
|
|
|
|
|
|
|
class PromptFormatter:
|
|
|
|
def __init__(self, t2i, opt):
|
2022-08-25 04:42:37 +00:00
|
|
|
self.t2i = t2i
|
|
|
|
self.opt = opt
|
|
|
|
|
2022-08-28 23:01:45 +00:00
|
|
|
# note: the t2i object should provide all these values.
|
|
|
|
# there should be no need to or against opt values
|
2022-08-25 04:42:37 +00:00
|
|
|
def normalize_prompt(self):
|
2022-08-26 07:15:42 +00:00
|
|
|
"""Normalize the prompt and switches"""
|
|
|
|
t2i = self.t2i
|
|
|
|
opt = self.opt
|
2022-08-25 04:42:37 +00:00
|
|
|
|
|
|
|
switches = list()
|
|
|
|
switches.append(f'"{opt.prompt}"')
|
|
|
|
switches.append(f'-s{opt.steps or t2i.steps}')
|
|
|
|
switches.append(f'-W{opt.width or t2i.width}')
|
|
|
|
switches.append(f'-H{opt.height or t2i.height}')
|
|
|
|
switches.append(f'-C{opt.cfg_scale or t2i.cfg_scale}')
|
2022-08-28 23:01:45 +00:00
|
|
|
switches.append(f'-A{opt.sampler_name or t2i.sampler_name}')
|
2022-09-03 07:08:03 +00:00
|
|
|
if opt.seamless or t2i.seamless:
|
|
|
|
switches.append(f'--seamless')
|
2022-08-25 04:42:37 +00:00
|
|
|
if opt.init_img:
|
|
|
|
switches.append(f'-I{opt.init_img}')
|
2022-09-01 04:50:28 +00:00
|
|
|
if opt.fit:
|
|
|
|
switches.append(f'--fit')
|
2022-08-25 04:42:37 +00:00
|
|
|
if opt.strength and opt.init_img is not None:
|
|
|
|
switches.append(f'-f{opt.strength or t2i.strength}')
|
2022-08-26 06:17:14 +00:00
|
|
|
if opt.gfpgan_strength:
|
|
|
|
switches.append(f'-G{opt.gfpgan_strength}')
|
2022-08-28 23:01:45 +00:00
|
|
|
if opt.upscale:
|
|
|
|
switches.append(f'-U {" ".join([str(u) for u in opt.upscale])}')
|
2022-08-25 04:42:37 +00:00
|
|
|
if t2i.full_precision:
|
|
|
|
switches.append('-F')
|
|
|
|
return ' '.join(switches)
|