|
import os |
|
from datetime import datetime |
|
|
|
import streamlit as st |
|
|
|
from lib import config, preset, txt2img_generate |
|
|
|
|
|
SERVICE_SESSION = { |
|
"Black Forest Labs": "api_key_black_forest_labs", |
|
"Fal": "api_key_fal", |
|
"Hugging Face": "api_key_hugging_face", |
|
"Together": "api_key_together", |
|
} |
|
|
|
SESSION_TOKEN = { |
|
"api_key_black_forest_labs": os.environ.get("BFL_API_KEY") or None, |
|
"api_key_fal": os.environ.get("FAL_KEY") or None, |
|
"api_key_hugging_face": os.environ.get("HF_TOKEN") or None, |
|
"api_key_together": os.environ.get("TOGETHER_API_KEY") or None, |
|
} |
|
|
|
|
|
|
|
PRESET_MODEL = { |
|
|
|
"flux-pro-1.1": preset.txt2img.flux_1_1_pro_bfl, |
|
"flux-pro": preset.txt2img.flux_pro_bfl, |
|
"flux-dev": preset.txt2img.flux_dev_bfl, |
|
|
|
"fal-ai/aura-flow": preset.txt2img.aura_flow, |
|
"fal-ai/flux/dev": preset.txt2img.flux_dev_fal, |
|
"fal-ai/flux/schnell": preset.txt2img.flux_schnell_fal, |
|
"fal-ai/flux-pro": preset.txt2img.flux_pro_fal, |
|
"fal-ai/flux-pro/v1.1": preset.txt2img.flux_1_1_pro_fal, |
|
"fal-ai/fooocus": preset.txt2img.fooocus, |
|
"fal-ai/kolors": preset.txt2img.kolors, |
|
"fal-ai/stable-diffusion-v3-medium": preset.txt2img.stable_diffusion_3, |
|
|
|
"black-forest-labs/flux.1-dev": preset.txt2img.flux_dev_hf, |
|
"black-forest-labs/flux.1-schnell": preset.txt2img.flux_schnell_hf, |
|
"stabilityai/stable-diffusion-xl-base-1.0": preset.txt2img.stable_diffusion_xl, |
|
|
|
"black-forest-labs/FLUX.1-schnell-Free": preset.txt2img.flux_schnell_free_together, |
|
} |
|
|
|
st.set_page_config( |
|
page_title=f"{config.title} | Text to Image", |
|
page_icon=config.icon, |
|
layout=config.layout, |
|
) |
|
|
|
|
|
if "api_key_black_forest_labs" not in st.session_state: |
|
st.session_state.api_key_black_forest_labs = "" |
|
|
|
if "api_key_fal" not in st.session_state: |
|
st.session_state.api_key_fal = "" |
|
|
|
if "api_key_hugging_face" not in st.session_state: |
|
st.session_state.api_key_hugging_face = "" |
|
|
|
if "api_key_together" not in st.session_state: |
|
st.session_state.api_key_together = "" |
|
|
|
if "running" not in st.session_state: |
|
st.session_state.running = False |
|
|
|
if "txt2img_messages" not in st.session_state: |
|
st.session_state.txt2img_messages = [] |
|
|
|
if "txt2img_seed" not in st.session_state: |
|
st.session_state.txt2img_seed = 0 |
|
|
|
st.logo("logo.png") |
|
st.sidebar.header("Settings") |
|
service = st.sidebar.selectbox( |
|
"Service", |
|
options=list(SERVICE_SESSION.keys()), |
|
disabled=st.session_state.running, |
|
index=2, |
|
) |
|
|
|
|
|
|
|
for display_name, session_key in SERVICE_SESSION.items(): |
|
if service == display_name: |
|
st.session_state[session_key] = st.sidebar.text_input( |
|
"API Key", |
|
type="password", |
|
value="" if SESSION_TOKEN[session_key] else st.session_state[session_key], |
|
disabled=bool(SESSION_TOKEN[session_key]) or st.session_state.running, |
|
help="Set by environment variable" if SESSION_TOKEN[session_key] else "Cleared on page refresh", |
|
) |
|
|
|
model = st.sidebar.selectbox( |
|
"Model", |
|
options=config.txt2img.models[service], |
|
index=config.txt2img.default_model[service], |
|
disabled=st.session_state.running, |
|
) |
|
|
|
|
|
st.html(""" |
|
<h1>Text to Image</h1> |
|
<p>Generate an image from a text prompt.</p> |
|
""") |
|
|
|
|
|
parameters = {} |
|
preset = PRESET_MODEL[model] |
|
for param in preset.parameters: |
|
if param == "model": |
|
parameters[param] = model |
|
if param == "seed": |
|
parameters[param] = st.sidebar.number_input( |
|
"Seed", |
|
min_value=-1, |
|
max_value=(1 << 53) - 1, |
|
value=-1, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "negative_prompt": |
|
parameters[param] = st.sidebar.text_area( |
|
"Negative Prompt", |
|
value=config.txt2img.negative_prompt, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "width": |
|
parameters[param] = st.sidebar.slider( |
|
"Width", |
|
step=64, |
|
value=1024, |
|
min_value=512, |
|
max_value=2048, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "height": |
|
parameters[param] = st.sidebar.slider( |
|
"Height", |
|
step=64, |
|
value=1024, |
|
min_value=512, |
|
max_value=2048, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "image_size": |
|
parameters[param] = st.sidebar.select_slider( |
|
"Image Size", |
|
options=config.txt2img.image_sizes, |
|
value=config.txt2img.default_image_size, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "aspect_ratio": |
|
parameters[param] = st.sidebar.select_slider( |
|
"Aspect Ratio", |
|
options=config.txt2img.aspect_ratios, |
|
value=config.txt2img.default_aspect_ratio, |
|
disabled=st.session_state.running, |
|
) |
|
if param in ["guidance_scale", "guidance"]: |
|
parameters[param] = st.sidebar.slider( |
|
"Guidance Scale", |
|
preset.guidance_scale_min, |
|
preset.guidance_scale_max, |
|
preset.guidance_scale, |
|
0.1, |
|
disabled=st.session_state.running, |
|
) |
|
if param in ["num_inference_steps", "steps"]: |
|
parameters[param] = st.sidebar.slider( |
|
"Inference Steps", |
|
preset.num_inference_steps_min, |
|
preset.num_inference_steps_max, |
|
preset.num_inference_steps, |
|
1, |
|
disabled=st.session_state.running, |
|
) |
|
if param in ["expand_prompt", "prompt_expansion"]: |
|
parameters[param] = st.sidebar.checkbox( |
|
"Prompt Expansion", |
|
value=False, |
|
disabled=st.session_state.running, |
|
) |
|
if param == "prompt_upsampling": |
|
parameters[param] = st.sidebar.checkbox( |
|
"Prompt Upsampling", |
|
value=False, |
|
disabled=st.session_state.running, |
|
) |
|
|
|
|
|
for message in st.session_state.txt2img_messages: |
|
role = message["role"] |
|
with st.chat_message(role): |
|
image_container = st.empty() |
|
|
|
with image_container.container(): |
|
if role == "user": |
|
with st.expander(message["content"]): |
|
|
|
st.html(""" |
|
<style> |
|
div[data-testid="stMarkdownContainer"] p:not(:last-of-type) { margin-bottom: 0 } |
|
</style> |
|
""") |
|
filtered_parameters = [ |
|
f"`{k}`: {v}" |
|
for k, v in message["parameters"].items() |
|
if k not in config.txt2img.hidden_parameters |
|
] |
|
st.markdown(f"`model`: {message['model']}\n\n" + "\n\n".join(filtered_parameters)) |
|
|
|
if role == "assistant": |
|
|
|
st.html(""" |
|
<style> |
|
div[data-testid="stImage"]:has(img[style*="max-width: 100%"]) { |
|
height: auto; |
|
max-width: 512px; |
|
} |
|
div[data-testid="stImage"] img[style*="max-width: 100%"] { |
|
border-radius: 8px; |
|
} |
|
</style> |
|
""") |
|
st.write(message["content"]) |
|
|
|
|
|
if st.session_state.txt2img_messages: |
|
button_container = st.empty() |
|
with button_container.container(): |
|
|
|
st.html(""" |
|
<style> |
|
div[data-testid="column"] { |
|
width: fit-content; |
|
min-width: 0; |
|
flex: none; |
|
} |
|
</style> |
|
""") |
|
|
|
col1, col2 = st.columns(2) |
|
with col1: |
|
if ( |
|
st.button("β", help="Delete last generation", disabled=st.session_state.running) |
|
and len(st.session_state.txt2img_messages) >= 2 |
|
): |
|
st.session_state.txt2img_messages.pop() |
|
st.session_state.txt2img_messages.pop() |
|
st.rerun() |
|
|
|
with col2: |
|
if st.button("ποΈ", help="Clear all generations", disabled=st.session_state.running): |
|
st.session_state.txt2img_messages = [] |
|
st.session_state.txt2img_seed = 0 |
|
st.rerun() |
|
else: |
|
button_container = None |
|
|
|
|
|
|
|
if prompt := st.chat_input( |
|
"What do you want to see?", |
|
on_submit=lambda: setattr(st.session_state, "running", True), |
|
): |
|
if "seed" in parameters and parameters["seed"] >= 0: |
|
st.session_state.txt2img_seed = parameters["seed"] |
|
else: |
|
st.session_state.txt2img_seed = int(datetime.now().timestamp() * 1e6) % (1 << 53) |
|
if "seed" in parameters: |
|
parameters["seed"] = st.session_state.txt2img_seed |
|
|
|
if button_container: |
|
button_container.empty() |
|
|
|
with st.chat_message("user"): |
|
st.markdown(prompt) |
|
|
|
with st.chat_message("assistant"): |
|
with st.spinner("Running..."): |
|
if preset.kwargs: |
|
parameters.update(preset.kwargs) |
|
session_key = f"api_key_{service.lower().replace(' ', '_')}" |
|
api_key = st.session_state[session_key] or SESSION_TOKEN[session_key] |
|
image = txt2img_generate(api_key, service, model, prompt, parameters) |
|
st.session_state.running = False |
|
|
|
st.session_state.txt2img_messages.append( |
|
{"role": "user", "content": prompt, "parameters": parameters, "model": PRESET_MODEL[model].name} |
|
) |
|
st.session_state.txt2img_messages.append({"role": "assistant", "content": image}) |
|
st.rerun() |
|
|