Some checks failed
Python Linting / Run Ruff (push) Has been cancelled
Python Linting / Run Pylint (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.10, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.11, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.12, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-unix-nightly (12.1, , linux, 3.11, [self-hosted Linux], nightly) (push) Has been cancelled
Execution Tests / test (macos-latest) (push) Has been cancelled
Execution Tests / test (ubuntu-latest) (push) Has been cancelled
Execution Tests / test (windows-latest) (push) Has been cancelled
Test server launches without errors / test (push) Has been cancelled
Unit Tests / test (macos-latest) (push) Has been cancelled
Unit Tests / test (ubuntu-latest) (push) Has been cancelled
Unit Tests / test (windows-2022) (push) Has been cancelled
Includes 30 custom nodes committed directly, 7 Civitai-exclusive loras stored via Git LFS, and a setup script that installs all dependencies and downloads HuggingFace-hosted models on vast.ai. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
90 lines
3.4 KiB
Python
90 lines
3.4 KiB
Python
import numpy as np
|
|
from PIL import Image, ImageChops
|
|
import torch
|
|
|
|
class NoisePlusBlend:
|
|
@classmethod
|
|
def INPUT_TYPES(cls):
|
|
return {
|
|
"required": {
|
|
"image": ("IMAGE",),
|
|
"noise_scale": ("FLOAT", {"default": 0.40, "min": 0.00, "max": 100.00, "step": 0.01}),
|
|
"blend_opacity": ("INT", {"default": 20, "min": 0, "max": 100}),
|
|
},
|
|
"optional": {
|
|
"mask": ("MASK",),
|
|
}
|
|
}
|
|
|
|
RETURN_TYPES = ("IMAGE", "IMAGE")
|
|
RETURN_NAMES = ("blended_image_output", "noise_output")
|
|
FUNCTION = "noise_plus_blend"
|
|
|
|
CATEGORY = "ControlAltAI Nodes/Image"
|
|
|
|
def tensor_to_pil(self, tensor_image):
|
|
"""Converts tensor to a PIL Image"""
|
|
tensor_image = tensor_image.squeeze(0) # Remove batch dimension if it exists
|
|
pil_image = Image.fromarray((tensor_image.cpu().numpy() * 255).astype(np.uint8))
|
|
return pil_image
|
|
|
|
def pil_to_tensor(self, pil_image):
|
|
"""Converts a PIL image back to a tensor"""
|
|
return torch.from_numpy(np.array(pil_image).astype(np.float32) / 255).unsqueeze(0)
|
|
|
|
def generate_gaussian_noise(self, width, height, noise_scale=0.05):
|
|
"""Generates Gaussian noise with a given scale."""
|
|
noise = np.random.normal(128, 128 * noise_scale, (height, width, 3)).astype(np.uint8)
|
|
return Image.fromarray(noise)
|
|
|
|
def soft_light_blend(self, base_image, noise_image, mask=None, opacity=15):
|
|
"""Blends noise over the base image using soft light, applying mask if present."""
|
|
# Resize noise to match base image size
|
|
noise_image = noise_image.resize(base_image.size)
|
|
|
|
base_image = base_image.convert('RGB')
|
|
noise_image = noise_image.convert('RGB')
|
|
|
|
noise_blended = ImageChops.soft_light(base_image, noise_image)
|
|
blended_image = Image.blend(base_image, noise_blended, opacity / 100)
|
|
|
|
# Apply mask only if it's provided, valid, and contains more than a single value
|
|
if mask is not None:
|
|
mask_pil = self.tensor_to_pil(mask).convert('L')
|
|
mask_resized = mask_pil.resize(base_image.size)
|
|
|
|
# Invert the mask by subtracting from 255
|
|
inverted_mask = ImageChops.invert(mask_resized)
|
|
|
|
# Apply the inverted mask to the composite blending
|
|
blended_image = Image.composite(base_image, blended_image, inverted_mask)
|
|
|
|
return blended_image
|
|
|
|
def noise_plus_blend(self, image, noise_scale=0.05, blend_opacity=15, mask=None):
|
|
"""Main function to generate noise, blend, and return results."""
|
|
# Convert Tensor image to PIL
|
|
base_image = self.tensor_to_pil(image)
|
|
image_size = base_image.size
|
|
|
|
# Generate Gaussian noise with the size of the input image
|
|
noise_image = self.generate_gaussian_noise(image_size[0], image_size[1], noise_scale)
|
|
|
|
# Blend the noise with the base image using soft light
|
|
blended_image = self.soft_light_blend(base_image, noise_image, mask, blend_opacity)
|
|
|
|
# Convert the final blended image back to tensor
|
|
noise_tensor = self.pil_to_tensor(noise_image)
|
|
blended_tensor = self.pil_to_tensor(blended_image)
|
|
|
|
# Return both the noise and blended image as tensors
|
|
return blended_tensor, noise_tensor
|
|
|
|
NODE_CLASS_MAPPINGS = {
|
|
"NoisePlusBlend": NoisePlusBlend,
|
|
}
|
|
|
|
NODE_DISPLAY_NAME_MAPPINGS = {
|
|
"NoisePlusBlend": "Noise Plus Blend",
|
|
}
|