# %%
# Copyright (c) 2021, NVIDIA CORPORATION. All rights reserved.
#
# NVIDIA CORPORATION and its licensors retain all intellectual property
# and proprietary rights in and to this software, related documentation
# and any modifications thereto. Any use, reproduction, disclosure or
# distribution of this software and related documentation without an express
# license agreement from NVIDIA CORPORATION is strictly prohibited.
from networks.mat import Generator
import gradio as gr
import gradio.components as gc
import base64
import glob
import os
import random
import re
from http import HTTPStatus
from io import BytesIO
from typing import Dict, List, NamedTuple, Optional, Tuple
import click
import cv2
import numpy as np
import PIL.Image
import torch
import torch.nn.functional as F
from PIL import Image, ImageDraw, ImageOps
from pydantic import BaseModel
import dnnlib
import legacy
pyspng = None
def num_range(s: str) -> List[int]:
'''Accept either a comma separated list of numbers 'a,b,c' or a range 'a-c' and return as a list of ints.'''
range_re = re.compile(r'^(\d+)-(\d+)$')
m = range_re.match(s)
if m:
return list(range(int(m.group(1)), int(m.group(2))+1))
vals = s.split(',')
return [int(x) for x in vals]
def copy_params_and_buffers(src_module, dst_module, require_all=False):
assert isinstance(src_module, torch.nn.Module)
assert isinstance(dst_module, torch.nn.Module)
src_tensors = {name: tensor for name,
tensor in named_params_and_buffers(src_module)}
for name, tensor in named_params_and_buffers(dst_module):
assert (name in src_tensors) or (not require_all)
if name in src_tensors:
tensor.copy_(src_tensors[name].detach()).requires_grad_(
tensor.requires_grad)
def params_and_buffers(module):
assert isinstance(module, torch.nn.Module)
return list(module.parameters()) + list(module.buffers())
def named_params_and_buffers(module):
assert isinstance(module, torch.nn.Module)
return list(module.named_parameters()) + list(module.named_buffers())
class Inpainter:
def __init__(self,
network_pkl,
resolution=512,
truncation_psi=1,
noise_mode='const',
sdevice='cpu'
):
self.resolution = resolution
self.truncation_psi = truncation_psi
self.noise_mode = noise_mode
print(f'Loading networks from: {network_pkl}')
self.device = torch.device(sdevice)
with dnnlib.util.open_url(network_pkl) as f:
G_saved = (
legacy.load_network_pkl(f)
['G_ema']
.to(self.device)
.eval()
.requires_grad_(False)) # type: ignore
net_res = 512 if resolution > 512 else resolution
self.G = (
Generator(
z_dim=512,
c_dim=0,
w_dim=512,
img_resolution=net_res,
img_channels=3
)
.to(self.device)
.eval()
.requires_grad_(False)
)
copy_params_and_buffers(G_saved, self.G, require_all=True)
def generate_images2(
self,
dpath: List[PIL.Image.Image],
mpath: List[Optional[PIL.Image.Image]],
seed: int = 42,
):
"""
Generate images using pretrained network pickle.
"""
resolution = self.resolution
truncation_psi = self.truncation_psi
noise_mode = self.noise_mode
# seed = 240 # pick up a random number
def seed_all(seed):
random.seed(seed)
np.random.seed(seed)
torch.manual_seed(seed)
torch.cuda.manual_seed(seed)
if seed is not None:
seed_all(seed)
# no Labels.
label = torch.zeros([1, self.G.c_dim], device=self.device)
def read_image(image):
image = np.array(image)
if image.ndim == 2:
image = image[:, :, np.newaxis] # HW => HWC
image = np.repeat(image, 3, axis=2)
image = image.transpose(2, 0, 1) # HWC => CHW
image = image[:3]
return image
if resolution != 512:
noise_mode = 'random'
results = []
with torch.no_grad():
for i, (ipath, m) in enumerate(zip(dpath, mpath)):
if seed is None:
seed_all(i)
image = read_image(ipath)
image = (torch.from_numpy(image).float().to(
self. device) / 127.5 - 1).unsqueeze(0)
mask = np.array(m).astype(np.float32) / 255.0
mask = torch.from_numpy(mask).float().to(
self. device).unsqueeze(0).unsqueeze(0)
z = torch.from_numpy(np.random.randn(
1, self.G.z_dim)).to(self.device)
output = self.G(image, mask, z, label,
truncation_psi=truncation_psi, noise_mode=noise_mode)
output = (output.permute(0, 2, 3, 1) * 127.5 +
127.5).round().clamp(0, 255).to(torch.uint8)
output = output[0].cpu().numpy()
results.append(PIL.Image.fromarray(output, 'RGB'))
return results
# if __name__ == "__main__":
# generate_images() # pylint: disable=no-value-for-parameter
# ----------------------------------------------------------------------------
def mask_to_alpha(img, mask):
img = img.copy()
img.putalpha(mask)
return img
def blend(src, target, mask):
mask = np.expand_dims(mask, axis=-1)
result = (1-mask) * src + mask * target
return Image.fromarray(result.astype(np.uint8))
def pad(img, size=(128, 128), tosize=(512, 512), border=1):
if isinstance(size, float):
size = (int(img.size[0] * size), int(img.size[1] * size))
# remove border
w, h = tosize
new_img = Image.new('RGBA', (w, h))
rimg = img.resize(size, resample=Image.Resampling.NEAREST)
rimg = ImageOps.crop(rimg, border=border)
tw, th = size
tw, th = tw - border*2, th - border*2
tc = ((w-tw)//2, (h-th)//2)
new_img.paste(rimg, tc)
mask = Image.new('L', (w, h))
white = Image.new('L', (tw, th), 255)
mask.paste(white, tc)
if 'A' in rimg.getbands():
mask.paste(rimg.getchannel('A'), tc)
return new_img, mask
def b64_to_img(b64):
return Image.open(BytesIO(base64.b64decode(b64)))
def img_to_b64(img):
with BytesIO() as f:
img.save(f, format='PNG')
return base64.b64encode(f.getvalue()).decode('utf-8')
class Predictor:
def __init__(self):
"""Load the model into memory to make running multiple predictions efficient"""
self.models = {
"places2": Inpainter(
network_pkl='models/Places_512_FullData.pkl',
resolution=512,
truncation_psi=1.,
noise_mode='const',
),
"places2+laion300k": Inpainter(
network_pkl='models/Places_512_FullData+LAION300k.pkl',
resolution=512,
truncation_psi=1.,
noise_mode='const',
),
"places2+laion300k+laion300k(opmasked)": Inpainter(
network_pkl='models/Places_512_FullData+LAION300k+OPM300k.pkl',
resolution=512,
truncation_psi=1.,
noise_mode='const',
),
"places2+laion300k+laion1200k(opmasked)": Inpainter(
network_pkl='models/Places_512_FullData+LAION300k+OPM1200k.pkl',
resolution=512,
truncation_psi=1.,
noise_mode='const',
),
}
# The arguments and types the model takes as input
def predict(
self,
img: Image.Image,
tosize=(512, 512),
border=5,
seed=42,
size=0.5,
model='places2',
) -> Image:
i, m = pad(
img,
size=size, # (328, 328),
tosize=tosize,
border=border
)
"""Run a single prediction on the model"""
imgs = self.models[model].generate_images2(
dpath=[i.resize((512, 512), resample=Image.Resampling.NEAREST)],
mpath=[m.resize((512, 512), resample=Image.Resampling.NEAREST)],
seed=seed,
)
img_op_raw = imgs[0].convert('RGBA')
img_op_raw = img_op_raw.resize(
tosize, resample=Image.Resampling.NEAREST)
inpainted = img_op_raw.copy()
# paste original image to remove inpainting/scaling artifacts
inpainted = blend(
i,
inpainted,
1-(np.array(m) / 255)
)
minpainted = mask_to_alpha(inpainted, m)
return inpainted, minpainted, ImageOps.invert(m)
def predict_tiled(
self,
img: Image.Image,
tosize=(512, 512),
border=5,
seed=42,
size=0.5,
model='places2',
) -> Image:
i, morig = pad(
img,
size=size, # (328, 328),
tosize=tosize,
border=border
)
i.putalpha(morig)
img = i
# img.save('0.png')
assert img.width == img.height
assert img.width > 512 and img.width <= 512*2
def tile_coords(image, n=2, tile_size=512):
assert image.width == image.height
offsets = np.linspace(0, image.width - tile_size, n).astype(int)
for i in range(n):
for j in range(n):
left = offsets[j]
upper = offsets[i]
right = left + tile_size
lower = upper + tile_size
# tile = image.crop((left, upper, right, lower))
yield [left, upper, right, lower]
for ix, tc in enumerate(tile_coords(img, n=2)):
i = img.crop(tc)
# i.save(f't{ix}.png')
m = i.getchannel('A')
"""Run a single prediction on the model"""
imgs = self.models[model].generate_images2(
dpath=[i.resize((512, 512), resample=Image.Resampling.NEAREST)],
mpath=[m.resize((512, 512), resample=Image.Resampling.NEAREST)],
seed=seed,
)
img_op_raw = imgs[0].convert('RGBA')
# img_op_raw = img_op_raw.resize(tosize, resample=Image.Resampling.NEAREST)
inpainted = img_op_raw.copy()
# paste original image to remove inpainting/scaling artifacts
inpainted = blend(
i,
inpainted,
1-(np.array(m) / 255)
)
# inpainted.save(f't{ix}_op.png')
minpainted = mask_to_alpha(inpainted, m)
# continue with partially inpainted image
# since the tiles overlap, the next tile will contain (possibly inpainted) parts of the previous tile
img.paste(inpainted, tc)
# restore original alpha channel
img.putalpha(morig)
return img.convert('RGB'), img, ImageOps.invert(img.getchannel('A'))
predictor = Predictor()
# %%
def _outpaint(img, tosize, border, seed, size, model, tiled):
if tiled:
img_op = predictor.predict_tiled(
img,
border=border,
seed=seed,
tosize=(tosize, tosize),
size=float(size),
model=model,
)
else:
img_op = predictor.predict(
img,
border=border,
seed=seed,
tosize=(tosize, tosize),
size=float(size),
model=model,
)
return img_op
# %%
with gr.Blocks() as demo:
maturl = 'https://github.com/fenglinglwb/MAT'
gr.Markdown(f'''
# MAT Primer for Stable Diffusion
## based on MAT: Mask-Aware Transformer for Large Hole Image Inpainting
### create a primer for use in stable diffusion outpainting
i have added 2 example scripts to the repo:
- outpainting_example1.py using the inpainting pipeline
- outpainting_example2.py using the img2img pipeline. this is basically what i used for the examples below
''')
gr.HTML(f'''{maturl}''')
with gr.Box():
with gr.Row():
gr.Markdown(f"""example with strength 0.5""")
with gr.Row():
gr.HTML(" ")
gr.HTML("")
gr.HTML("")
btn = gr.Button("Run", variant="primary")
with gr.Row():
with gr.Column():
searchimage = gc.Image(label="image", type='pil', image_mode='RGBA')
to_size = gc.Slider(1, 1920, 512, step=1, label='output size')
border = gc.Slider(1, 50, 0, step=1, label='border to crop from the image before outpainting')
seed = gc.Slider(1, 65536, 10, step=1, label='seed')
size = gc.Slider(0, 1, .5, step=0.01,label='scale of the image before outpainting')
tiled = gc.Checkbox(label='tiled: run the network with 4 tiles of size 512x512 . only usable if output size >512 and <=1024', value=False)
model = gc.Dropdown(
choices=['places2',
'places2+laion300k',
'places2+laion300k+laion300k(opmasked)',
'places2+laion300k+laion1200k(opmasked)'],
value='places2+laion300k+laion1200k(opmasked)',
label='model',
)
with gr.Column():
outwithoutalpha = gc.Image(label="primed image without alpha channel", type='pil', image_mode='RGBA')
mask = gc.Image(label="outpainting mask", type='pil')
out = gc.Image(label="primed image with alpha channel",type='pil', image_mode='RGBA')
btn.click(
fn=_outpaint,
inputs=[searchimage, to_size, border, seed, size, model,tiled],
outputs=[outwithoutalpha, out, mask])
# %% launch
demo.launch()