aboutsummaryrefslogtreecommitdiff
path: root/modules/styles.py
blob: bc7f070ffcd91d5f401789b6be5ba2d07bdd4144 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
# We need this so Python doesn't complain about the unknown StableDiffusionProcessing-typehint at runtime
from __future__ import annotations

import csv
import os
import os.path
import typing
import collections.abc as abc
import tempfile
import shutil

if typing.TYPE_CHECKING:
    # Only import this when code is being type-checked, it doesn't have any effect at runtime
    from .processing import StableDiffusionProcessing


class PromptStyle(typing.NamedTuple):
    name: str
    prompt: str
    negative_prompt: str


def load_styles(path: str) -> dict[str, PromptStyle]:
    styles = {"None": PromptStyle("None", "", "")}

    if os.path.exists(path):
        with open(path, "r", encoding="utf8", newline='') as file:
            reader = csv.DictReader(file)
            for row in reader:
                # Support loading old CSV format with "name, text"-columns
                prompt = row["prompt"] if "prompt" in row else row["text"]
                negative_prompt = row.get("negative_prompt", "")
                styles[row["name"]] = PromptStyle(row["name"], prompt, negative_prompt)

    return styles


def merge_prompts(style_prompt: str, prompt: str) -> str:
    parts = filter(None, (prompt.strip(), style_prompt.strip()))
    return ", ".join(parts)


def apply_style(processing: StableDiffusionProcessing, style: PromptStyle) -> None:
    if isinstance(processing.prompt, list):
        processing.prompt = [merge_prompts(style.prompt, p) for p in processing.prompt]
    else:
        processing.prompt = merge_prompts(style.prompt, processing.prompt)

    if isinstance(processing.negative_prompt, list):
        processing.negative_prompt = [merge_prompts(style.negative_prompt, p) for p in processing.negative_prompt]
    else:
        processing.negative_prompt = merge_prompts(style.negative_prompt, processing.negative_prompt)


def save_styles(path: str, styles: abc.Iterable[PromptStyle]) -> None:
    # Write to temporary file first, so we don't nuke the file if something goes wrong
    fd, temp_path = tempfile.mkstemp(".csv")
    with os.fdopen(fd, "w", encoding="utf8", newline='') as file:
        # _fields is actually part of the public API: typing.NamedTuple is a replacement for collections.NamedTuple,
        # and collections.NamedTuple has explicit documentation for accessing _fields. Same goes for _asdict()
        writer = csv.DictWriter(file, fieldnames=PromptStyle._fields)
        writer.writeheader()
        writer.writerows(style._asdict() for style in styles)

    # Always keep a backup file around
    if os.path.exists(path):
        shutil.move(path, path + ".bak")
    shutil.move(temp_path, path)