| import functools |
| import random |
|
|
| import gradio as gr |
| import torch |
|
|
| from fabric.generator import AttentionBasedGenerator |
|
|
|
|
| |
| model_name = "" |
| model_ckpt = "https://huggingface.co/Lykon/DreamShaper/blob/main/DreamShaper_7_pruned.safetensors" |
|
|
| class GeneratorWrapper: |
| def __init__(self, model_name=None, model_ckpt=None): |
| self.model_name = model_name if model_name else None |
| self.model_ckpt = model_ckpt if model_ckpt else None |
| self.dtype = torch.float16 if torch.cuda.is_available() else torch.float32 |
| self.device = "cuda" if torch.cuda.is_available() else "cpu" |
|
|
| self.reload() |
|
|
| def generate(self, *args, **kwargs): |
| if not hasattr(self, "generator"): |
| self.reload() |
| return self.generator.generate(*args, **kwargs) |
|
|
| def to(self, device): |
| return self.generator.to(device) |
|
|
| def reload(self): |
| if hasattr(self, "generator"): |
| del self.generator |
| if self.device == "cuda": |
| torch.cuda.empty_cache() |
| self.generator = AttentionBasedGenerator( |
| model_name=self.model_name, |
| model_ckpt=self.model_ckpt, |
| torch_dtype=self.dtype, |
| ).to(self.device) |
|
|
| generator = GeneratorWrapper(model_name, model_ckpt) |
|
|
|
|
| css = """ |
| .btn-green { |
| background-image: linear-gradient(to bottom right, #86efac, #22c55e) !important; |
| border-color: #22c55e !important; |
| color: #166534 !important; |
| } |
| .btn-green:hover { |
| background-image: linear-gradient(to bottom right, #86efac, #86efac) !important; |
| } |
| .btn-red { |
| background: linear-gradient(to bottom right, #fda4af, #fb7185) !important; |
| border-color: #fb7185 !important; |
| color: #9f1239 !important; |
| } |
| .btn-red:hover {background: linear-gradient(to bottom right, #fda4af, #fda4af) !important;} |
| |
| /*****/ |
| |
| .dark .btn-green { |
| background-image: linear-gradient(to bottom right, #047857, #065f46) !important; |
| border-color: #047857 !important; |
| color: #ffffff !important; |
| } |
| .dark .btn-green:hover { |
| background-image: linear-gradient(to bottom right, #047857, #047857) !important; |
| } |
| .dark .btn-red { |
| background: linear-gradient(to bottom right, #be123c, #9f1239) !important; |
| border-color: #be123c !important; |
| color: #ffffff !important; |
| } |
| .dark .btn-red:hover {background: linear-gradient(to bottom right, #be123c, #be123c) !important;} |
| """ |
|
|
| def generate_fn( |
| feedback_enabled, |
| max_feedback_imgs, |
| prompt, |
| neg_prompt, |
| liked, |
| disliked, |
| denoising_steps, |
| guidance_scale, |
| feedback_start, |
| feedback_end, |
| min_weight, |
| max_weight, |
| neg_scale, |
| batch_size, |
| seed, |
| progress=gr.Progress(track_tqdm=True), |
| ): |
| try: |
| if seed < 0: |
| seed = random.randint(1,9999999999999999) |
| print("seed: ", seed) |
|
|
| max_feedback_imgs = max(0, int(max_feedback_imgs)) |
| total_images = (len(liked) if liked else 0) + (len(disliked) if disliked else 0) |
|
|
| if not feedback_enabled: |
| liked = [] |
| disliked = [] |
| elif total_images > max_feedback_imgs: |
| if liked and disliked: |
| max_disliked = min(len(disliked), max_feedback_imgs // 2) |
| max_liked = min(len(liked), max_feedback_imgs - max_disliked) |
| if max_liked > len(liked): |
| max_disliked = max_feedback_imgs - max_liked |
| liked = liked[-max_liked:] |
| disliked = disliked[-max_disliked:] |
| elif liked: |
| liked = liked[-max_feedback_imgs:] |
| disliked = [] |
| else: |
| liked = [] |
| disliked = disliked[-max_feedback_imgs:] |
| |
| |
| generate_kwargs = { |
| "prompt": prompt, |
| "negative_prompt": neg_prompt, |
| "liked": liked, |
| "disliked": disliked, |
| "denoising_steps": denoising_steps, |
| "guidance_scale": guidance_scale, |
| "feedback_start": feedback_start, |
| "feedback_end": feedback_end, |
| "min_weight": min_weight, |
| "max_weight": max_weight, |
| "neg_scale": neg_scale, |
| "seed": seed, |
| "n_images": batch_size, |
| } |
|
|
| try: |
| images = generator.generate(**generate_kwargs) |
| except RuntimeError as err: |
| if 'out of memory' in str(err): |
| generator.reload() |
| raise |
| return [(img, f"Image {i+1}") for i, img in enumerate(images)], images, seed |
| except Exception as err: |
| raise gr.Error(str(err)) |
|
|
|
|
| def add_img_from_list(i, curr_imgs, all_imgs): |
| if all_imgs is None: |
| all_imgs = [] |
| if i >= 0 and i < len(curr_imgs): |
| all_imgs.append(curr_imgs[i]) |
| return all_imgs, all_imgs |
|
|
| def add_img(img, all_imgs): |
| if all_imgs is None: |
| all_imgs = [] |
| all_imgs.append(img) |
| return None, all_imgs, all_imgs |
|
|
| def remove_img_from_list(event: gr.SelectData, imgs): |
| if event.index >= 0 and event.index < len(imgs): |
| imgs.pop(event.index) |
| return imgs, imgs |
|
|
| def duplicate_seed_value(seed): |
| return seed |
|
|
| with gr.Blocks(css=css) as demo: |
|
|
| liked_imgs = gr.State([]) |
| disliked_imgs = gr.State([]) |
| curr_imgs = gr.State([]) |
|
|
| with gr.Row(): |
| with gr.Column(scale=100): |
| prompt = gr.Textbox(label="Prompt") |
| neg_prompt = gr.Textbox(label="Negative prompt", value="lowres, bad anatomy, bad hands, cropped, worst quality") |
| submit_btn = gr.Button("Generate", variant="primary", min_width="96px") |
|
|
| with gr.Row(equal_height=False): |
| with gr.Column(): |
| denoising_steps = gr.Slider(1, 100, value=20, step=1, label="Sampling steps") |
| guidance_scale = gr.Slider(0.0, 30.0, value=6, step=0.25, label="CFG scale") |
| batch_size = gr.Slider(1, 10, value=4, step=1, label="Batch size", interactive=False) |
| seed = gr.Number(-1, minimum=-1, precision=0, label="Seed") |
| max_feedback_imgs = gr.Slider(0, 20, value=6, step=1, label="Max. feedback images", info="Maximum number of liked/disliked images to be used. If exceeded, only the most recent images will be used as feedback. (NOTE: large number of feedback imgs => high VRAM requirements)") |
| feedback_enabled = gr.Checkbox(True, label="Enable feedback", interactive=True) |
|
|
| with gr.Accordion("Liked Images", open=True): |
| liked_img_input = gr.Image(type="pil", shape=(512, 512), height=128, label="Upload liked image") |
| like_gallery = gr.Gallery(label="👍 Liked images (click to remove)", columns=[3, 4, 3, 4, 5, 6], height=256, allow_preview=False) |
| clear_liked_btn = gr.Button("Clear likes") |
|
|
| with gr.Accordion("Disliked Images", open=True): |
| disliked_img_input = gr.Image(type="pil", shape=(512, 512), height=128, label="Upload disliked image") |
| dislike_gallery = gr.Gallery(label="👎 Disliked images (click to remove)", columns=[3, 4, 3, 4, 5, 6], height=256, allow_preview=False) |
| clear_disliked_btn = gr.Button("Clear dislikes") |
|
|
| with gr.Accordion("Feedback parameters", open=False): |
| feedback_start = gr.Slider(0.0, 1.0, value=0.0, label="Feedback start", info="Fraction of denoising steps starting from which to use max. feedback weight.") |
| feedback_end = gr.Slider(0.0, 1.0, value=0.8, label="Feedback end", info="Up to what fraction of denoising steps to use max. feedback weight.") |
| feedback_min_weight = gr.Slider(0.0, 1.0, value=0.0, label="Feedback min. weight", info="Attention weight of feedback images when turned off (set to 0.0 to disable)") |
| feedback_max_weight = gr.Slider(0.0, 1.0, value=0.8, label="Feedback max. weight", info="Attention weight of feedback images when turned on (set to 0.0 to disable)") |
| feedback_neg_scale = gr.Slider(0.0, 1.0, value=0.5, label="Neg. feedback scale", info="Attention weight of disliked images relative to liked images (set to 0.0 to disable negative feedback)") |
|
|
| with gr.Column(): |
| gallery = gr.Gallery(label="Generated images") |
|
|
| like_btns = [] |
| dislike_btns = [] |
| with gr.Row(): |
| for i in range(0, 2): |
| like_btn = gr.Button(f"👍 Image {i+1}", elem_classes="btn-green") |
| like_btns.append(like_btn) |
| with gr.Row(): |
| for i in range(2, 4): |
| like_btn = gr.Button(f"👍 Image {i+1}", elem_classes="btn-green") |
| like_btns.append(like_btn) |
| with gr.Row(): |
| for i in range(0, 2): |
| dislike_btn = gr.Button(f"👎 Image {i+1}", elem_classes="btn-red") |
| dislike_btns.append(dislike_btn) |
| with gr.Row(): |
| for i in range(2, 4): |
| dislike_btn = gr.Button(f"👎 Image {i+1}", elem_classes="btn-red") |
| dislike_btns.append(dislike_btn) |
|
|
| prev_seed = gr.Number(-1, label="Previous seed", interactive=False) |
| prev_seed_hid = gr.Number(-1, visible=False) |
|
|
| generate_params = [ |
| feedback_enabled, |
| max_feedback_imgs, |
| prompt, |
| neg_prompt, |
| liked_imgs, |
| disliked_imgs, |
| denoising_steps, |
| guidance_scale, |
| feedback_start, |
| feedback_end, |
| feedback_min_weight, |
| feedback_max_weight, |
| feedback_neg_scale, |
| batch_size, |
| seed, |
| ] |
| submit_btn.click(generate_fn, generate_params, [gallery, curr_imgs, prev_seed_hid], queue=True) |
| prev_seed_hid.change(duplicate_seed_value, prev_seed_hid, prev_seed, queue=False) |
| |
| for i, like_btn in enumerate(like_btns): |
| like_btn.click(functools.partial(add_img_from_list, i), [curr_imgs, liked_imgs], [like_gallery, liked_imgs], queue=False) |
| for i, dislike_btn in enumerate(dislike_btns): |
| dislike_btn.click(functools.partial(add_img_from_list, i), [curr_imgs, disliked_imgs], [dislike_gallery, disliked_imgs], queue=False) |
|
|
| like_gallery.select(remove_img_from_list, [liked_imgs], [like_gallery, liked_imgs], queue=False) |
| dislike_gallery.select(remove_img_from_list, [disliked_imgs], [dislike_gallery, disliked_imgs], queue=False) |
|
|
| liked_img_input.upload(add_img, [liked_img_input, liked_imgs], [liked_img_input, like_gallery, liked_imgs], queue=False) |
| disliked_img_input.upload(add_img, [disliked_img_input, disliked_imgs], [disliked_img_input, dislike_gallery, disliked_imgs], queue=False) |
|
|
| clear_liked_btn.click(lambda: [[], []], None, [liked_imgs, like_gallery], queue=False) |
| clear_disliked_btn.click(lambda: [[], []], None, [disliked_imgs, dislike_gallery], queue=False) |
|
|
| demo.queue(1) |
| demo.launch(debug=True) |