| import torch |
| from PIL import Image |
| from .imagefunc import log, tensor2pil, pil2tensor, image2mask |
|
|
|
|
| class ExtendCanvas: |
|
|
| def __init__(self): |
| self.NODE_NAME = 'ExtendCanvas' |
|
|
| @classmethod |
| def INPUT_TYPES(self): |
|
|
| return { |
| "required": { |
| "image": ("IMAGE", ), |
| "invert_mask": ("BOOLEAN", {"default": True}), |
| "top": ("INT", {"default": 0, "min": 0, "max": 99999, "step": 1}), |
| "bottom": ("INT", {"default": 0, "min": 0, "max": 99999, "step": 1}), |
| "left": ("INT", {"default": 0, "min": 0, "max": 99999, "step": 1}), |
| "right": ("INT", {"default": 0, "min": 0, "max": 99999, "step": 1}), |
| "color": ("COLOR", {"default": "#000000"},), |
| }, |
| "optional": { |
| "mask": ("MASK",), |
| } |
| } |
|
|
| RETURN_TYPES = ("IMAGE", "MASK",) |
| RETURN_NAMES = ("image", "mask") |
| FUNCTION = 'extend_canvas' |
| CATEGORY = '😺dzNodes/LayerUtility' |
|
|
| def extend_canvas(self, image, invert_mask, |
| top, bottom, left, right, color, |
| mask=None, |
| ): |
|
|
| l_images = [] |
| l_masks = [] |
| ret_images = [] |
| ret_masks = [] |
|
|
| for l in image: |
| l_images.append(torch.unsqueeze(l, 0)) |
| m = tensor2pil(l) |
| if m.mode == 'RGBA': |
| l_masks.append(m.split()[-1]) |
|
|
| if mask is not None: |
| if mask.dim() == 2: |
| mask = torch.unsqueeze(mask, 0) |
| l_masks = [] |
| for m in mask: |
| if invert_mask: |
| m = 1 - m |
| l_masks.append(tensor2pil(torch.unsqueeze(m, 0)).convert('L')) |
| else: |
| if len(l_masks) == 0: |
| l_masks.append(Image.new('L', size=tensor2pil(l_images[0]).size, color='white')) |
|
|
| max_batch = max(len(l_images), len(l_masks)) |
| for i in range(max_batch): |
|
|
| _image = l_images[i] if i < len(l_images) else l_images[-1] |
| _image = tensor2pil(_image).convert('RGB') |
| _mask = l_masks[i] if i < len(l_masks) else l_masks[-1] |
|
|
| width = _image.width + left + right |
| height = _image.height + top + bottom |
| _canvas = Image.new('RGB', (width, height), color) |
| _mask_canvas = Image.new('L', (width, height), "black") |
|
|
| _canvas.paste(_image, box=(left,top)) |
| _mask_canvas.paste(_mask.convert('L'), box=(left, top)) |
|
|
| ret_images.append(pil2tensor(_canvas)) |
| ret_masks.append(image2mask(_mask_canvas)) |
|
|
| log(f"{self.NODE_NAME} Processed {len(ret_images)} image(s).", message_type='finish') |
| return (torch.cat(ret_images, dim=0), torch.cat(ret_masks, dim=0),) |
|
|
|
|
| NODE_CLASS_MAPPINGS = { |
| "LayerUtility: ExtendCanvas": ExtendCanvas |
| } |
|
|
| NODE_DISPLAY_NAME_MAPPINGS = { |
| "LayerUtility: ExtendCanvas": "LayerUtility: ExtendCanvas" |
| } |