Merge branch 'ui'

This commit is contained in:
echo 2026-05-24 12:41:51 +02:00
commit 42b58b02bf
20 changed files with 1741 additions and 1914 deletions

View File

@ -34,7 +34,8 @@ jobs:
- name: Test
run: |
CLAY_DIR="$(pwd)/vendor/clay/bindings/odin/clay-odin"
odin test tests -collection:clay="$CLAY_DIR"
BUILD_DIR="$(pwd)/build"
odin test tests -collection:clay="$CLAY_DIR" -extra-linker-flags:"-L$BUILD_DIR -losdialog"
- name: Package
run: ./scripts/package.sh

3
.gitmodules vendored
View File

@ -1,3 +1,6 @@
[submodule "odin/vendor/clay"]
path = odin/vendor/clay
url = https://github.com/nicbarker/clay.git
[submodule "odin/vendor/osdialog"]
path = odin/vendor/osdialog
url = https://github.com/AndrewBelt/osdialog.git

View File

@ -3,9 +3,15 @@ set -euo pipefail
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
CLAY_DIR="$SCRIPT_DIR/vendor/clay/bindings/odin/clay-odin"
BUILD_DIR="$SCRIPT_DIR/build"
# Build C dependencies
"$SCRIPT_DIR/build_osdialog.sh"
mkdir -p bin
odin build src/app \
-out:bin/comic_odin \
-debug \
-collection:clay="$CLAY_DIR"
-collection:clay="$CLAY_DIR" \
-extra-linker-flags:"-L$BUILD_DIR -losdialog"

BIN
odin/build/libosdialog.a Normal file

Binary file not shown.

35
odin/build_osdialog.sh Executable file
View File

@ -0,0 +1,35 @@
#!/bin/bash
# Build osdialog static library for Linux (zenity backend)
set -e
OSDIALOG_DIR="$(dirname "$0")/vendor/osdialog"
OUTPUT_DIR="$(dirname "$0")/build"
mkdir -p "$OUTPUT_DIR"
LIB_OUT="$OUTPUT_DIR/libosdialog.a"
# Only rebuild if sources changed
SOURCES=("$OSDIALOG_DIR/osdialog.c" "$OSDIALOG_DIR/osdialog_zenity.c")
NEED_REBUILD=false
for src in "${SOURCES[@]}"; do
if [ "$src" -nt "$LIB_OUT" ] 2>/dev/null; then
NEED_REBUILD=true
break
fi
done
if [ "$NEED_REBUILD" = false ] && [ -f "$LIB_OUT" ]; then
echo "osdialog already built. Skipping."
exit 0
fi
echo "Building osdialog (zenity backend)..."
for src in "${SOURCES[@]}"; do
obj="${OUTPUT_DIR}/$(basename "${src%.c}.o")"
echo " CC $src"
gcc -c -O2 -std=c99 -I"$OSDIALOG_DIR" "$src" -o "$obj"
done
ar rcs "$LIB_OUT" "$OUTPUT_DIR"/osdialog.o "$OUTPUT_DIR"/osdialog_zenity.o
echo " AR $LIB_OUT"
echo "osdialog build complete."

1
odin/buildandrun.md Normal file
View File

@ -0,0 +1 @@
comic/odin ui  ? ✗ ./build.sh && ./bin/comic_odin gui

View File

@ -9,515 +9,31 @@
"last_modified_iso": ""
},
"user_mode": 0,
"story_idea": "2 cars racing in down town newyork",
"story_idea": "two balls roli",
"story_genre": "action",
"target_audience": "general",
"art_style": "manga",
"script": {
"title": "Midnight Rush",
"synopsis": "Generated comic synopsis",
"title": "",
"synopsis": "",
"characters": [
],
"pages": [
{
"page_number": 1,
"layout_type": 0,
"panels": [
{
"panel_id": "panel_001_001",
"panel_number": 1,
"shot_type": 2,
"description": "Wide shot of New York City skyline at night, neon lights reflecting on wet streets. Two cars, a red Ferrari and a black Lamborghini, are side by side at a traffic light on a multi-lane avenue.",
"characters_present": [
],
"dialogue": [
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_001_002",
"panel_number": 2,
"shot_type": 2,
"description": "Close-up on the Ferrari driver, a young man in a leather jacket with a determined expression. His hand grips the gear shift.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "This is it.",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_001_003",
"panel_number": 3,
"shot_type": 2,
"description": "Close-up on the Lamborghini driver, a woman with sunglasses and a smirk. She revs the engine.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Ready to lose, pretty boy?",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_001_004",
"panel_number": 4,
"shot_type": 2,
"description": "Traffic light turns green. Both cars accelerate, tires screeching, smoke billowing. Speed lines emphasize motion.",
"characters_present": [
],
"dialogue": [
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_001_005",
"panel_number": 5,
"shot_type": 2,
"description": "Medium shot of the cars weaving through traffic. The Ferrari barely misses a taxi, sparks flying from the curb.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Whoa!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_001_006",
"panel_number": 6,
"shot_type": 2,
"description": "The Lamborghini cuts in front of a bus, horn blaring. The driver laughs.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Ha! Too slow!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
}
]
},
{
"page_number": 2,
"layout_type": 0,
"panels": [
{
"panel_id": "panel_002_001",
"panel_number": 1,
"shot_type": 2,
"description": "Both cars enter a sharp curve. The Ferrari drifts close to a parked car, side mirror clipping it off. Glass shatters.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Sorry!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_002_002",
"panel_number": 2,
"shot_type": 2,
"description": "The Lamborghini takes the inside line, gaining a car length. The Ferrari driver grits his teeth.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "See ya!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_002_003",
"panel_number": 3,
"shot_type": 2,
"description": "Straightaway ahead. The Ferrari spots a shortcut through an alley. He swerves into it.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Not yet!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_002_004",
"panel_number": 4,
"shot_type": 2,
"description": "The alley is narrow, trash cans flying as the Ferrari speeds through. A cat jumps out of the way.",
"characters_present": [
],
"dialogue": [
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_002_005",
"panel_number": 5,
"shot_type": 2,
"description": "The Ferrari exits the alley just ahead of the Lamborghini. Both cars race toward the finish line (a bridge entrance).",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "What?!",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
},
{
"panel_id": "panel_002_006",
"panel_number": 6,
"shot_type": 2,
"description": "The Ferrari crosses the bridge first, winning. The drivers slow down, windows rolled down. The woman nods in respect.",
"characters_present": [
],
"dialogue": [
{
"speaker_id": "",
"text": "Nice move.",
"bubble_type": 0,
"emotion": 4
},
{
"speaker_id": "",
"text": "You're not bad yourself.",
"bubble_type": 0,
"emotion": 4
}
],
"caption": "",
"sound_effects": [
],
"transition_from_previous": 0
}
]
}
]
},
"characters": [
],
"panel_images": {
"panel_001_003": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_003_panel_001_003.png",
"width": 1024,
"height": 1024,
"seed": 3,
"prompt": "local panel 3"
},
"panel_002_001": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_007_panel_002_001.png",
"width": 1024,
"height": 1024,
"seed": 7,
"prompt": "local panel 7"
},
"panel_002_006": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_012_panel_002_006.png",
"width": 1024,
"height": 1024,
"seed": 12,
"prompt": "local panel 12"
},
"panel_001_004": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_004_panel_001_004.png",
"width": 1024,
"height": 1024,
"seed": 4,
"prompt": "local panel 4"
},
"panel_002_005": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_011_panel_002_005.png",
"width": 1024,
"height": 1024,
"seed": 11,
"prompt": "local panel 11"
},
"panel_001_005": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_005_panel_001_005.png",
"width": 1024,
"height": 1024,
"seed": 5,
"prompt": "local panel 5"
},
"panel_002_004": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_010_panel_002_004.png",
"width": 1024,
"height": 1024,
"seed": 10,
"prompt": "local panel 10"
},
"panel_001_006": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_006_panel_001_006.png",
"width": 1024,
"height": 1024,
"seed": 6,
"prompt": "local panel 6"
},
"panel_001_001": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_001_panel_001_001.png",
"width": 1024,
"height": 1024,
"seed": 1,
"prompt": "local panel 1"
},
"panel_002_003": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_009_panel_002_003.png",
"width": 1024,
"height": 1024,
"seed": 9,
"prompt": "local panel 9"
},
"panel_001_002": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_002_panel_001_002.png",
"width": 1024,
"height": 1024,
"seed": 2,
"prompt": "local panel 2"
},
"panel_002_002": {
"url": "file:///tmp/comic-gui-local-panels-0387429411/panel_008_panel_002_002.png",
"width": 1024,
"height": 1024,
"seed": 8,
"prompt": "local panel 8"
}
},
"panel_errors": {
},
"page_layouts": [
{
"page_number": 1,
"pattern_id": "grid-2x2",
"panels": [
{
"panel_id": "panel_001_001",
"panel_number": 1,
"layout_cell": {
"x": 0.02000000,
"y": 0.02000000,
"w": 0.47000000,
"h": 0.47000000
}
},
{
"panel_id": "panel_001_002",
"panel_number": 2,
"layout_cell": {
"x": 0.50999999,
"y": 0.02000000,
"w": 0.47000000,
"h": 0.47000000
}
},
{
"panel_id": "panel_001_003",
"panel_number": 3,
"layout_cell": {
"x": 0.02000000,
"y": 0.50999999,
"w": 0.47000000,
"h": 0.47000000
}
},
{
"panel_id": "panel_001_004",
"panel_number": 4,
"layout_cell": {
"x": 0.50999999,
"y": 0.50999999,
"w": 0.47000000,
"h": 0.47000000
}
}
],
"width": 2480,
"height": 3508
},
{
"page_number": 2,
"pattern_id": "dialogue-heavy",
"panels": [
{
"panel_id": "panel_001_005",
"panel_number": 5,
"layout_cell": {
"x": 0.02000000,
"y": 0.02000000,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_001_006",
"panel_number": 6,
"layout_cell": {
"x": 0.50999999,
"y": 0.02000000,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_001",
"panel_number": 1,
"layout_cell": {
"x": 0.02000000,
"y": 0.25999999,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_002",
"panel_number": 2,
"layout_cell": {
"x": 0.50999999,
"y": 0.25999999,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_003",
"panel_number": 3,
"layout_cell": {
"x": 0.02000000,
"y": 0.50000000,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_004",
"panel_number": 4,
"layout_cell": {
"x": 0.50999999,
"y": 0.50000000,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_005",
"panel_number": 5,
"layout_cell": {
"x": 0.02000000,
"y": 0.74000001,
"w": 0.47000000,
"h": 0.22000000
}
},
{
"panel_id": "panel_002_006",
"panel_number": 6,
"layout_cell": {
"x": 0.50999999,
"y": 0.74000001,
"w": 0.47000000,
"h": 0.22000000
}
}
],
"width": 2480,
"height": 3508
}
],
"speech_bubbles": {
@ -526,7 +42,7 @@
"page_size": 0,
"color_profile": 0,
"workflow": {
"current_step": 5,
"current_step": 0,
"completed_steps": [
],

View File

@ -15,7 +15,8 @@ echo "=> Building comic-odin v${VERSION} (${OS}-${ARCH})"
echo "=> Running test suite"
CLAY_DIR="$ROOT_DIR/vendor/clay/bindings/odin/clay-odin"
odin test tests -collection:clay="$CLAY_DIR"
BUILD_DIR="$ROOT_DIR/build"
odin test tests -collection:clay="$CLAY_DIR" -extra-linker-flags:"-L$BUILD_DIR -losdialog"
mkdir -p dist
PKG_NAME="comic-odin-${VERSION}-${OS}-${ARCH}"

View File

@ -9,20 +9,6 @@ import "../core"
import "../shared"
import "../ui"
action_generate_local_script :: proc(controller: ^ui.App_Controller, pages: int) -> string {
story := controller.state.story_idea
if len(story) == 0 {
story = "A local GUI adventure"
}
script := build_local_script(story, pages)
core.dispose_script(&controller.state.script)
controller.state.script = script
controller.state.characters = controller.state.script.characters
controller.active_screen = .Script
controller.state.workflow.current_step = .Script_Review
return "Generated local script"
}
action_generate_deepseek_script :: proc(controller: ^ui.App_Controller, pages: int) -> string {
cfg := shared.load_config()
if len(cfg.deepseek_api_key) == 0 {
@ -47,74 +33,9 @@ action_generate_deepseek_script :: proc(controller: ^ui.App_Controller, pages: i
return "Generated DeepSeek script"
}
action_generate_local_panels :: proc(controller: ^ui.App_Controller) -> string {
panels := collect_script_panels(controller.state.script)
defer delete(panels)
if len(panels) == 0 {
return "No script panels available"
}
images, ierr := build_local_panel_images(panels)
if !shared.is_ok(ierr) {
return ierr.message
}
for _, img in controller.state.panel_images {
delete(img.url)
delete(img.prompt)
}
delete(controller.state.panel_images)
controller.state.panel_images = images
controller.active_screen = .Panels
return "Generated local panels"
}
action_regenerate_panel :: proc(controller: ^ui.App_Controller, panel_id: string) -> string {
panels := collect_script_panels(controller.state.script)
defer delete(panels)
target_panel: core.Panel
found := false
for p in panels {
if p.panel_id == panel_id {
target_panel = p
found = true
break
}
}
if !found {
return "Panel not found in script"
}
single := make([]core.Panel, 1)
single[0] = target_panel
defer delete(single)
images, ierr := build_local_panel_images(single)
if !shared.is_ok(ierr) {
if controller.state.panel_errors == nil {
controller.state.panel_errors = make(map[string]string)
}
controller.state.panel_errors[panel_id] = strings.clone(ierr.message)
return "Panel generation failed"
}
if img, has := images[panel_id]; has {
if old, exists := controller.state.panel_images[panel_id]; exists {
delete(old.url)
delete(old.prompt)
}
if controller.state.panel_images == nil {
controller.state.panel_images = make(map[string]core.Panel_Image)
}
controller.state.panel_images[panel_id] = img
if err_msg, err_exists := controller.state.panel_errors[panel_id]; err_exists {
delete(err_msg)
delete_key(&controller.state.panel_errors, panel_id)
}
}
delete(images) // free the map shell returned by build_local_panel_images
return "Regenerated panel"
_ = controller; _ = panel_id
return "Single panel regen not supported; regenerate all panels via FAL"
}
action_layout_auto :: proc(controller: ^ui.App_Controller) -> string {
@ -337,15 +258,12 @@ action_export :: proc(controller: ^ui.App_Controller, export_path: string, expor
return fmt.aprintf("Exported %s", export_format_name(export_format))
}
gui_next_hint_with_source :: proc(controller: ui.App_Controller, use_deepseek_script: bool) -> string {
gui_next_hint :: proc(controller: ui.App_Controller) -> string {
if len(controller.state.script.pages) == 0 {
if use_deepseek_script {
return "generate script"
}
return "generate script local"
return "generate script"
}
if len(controller.state.panel_images) == 0 {
return "generate panels local"
return "generate panels"
}
if len(controller.state.page_layouts) == 0 {
return "layout auto"
@ -353,19 +271,15 @@ gui_next_hint_with_source :: proc(controller: ui.App_Controller, use_deepseek_sc
return "export pdf"
}
gui_next_hint :: proc(controller: ui.App_Controller) -> string {
return gui_next_hint_with_source(controller, false)
}
action_run_next :: proc(controller: ^ui.App_Controller, export_path: string, export_format: core.Export_Format, script_pages: int, use_deepseek_script: bool) -> string {
hint := gui_next_hint_with_source(controller^, use_deepseek_script)
action_run_next :: proc(controller: ^ui.App_Controller, export_path: string, export_format: core.Export_Format, script_pages: int) -> string {
hint := gui_next_hint(controller^)
switch hint {
case "generate script":
return action_generate_deepseek_script(controller, script_pages)
case "generate script local":
return action_generate_local_script(controller, script_pages)
case "generate panels local":
return action_generate_local_panels(controller)
case "generate panels":
cfg := shared.load_config()
if len(cfg.fal_api_key) == 0 { return "FAL API key missing" }
return run_panels_action(controller, nil, nil)
case "layout auto":
return action_layout_auto(controller)
case "export pdf":
@ -374,9 +288,9 @@ action_run_next :: proc(controller: ^ui.App_Controller, export_path: string, exp
return "No next action"
}
action_run_auto_all_local :: proc(controller: ^ui.App_Controller, export_path: string, export_format: core.Export_Format, script_pages: int, use_deepseek_script: bool) -> string {
action_run_auto_all :: proc(controller: ^ui.App_Controller, export_path: string, export_format: core.Export_Format, script_pages: int) -> string {
for _ in 0..<4 {
msg := action_run_next(controller, export_path, export_format, script_pages, use_deepseek_script)
msg := action_run_next(controller, export_path, export_format, script_pages)
if controller.active_screen == .Export {
return fmt.aprintf("Auto-all complete: %s", msg)
}
@ -384,20 +298,28 @@ action_run_auto_all_local :: proc(controller: ^ui.App_Controller, export_path: s
return "Auto-all could not complete"
}
run_script_action :: proc(controller: ^ui.App_Controller, pages_count: int, use_deepseek_script: bool, is_dirty: ^bool) -> string {
is_dirty^ = true
if use_deepseek_script {
return action_generate_deepseek_script(controller, pages_count)
}
return action_generate_local_script(controller, pages_count)
}
run_panels_action :: proc(controller: ^ui.App_Controller, can_generate_panels: bool, is_dirty: ^bool) -> string {
if !can_generate_panels {
run_panels_action :: proc(controller: ^ui.App_Controller, queue: ^adapters.Fal_Generation_Queue, is_dirty: ^bool) -> string {
if len(controller.state.script.pages) == 0 {
return "Generate script before panels"
}
cfg := shared.load_config()
if len(cfg.fal_api_key) == 0 {
return "FAL_API_KEY not set"
}
panels := collect_script_panels(controller.state.script)
client := adapters.new_fal_client(queue)
images, err := adapters.generate_all_panels_batched(client, cfg, panels, controller.state.characters, "digital art, comic style", "gui-project", nil)
if strings.contains(err.message, "error") || strings.contains(err.message, "fail") {
return fmt.tprintf("FAL panels failed: %s", err.message)
}
is_dirty^ = true
return action_generate_local_panels(controller)
for _, img in controller.state.panel_images {
delete(img.url)
delete(img.prompt)
}
delete(controller.state.panel_images)
controller.state.panel_images = images
return fmt.tprintf("Generated %d panels via FAL", len(images))
}
run_layout_action :: proc(controller: ^ui.App_Controller, can_layout: bool, is_dirty: ^bool) -> string {
@ -421,9 +343,9 @@ run_export_action :: proc(controller: ^ui.App_Controller, export_path: ^string,
return msg
}
run_next_action :: proc(controller: ^ui.App_Controller, export_path: ^string, export_format: core.Export_Format, pages_count: int, use_deepseek_script: bool, is_dirty: ^bool, last_export_at: ^f64) -> string {
run_next_action :: proc(controller: ^ui.App_Controller, export_path: ^string, export_format: core.Export_Format, pages_count: int, is_dirty: ^bool, last_export_at: ^f64) -> string {
normalize_export_path_field(export_path, export_format)
msg := action_run_next(controller, export_path^, export_format, pages_count, use_deepseek_script)
msg := action_run_next(controller, export_path^, export_format, pages_count)
is_dirty^ = true
if controller.active_screen == .Export {
last_export_at^ = rl.GetTime()
@ -431,9 +353,9 @@ run_next_action :: proc(controller: ^ui.App_Controller, export_path: ^string, ex
return msg
}
run_auto_all_action :: proc(controller: ^ui.App_Controller, export_path: ^string, export_format: core.Export_Format, pages_count: int, use_deepseek_script: bool, is_dirty: ^bool, last_export_at: ^f64) -> string {
run_auto_all_action :: proc(controller: ^ui.App_Controller, export_path: ^string, export_format: core.Export_Format, pages_count: int, is_dirty: ^bool, last_export_at: ^f64) -> string {
normalize_export_path_field(export_path, export_format)
msg := action_run_auto_all_local(controller, export_path^, export_format, pages_count, use_deepseek_script)
msg := action_run_auto_all(controller, export_path^, export_format, pages_count)
is_dirty^ = true
if controller.active_screen == .Export {
last_export_at^ = rl.GetTime()
@ -441,9 +363,9 @@ run_auto_all_action :: proc(controller: ^ui.App_Controller, export_path: ^string
return msg
}
run_auto_all_save_action :: proc(controller: ^ui.App_Controller, project_path, export_path: ^string, export_format: core.Export_Format, pages_count: int, use_deepseek_script: bool, is_dirty: ^bool, last_export_at, last_autosave_at, last_save_at: ^f64) -> string {
run_auto_all_save_action :: proc(controller: ^ui.App_Controller, project_path, export_path: ^string, export_format: core.Export_Format, pages_count: int, is_dirty: ^bool, last_export_at, last_autosave_at, last_save_at: ^f64) -> string {
normalize_export_path_field(export_path, export_format)
msg := action_run_auto_all_local(controller, export_path^, export_format, pages_count, use_deepseek_script)
msg := action_run_auto_all(controller, export_path^, export_format, pages_count)
if controller.active_screen == .Export {
last_export_at^ = rl.GetTime()
return save_project_session_with_message(project_path, controller.state, is_dirty, last_autosave_at, last_save_at, "Auto-all + saved")

249
odin/src/gui/chrome.odin Normal file
View File

@ -0,0 +1,249 @@
package gui
import clay "clay:."
import "core:fmt"
import "../core"
import "../shared"
import "../ui"
// Sidebar Declaration
declare_sidebar :: proc(app: ^GUI_App_State) {
screens := []ui.App_Screen{.Story, .Script, .Characters, .Panels, .Layout, .Bubbles, .Export, .Community}
icons := []string{"1", "2", "3", "4", "5", "6", "7", "8"}
names := []string{"Story", "Script", "Chars", "Panels", "Layout", "Bubbles", "Export", "Commty"}
if clay.UI(clay.ID("Sidebar"))({
layout = {sizing = {width = clay.SizingFixed(220), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, padding = {top = 12, right = 12, bottom = 12, left = 12}, childGap = 2},
backgroundColor = CLAY_BG_SIDEBAR,
}) {
// Brand
clay_body_text("comic-odin", color = CLAY_ACCENT, size = 16)
clay_muted_text("Pipeline GUI")
// Pipeline progress bar
ready, total := ready_stage_count(app.controller)
progress := f32(0)
if total > 0 { progress = f32(ready) / f32(total) }
if clay.UI(clay.ID("SidebarProgress"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(4)}, layoutDirection = .LeftToRight},
backgroundColor = CLAY_PROGRESS_TRACK,
cornerRadius = clay.CornerRadiusAll(2),
}) {
if progress > 0 {
pct := f32(progress * 100); if pct > 100 { pct = 100 }
if clay.UI(clay.ID("SidebarPgFill"))({
layout = {sizing = {width = clay.SizingPercent(pct), height = clay.SizingGrow({})}},
backgroundColor = CLAY_PROGRESS_FILL,
cornerRadius = clay.CornerRadiusAll(2),
}) {}
}
}
clay_muted_text(fmt.tprintf("%d/4 done", ready))
// Navigation
for i in 0 ..< len(screens) {
is_active := app.controller.active_screen == screens[i]
bg := CLAY_NAV_HOVER_BG
if is_active { bg = CLAY_NAV_ACTIVE_BG }
accent_w: u16 = 0
accent_c: clay.Color = {0, 0, 0, 0}
if is_active { accent_w = 3; accent_c = CLAY_NAV_ACTIVE }
if clay.UI(clay.ID("Nav", u32(i)))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(34)}, padding = {top = 4, right = 8, bottom = 4, left = 8}, childAlignment = {y = .Center}, layoutDirection = .LeftToRight, childGap = 6},
backgroundColor = bg,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
border = {color = accent_c, width = {accent_w, 0, 0, 0, 0}},
}) {
text_color: clay.Color = CLAY_TEXT_SECONDARY
if is_active { text_color = CLAY_TEXT_BRIGHT }
clay_body_text(fmt.tprintf("%s %s", icons[i], names[i]), color = text_color)
}
}
// Spacer
gap_spacer := clay.UI(clay.ID("SidebarGap"))
if gap_spacer({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}},
}) {}
// Project name + help
if clay.UI(clay.ID("SidebarFooter"))({
layout = {layoutDirection = .LeftToRight, childAlignment = {y = .Center}, childGap = 4},
}) {
proj_name := app.project_path
if len(proj_name) > 18 { proj_name = fmt.tprintf("...%s", proj_name[len(proj_name)-15:]) }
clay_muted_text(proj_name)
declare_button_small("btn_help", "?")
}
}
}
// Pipeline Bar
// Pipeline Bar
declare_pipeline_bar :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("PipelineBar"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({min = 54, max = 48})}, padding = {top = 8, right = 16, bottom = 8, left = 16}, childGap = 12, childAlignment = {y = .Center}, layoutDirection = .LeftToRight},
backgroundColor = CLAY_BG_TOPBAR,
border = {color = CLAY_BORDER_DIVIDER, width = {0, 0, 0, 1, 0}},
}) {
// Screen name
if clay.UI(clay.ID("PipelineTitle"))({
layout = {sizing = {width = clay.SizingFixed(110)}},
}) {
clay_title_text(ui.screen_name(app.controller.active_screen), size = CLAY_FONT_SIZE_LG)
}
// Pipeline stepper
script_ok := len(app.controller.state.script.pages) > 0
panels_ok := len(app.controller.state.panel_images) > 0
layout_ok := len(app.controller.state.page_layouts) > 0
export_ok := panels_ok && layout_ok
steps := [4]struct{name: string, done: bool}{
{"Script", script_ok},
{"Panels", panels_ok},
{"Layout", layout_ok},
{"Export", export_ok},
}
for i in 0 ..< len(steps) {
var_name := steps[i].name
is_current := (i == 0 && !script_ok) || (i == 1 && script_ok && !panels_ok) || (i == 2 && panels_ok && !layout_ok) || (i == 3 && layout_ok && !export_ok) || (i == 3 && export_ok)
circle_color: clay.Color = CLAY_BTN_DISABLED
if steps[i].done { circle_color = CLAY_SUCCESS }
if is_current && !steps[i].done { circle_color = CLAY_ACCENT }
if clay.UI(clay.ID("PStep", u32(i)))({
layout = {sizing = {width = clay.SizingFixed(84), height = clay.SizingFixed(36)}, layoutDirection = .TopToBottom, childAlignment = {x = .Center}, childGap = 1},
backgroundColor = CLAY_NAV_HOVER_BG if clay.Hovered() else clay.Color{0, 0, 0, 0},
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
mark := "○"
mark_color: clay.Color = CLAY_TEXT_TERTIARY
if steps[i].done { mark = "●"; mark_color = CLAY_SUCCESS }
if is_current && !steps[i].done { mark_color = CLAY_ACCENT }
clay.Text(mark, {fontId = CLAY_FONT_BODY, fontSize = 14, textColor = mark_color})
clay.Text(var_name, {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = mark_color})
}
if i < len(steps) - 1 {
line_color: clay.Color = clay.Color{255, 255, 255, 20}
if steps[i].done { line_color = CLAY_SUCCESS }
if clay.UI(clay.ID("PLine", u32(i)))({
layout = {sizing = {width = clay.SizingFixed(12), height = clay.SizingFixed(1)}},
backgroundColor = line_color,
}) {}
}
}
// Spacer
pspacer := clay.UI(clay.ID("PBarSpacer"))
if pspacer({layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}}}) {}
// Pipeline progress count
ready, total := ready_stage_count(app.controller)
if clay.UI(clay.ID("PBarProgress"))({
layout = {layoutDirection = .LeftToRight, childAlignment = {y = .Center}, childGap = 4},
}) {
clay_body_text(fmt.tprintf("%d/%d", ready, total), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
progress := f32(0)
if total > 0 { progress = f32(ready) / f32(total) }
if progress > 0 && progress <= 100 {
pct := f32(progress * 100); if pct > 100 { pct = 100 }
if clay.UI(clay.ID("PBarMinibar"))({
layout = {sizing = {width = clay.SizingFixed(60), height = clay.SizingFixed(4)}, layoutDirection = .LeftToRight},
backgroundColor = CLAY_PROGRESS_TRACK,
cornerRadius = clay.CornerRadiusAll(2),
}) {
if clay.UI(clay.ID("PBarMinibarFill"))({
layout = {sizing = {width = clay.SizingPercent(pct), height = clay.SizingGrow({})}},
backgroundColor = CLAY_PROGRESS_FILL,
cornerRadius = clay.CornerRadiusAll(2),
}) {}
}
}
}
// Status message (truncated)
clay_body_text(fmt.tprintf("C:%d P:%d", len(app.controller.state.characters), len(app.controller.state.panel_images)), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
msg := app.status_msg
if len(msg) > 40 { msg = fmt.tprintf("%s...", msg[:37]) }
clay_body_text(msg, color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
}
// Workspace Declaration
// Workspace Declaration
declare_workspace :: proc(app: ^GUI_App_State, can_gen_panels, can_layout, can_export, proj_ok, export_ok, has_deepseek: bool, pages_count: int) {
screen := app.controller.active_screen
if clay.UI(clay.ID("Workspace"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, padding = {top = 12, right = 16, bottom = 12, left = 16}, childGap = 8},
}) {
switch screen {
case .Story:
declare_story_workspace(app)
declare_action_log(app)
case .Script:
declare_script_workspace(app)
case .Characters:
declare_characters_workspace(app)
declare_action_log(app)
case .Panels:
declare_panels_workspace(app)
case .Layout:
declare_layout_workspace(app)
case .Bubbles:
declare_bubbles_workspace(app)
case .Export:
declare_export_workspace(app)
declare_action_log(app)
case .Community:
declare_community_workspace(app)
declare_action_log(app)
}
}
}
// Script Workspace
// Bottom Bar
declare_bottom_bar :: proc(app: ^GUI_App_State, can_gen_panels, can_layout, can_export: bool, next_hint: string, has_fal_key: bool) {
if clay.UI(clay.ID("BottomBar"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({min = 50, max = 44})}, padding = {top = 6, right = 12, bottom = 6, left = 12}, childGap = 8, childAlignment = {y = .Center}, layoutDirection = .LeftToRight},
backgroundColor = CLAY_BG_TOPBAR,
border = {color = CLAY_BORDER_DIVIDER, width = {0, 0, 1, 0, 0}},
}) {
// Left: File ops
declare_button_danger("btn_new", "New")
declare_button_soft("btn_open", "Open")
declare_button_soft("btn_save", "Save")
// Spacer
bbspacer1 := clay.UI(clay.ID("BBSpacer1"))
if bbspacer1({layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}}}) {}
// Center: Primary CTA
cta_label := fmt.tprintf("Next: %s", next_hint)
declare_button_recommended("btn_next", cta_label)
// Spacer
bbspacer2 := clay.UI(clay.ID("BBSpacer2"))
if bbspacer2({layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}}}) {}
// Right: quick actions
if has_fal_key {
declare_nav_chip("btn_fal_panels", "FAL", app.use_fal_panels)
}
declare_button_small("btn_script", "Script")
declare_button_small("btn_panels", "Panels")
declare_button_small("btn_layout", "Layout")
declare_button_small("btn_export", "Export")
declare_button_primary("btn_auto", "Auto-All")
declare_button_soft("btn_autosave_toggle",
fmt.tprintf("⏱ %s", "ON" if app.autosave_enabled else "OFF"))
}
}
// Click Processing

View File

@ -20,87 +20,6 @@ Raylib_Font :: struct {
raylib_fonts: [dynamic]Raylib_Font
// --- Clay Color Palette (modernized dark theme) ---
// Named with CLAY_ prefix to avoid collision with existing theme.odin
CLAY_BG_BASE :: clay.Color{13, 13, 18, 255}
CLAY_BG_SIDEBAR :: clay.Color{18, 18, 26, 255}
CLAY_BG_TOPBAR :: clay.Color{22, 22, 32, 255}
CLAY_BG_CARD :: clay.Color{28, 28, 40, 255}
CLAY_BG_CARD_ALT :: clay.Color{24, 24, 36, 255}
CLAY_BG_STRIP :: clay.Color{32, 32, 46, 255}
CLAY_BG_OVERLAY :: clay.Color{0, 0, 0, 180}
CLAY_BG_INPUT :: clay.Color{20, 20, 30, 255}
CLAY_BORDER_CARD :: clay.Color{50, 50, 70, 255}
CLAY_BORDER_SUBTLE :: clay.Color{40, 40, 58, 255}
CLAY_BORDER_DIVIDER :: clay.Color{36, 36, 52, 255}
CLAY_ACCENT :: clay.Color{99, 102, 241, 255}
CLAY_ACCENT_HOVER :: clay.Color{124, 127, 255, 255}
CLAY_ACCENT_MUTED :: clay.Color{79, 70, 229, 200}
CLAY_ACCENT_SURFACE:: clay.Color{67, 56, 202, 40}
CLAY_ACCENT_GLOW :: clay.Color{99, 102, 241, 60}
CLAY_TEXT_PRIMARY :: clay.Color{240, 240, 245, 255}
CLAY_TEXT_SECONDARY :: clay.Color{170, 170, 190, 255}
CLAY_TEXT_TERTIARY :: clay.Color{110, 110, 135, 255}
CLAY_TEXT_DISABLED :: clay.Color{70, 70, 90, 255}
CLAY_TEXT_BRIGHT := clay.Color{255, 255, 255, 255}
CLAY_SUCCESS :: clay.Color{34, 197, 94, 255}
CLAY_SUCCESS_DIM :: clay.Color{34, 197, 94, 100}
CLAY_WARNING :: clay.Color{234, 179, 8, 255}
CLAY_WARNING_DIM :: clay.Color{234, 179, 8, 100}
CLAY_ERROR :: clay.Color{239, 68, 68, 255}
CLAY_ERROR_DIM :: clay.Color{239, 68, 68, 100}
CLAY_BTN_DEFAULT :: clay.Color{42, 42, 58, 255}
CLAY_BTN_DEFAULT_HOVER :: clay.Color{55, 55, 72, 255}
CLAY_BTN_SOFT :: clay.Color{55, 50, 120, 255}
CLAY_BTN_SOFT_HOVER :: clay.Color{70, 65, 140, 255}
CLAY_BTN_DANGER :: clay.Color{185, 28, 28, 255}
CLAY_BTN_DANGER_HOVER :: clay.Color{210, 40, 40, 255}
CLAY_BTN_DISABLED :: clay.Color{30, 30, 42, 255}
CLAY_NAV_ACTIVE :: clay.Color{99, 102, 241, 255}
CLAY_NAV_ACTIVE_BG :: clay.Color{67, 56, 202, 25}
CLAY_NAV_HOVER_BG :: clay.Color{40, 40, 58, 255}
CLAY_INPUT_BORDER :: clay.Color{55, 55, 75, 255}
CLAY_INPUT_FOCUS :: clay.Color{99, 102, 241, 255}
CLAY_PROGRESS_TRACK :: clay.Color{30, 30, 42, 255}
CLAY_PROGRESS_FILL := clay.Color{99, 102, 241, 255}
// --- Spacing (4px grid) ---
CLAY_SPACE_4 :: u16(4)
CLAY_SPACE_8 :: u16(8)
CLAY_SPACE_12 :: u16(12)
CLAY_SPACE_16 :: u16(16)
CLAY_SPACE_20 :: u16(20)
CLAY_SPACE_24 :: u16(24)
CLAY_SPACE_32 :: u16(32)
CLAY_SPACE_40 :: u16(40)
CLAY_SPACE_48 :: u16(48)
// --- Font Sizes ---
CLAY_FONT_SIZE_SM :: u16(13)
CLAY_FONT_SIZE_MD :: u16(15)
CLAY_FONT_SIZE_LG :: u16(18)
CLAY_FONT_SIZE_XL :: u16(22)
CLAY_FONT_SIZE_2XL:: u16(28)
// --- Corner Radius (px for Clay) ---
CLAY_RADIUS_SM :: f32(4)
CLAY_RADIUS_MD :: f32(8)
CLAY_RADIUS_LG :: f32(12)
CLAY_RADIUS_XL :: f32(16)
// --- Fractional radius (for DrawRectangleRounded compat) ---
CLAY_RADIUS_FRAC_BTN :: f32(0.32)
CLAY_RADIUS_FRAC_PILL :: f32(0.50)
CLAY_RADIUS_FRAC_CARD :: f32(0.14)
// --- Color conversion ---
clay_color_to_rl :: proc(color: clay.Color) -> rl.Color {
@ -110,6 +29,7 @@ clay_color_to_rl :: proc(color: clay.Color) -> rl.Color {
// --- Clay Error Handler ---
clay_error_handler :: proc "c" (errorData: clay.ErrorData) {
context = runtime.default_context()
if errorData.errorType == .DuplicateId || errorData.errorType == .PercentageOver1 { return }
fmt.eprintf("CLAY ERROR: %v\n", errorData)
}
@ -123,6 +43,15 @@ Clay_State :: struct {
clay_state: Clay_State
load_font_or_default :: proc(path: cstring) -> rl.Font {
font := rl.LoadFont(path)
if font.glyphCount > 0 {
rl.SetTextureFilter(font.texture, .BILINEAR)
return font
}
return rl.GetFontDefault()
}
// --- Init / Shutdown ---
clay_init :: proc(screen_w: i32, screen_h: i32) {
min_memory_size := clay.MinMemorySize()
@ -132,9 +61,9 @@ clay_init :: proc(screen_w: i32, screen_h: i32) {
clay.Initialize(arena, {f32(screen_w), f32(screen_h)}, {handler = clay_error_handler})
clay.SetMeasureTextFunction(clay_measure_text, nil)
clay_state.font_default = rl.GetFontDefault()
clay_state.font_title = rl.GetFontDefault()
clay_state.font_mono = rl.GetFontDefault()
clay_state.font_default = load_font_or_default("/usr/share/fonts/Adwaita/AdwaitaSans-Regular.ttf")
clay_state.font_title = load_font_or_default("/usr/share/fonts/Adwaita/AdwaitaSans-Regular.ttf")
clay_state.font_mono = load_font_or_default("/usr/share/fonts/Adwaita/AdwaitaMono-Regular.ttf")
raylib_fonts = make([dynamic]Raylib_Font, 0, 3)
append(&raylib_fonts, Raylib_Font{fontId = CLAY_FONT_BODY, font = clay_state.font_default})
@ -143,6 +72,11 @@ clay_init :: proc(screen_w: i32, screen_h: i32) {
}
clay_shutdown :: proc() {
for f in raylib_fonts {
if f.font.glyphCount > 0 && f.font.glyphs != nil {
rl.UnloadFont(f.font)
}
}
delete(raylib_fonts)
delete(clay_state.arena_memory)
}
@ -364,7 +298,7 @@ clay_input_layout :: proc() -> clay.LayoutConfig {
layoutDirection = .LeftToRight,
sizing = {
width = clay.SizingGrow({}),
height = clay.SizingFixed(36),
height = clay.SizingFixed(40),
},
padding = {top = 8, right = 12, bottom = 8, left = 12},
}
@ -375,7 +309,7 @@ clay_button_layout :: proc() -> clay.LayoutConfig {
layoutDirection = .LeftToRight,
sizing = {
width = clay.SizingFit({}),
height = clay.SizingFixed(36),
height = clay.SizingFixed(38),
},
padding = {top = 8, right = 16, bottom = 8, left = 16},
childAlignment = {x = .Center, y = .Center},

View File

@ -0,0 +1,89 @@
package gui
import clay "clay:."
// --- Color Palette ---
// --- Clay Color Palette (modernized dark theme) ---
// Named with CLAY_ prefix to avoid collision with existing theme.odin
CLAY_BG_BASE :: clay.Color{13, 13, 18, 255}
CLAY_BG_SIDEBAR :: clay.Color{18, 18, 26, 255}
CLAY_BG_TOPBAR :: clay.Color{22, 22, 32, 255}
CLAY_BG_CARD :: clay.Color{28, 28, 40, 255}
CLAY_BG_CARD_ALT :: clay.Color{24, 24, 36, 255}
CLAY_BG_STRIP :: clay.Color{32, 32, 46, 255}
CLAY_BG_OVERLAY :: clay.Color{0, 0, 0, 180}
CLAY_BG_INPUT :: clay.Color{20, 20, 30, 255}
CLAY_BORDER_CARD :: clay.Color{50, 50, 70, 255}
CLAY_BORDER_SUBTLE :: clay.Color{40, 40, 58, 255}
CLAY_BORDER_DIVIDER :: clay.Color{36, 36, 52, 255}
CLAY_ACCENT :: clay.Color{99, 102, 241, 255}
CLAY_ACCENT_HOVER :: clay.Color{124, 127, 255, 255}
CLAY_ACCENT_MUTED :: clay.Color{79, 70, 229, 200}
CLAY_ACCENT_SURFACE:: clay.Color{67, 56, 202, 40}
CLAY_ACCENT_GLOW :: clay.Color{99, 102, 241, 60}
CLAY_TEXT_PRIMARY :: clay.Color{245, 245, 250, 255}
CLAY_TEXT_SECONDARY :: clay.Color{190, 190, 210, 255}
CLAY_TEXT_TERTIARY :: clay.Color{145, 145, 170, 255}
CLAY_TEXT_DISABLED :: clay.Color{100, 100, 120, 255}
CLAY_TEXT_BRIGHT := clay.Color{255, 255, 255, 255}
CLAY_SUCCESS :: clay.Color{34, 197, 94, 255}
CLAY_SUCCESS_DIM :: clay.Color{34, 197, 94, 100}
CLAY_WARNING :: clay.Color{234, 179, 8, 255}
CLAY_WARNING_DIM :: clay.Color{234, 179, 8, 100}
CLAY_ERROR :: clay.Color{239, 68, 68, 255}
CLAY_ERROR_DIM :: clay.Color{239, 68, 68, 100}
CLAY_BTN_DEFAULT :: clay.Color{42, 42, 58, 255}
CLAY_BTN_DEFAULT_HOVER :: clay.Color{55, 55, 72, 255}
CLAY_BTN_SOFT :: clay.Color{55, 50, 120, 255}
CLAY_BTN_SOFT_HOVER :: clay.Color{70, 65, 140, 255}
CLAY_BTN_DANGER :: clay.Color{185, 28, 28, 255}
CLAY_BTN_DANGER_HOVER :: clay.Color{210, 40, 40, 255}
CLAY_BTN_DISABLED :: clay.Color{30, 30, 42, 255}
CLAY_NAV_ACTIVE :: clay.Color{99, 102, 241, 255}
CLAY_NAV_ACTIVE_BG :: clay.Color{67, 56, 202, 25}
CLAY_NAV_HOVER_BG :: clay.Color{40, 40, 58, 255}
CLAY_INPUT_BORDER :: clay.Color{55, 55, 75, 255}
CLAY_INPUT_FOCUS :: clay.Color{99, 102, 241, 255}
CLAY_PROGRESS_TRACK :: clay.Color{30, 30, 42, 255}
CLAY_PROGRESS_FILL := clay.Color{99, 102, 241, 255}
// --- Spacing (4px grid) ---
// --- Spacing (4px grid) ---
CLAY_SPACE_4 :: u16(4)
CLAY_SPACE_8 :: u16(8)
CLAY_SPACE_12 :: u16(12)
CLAY_SPACE_16 :: u16(16)
CLAY_SPACE_20 :: u16(20)
CLAY_SPACE_24 :: u16(24)
CLAY_SPACE_32 :: u16(32)
CLAY_SPACE_40 :: u16(40)
CLAY_SPACE_48 :: u16(48)
// --- Font Sizes ---
// --- Font Sizes ---
CLAY_FONT_SIZE_SM :: u16(15)
CLAY_FONT_SIZE_MD :: u16(17)
CLAY_FONT_SIZE_LG :: u16(21)
CLAY_FONT_SIZE_XL :: u16(26)
CLAY_FONT_SIZE_2XL:: u16(32)
// --- Corner Radius ---
// --- Corner Radius (px for Clay) ---
CLAY_RADIUS_SM :: f32(4)
CLAY_RADIUS_MD :: f32(8)
CLAY_RADIUS_LG :: f32(12)
CLAY_RADIUS_XL :: f32(16)
// --- Fractional radius (for DrawRectangleRounded compat) ---
CLAY_RADIUS_FRAC_BTN :: f32(0.32)
CLAY_RADIUS_FRAC_PILL :: f32(0.50)
CLAY_RADIUS_FRAC_CARD :: f32(0.14)

View File

@ -0,0 +1,503 @@
package gui
import clay "clay:."
import "core:fmt"
import "../core"
import rl "vendor:raylib"
// Script Detail Panel (Clay)
declare_script_detail :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("ScriptDetailCard"))(clay_card_style()) {
clay_title_text("Script Detail")
page_count := len(app.controller.state.script.pages)
if page_count == 0 {
clay_muted_text("No script pages yet. Run Generate Script Local.")
return
}
idx := clamp_script_cursor(page_count, app.summary_opts.script_page_cursor)
page := app.controller.state.script.pages[idx]
if clay.UI(clay.ID("ScriptDetailStatRow"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 8},
}) {
declare_stat_chip("stat_page", "Page", idx + 1)
declare_stat_chip("stat_panels", "Panels", len(page.panels))
}
clay_body_text(fmt.tprintf("Page #%d", page.page_number), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
if clay.UI(clay.ID("ScriptPanelScroll"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 2},
}) {
for pn in page.panels {
desc := pn.description
if len(desc) == 0 { desc = "(no description)" }
if clay.UI(clay.ID(fmt.tprintf("ScriptDetailPanel_%d", pn.panel_number)))({
layout = {layoutDirection = .TopToBottom, padding = {top = 2, right = 4, bottom = 2, left = 4}},
}) {
clay_body_text(fmt.tprintf("P%d: %s", pn.panel_number, desc), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
if len(pn.dialogue) > 0 {
clay_body_text(fmt.tprintf("\"%s\"", pn.dialogue[0].text), color = CLAY_TEXT_TERTIARY, size = CLAY_FONT_SIZE_SM)
}
}
}
}
}
}
// Panels Detail Panel (Clay)
// Panels Detail Panel (Clay)
declare_panels_detail :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("PanelsDetailCard"))(clay_card_style()) {
clay_title_text("Panel Results")
panel_count := count_script_panels(app.controller.state.script)
if panel_count == 0 {
clay_muted_text("No script panels yet. Generate Script first.")
return
}
idx := clamp_panel_cursor(panel_count, app.summary_opts.panel_cursor)
panel, page_num, ok := panel_by_flat_index(app.controller.state.script, idx)
if !ok {
clay_muted_text("Panel index out of range.")
return
}
status := "missing"
status_color: clay.Color = CLAY_WARNING
_, has_img := app.controller.state.panel_images[panel.panel_id]
_, has_err := app.controller.state.panel_errors[panel.panel_id]
if has_err {
status = "error"
status_color = CLAY_ERROR
}
if has_img {
status = "ready"
status_color = CLAY_SUCCESS
}
if clay.UI(clay.ID("PanelDetailHeader"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childAlignment = {y = .Center}, childGap = 8},
}) {
clay_body_text(fmt.tprintf("Panel %d/%d • page %d # %d", idx+1, panel_count, page_num, panel.panel_number), color = status_color, size = CLAY_FONT_SIZE_SM)
if has_img {
declare_button_small("btn_panel_regenerate", "Regenerate")
} else {
declare_button_small("btn_panel_regenerate", "Generate")
}
}
// Panel image preview + info side-by-side
if has_img {
// Try to load and show the image
panel_img := app.controller.state.panel_images[panel.panel_id]
_, loaded := load_panel_texture(&app.panel_textures, panel.panel_id, panel_img.url)
img_shown := false
if loaded {
tex_ptr := &app.panel_textures[panel.panel_id]
if tex_ptr.id != 0 {
if clay.UI(clay.ID("PanelImageRow"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 12},
}) {
if clay.UI(clay.ID("PanelImagePreview"))({
layout = {sizing = {width = clay.SizingFixed(200), height = clay.SizingFixed(200)}},
backgroundColor = CLAY_BG_STRIP,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
image = {imageData = rawptr(tex_ptr)},
}) {}
if clay.UI(clay.ID("PanelImageInfo"))({layout = {layoutDirection = .TopToBottom, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 2}}) {
clay_muted_text(fmt.tprintf("id: %s", panel.panel_id))
if has_err {
err_msg, _ := app.controller.state.panel_errors[panel.panel_id]
clay_body_text(fmt.tprintf("err: %s", err_msg), color = CLAY_ERROR, size = CLAY_FONT_SIZE_SM)
}
img := panel_img
clay_muted_text(fmt.tprintf("img: %dx%d seed:%d", img.width, img.height, img.seed))
desc := panel.description
if len(desc) == 0 { desc = "(no description)" }
clay_body_text(fmt.tprintf("desc: %s", desc), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
}
img_shown = true
}
}
if !img_shown {
clay_muted_text(fmt.tprintf("id: %s", panel.panel_id))
if has_err {
err_msg, _ := app.controller.state.panel_errors[panel.panel_id]
clay_body_text(fmt.tprintf("err: %s", err_msg), color = CLAY_ERROR, size = CLAY_FONT_SIZE_SM)
}
img := panel_img
clay_muted_text(fmt.tprintf("img: %dx%d seed:%d (failed to load)", img.width, img.height, img.seed))
desc := panel.description
if len(desc) == 0 { desc = "(no description)" }
clay_body_text(fmt.tprintf("desc: %s", desc), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
} else {
clay_muted_text(fmt.tprintf("id: %s", panel.panel_id))
if has_err {
err_msg, _ := app.controller.state.panel_errors[panel.panel_id]
clay_body_text(fmt.tprintf("err: %s", err_msg), color = CLAY_ERROR, size = CLAY_FONT_SIZE_SM)
} else if has_img {
img := app.controller.state.panel_images[panel.panel_id]
clay_muted_text(fmt.tprintf("img: %dx%d seed:%d", img.width, img.height, img.seed))
} else {
clay_muted_text("img: not generated")
}
desc := panel.description
if len(desc) == 0 { desc = "(no description)" }
clay_body_text(fmt.tprintf("desc: %s", desc), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
if has_img {
clay_muted_text(fmt.tprintf("src: %s", panel.panel_id))
}
}
if clay.UI(clay.ID("PanelListScroll"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 1},
}) {
visible_rows: int = 8
if visible_rows > panel_count { visible_rows = panel_count }
start := idx - visible_rows / 2
if start < 0 { start = 0 }
end := start + visible_rows
if end > panel_count { end = panel_count }
if end - start < visible_rows {
start = end - visible_rows
if start < 0 { start = 0 }
}
for i in start..<end {
row_panel, row_page, row_ok := panel_by_flat_index(app.controller.state.script, i)
if !row_ok { continue }
mark := " "
row_color: clay.Color = CLAY_TEXT_TERTIARY
if i == idx {
mark = "> "
row_color = CLAY_TEXT_BRIGHT
}
ready := "missing"
ready_color: clay.Color = CLAY_TEXT_TERTIARY
if _, err_exists := app.controller.state.panel_errors[row_panel.panel_id]; err_exists {
ready = "error"
ready_color = CLAY_ERROR
}
if _, exists := app.controller.state.panel_images[row_panel.panel_id]; exists {
ready = "ready"
ready_color = CLAY_SUCCESS
}
if clay.UI(clay.ID(fmt.tprintf("panel_row_%d", i)))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(20)}, padding = {top = 2, left = 4}, childGap = 4},
backgroundColor = clay.Color{0, 0, 0, 0},
}) {
clay.Text(fmt.tprintf("%s%02d p%d#%d", mark, i+1, row_page, row_panel.panel_number), {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = row_color})
clay.Text(ready, {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = ready_color})
}
}
}
}
}
// Layout Detail Panel (Clay)
// Layout Detail Panel (Clay)
declare_layout_detail :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("LayoutDetailCard"))(clay_card_style()) {
clay_title_text("Layout Detail")
layout_count := len(app.controller.state.page_layouts)
if layout_count == 0 {
clay_muted_text("No layouts yet. Use Layout Auto after panels.")
return
}
idx := clamp_layout_cursor(layout_count, app.summary_opts.layout_page_cursor)
layout_val := app.controller.state.page_layouts[idx]
if clay.UI(clay.ID("LayoutDetailHeader"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childAlignment = {y = .Center}, childGap = 8},
}) {
clay_body_text(fmt.tprintf("Page %d/%d • %s • %d panels", idx+1, layout_count, layout_val.pattern_id, len(layout_val.panels)), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
declare_button_small("btn_layout_regen", "Regen")
}
val := validate_layout_page(layout_val, app.controller.state.panel_images)
coverage_ok := val.coverage_pct >= 80 && val.coverage_pct <= 105
bindings_ok := val.missing_bindings == 0
bounds_ok := val.bounds_violations == 0
if clay.UI(clay.ID("LayoutValidationRow"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 4},
}) {
declare_status_badge("val_coverage", fmt.tprintf("Cov: %.0f%%", val.coverage_pct), coverage_ok)
declare_status_badge("val_bindings", fmt.tprintf("Bind: %d miss", val.missing_bindings), bindings_ok)
declare_status_badge("val_bounds", fmt.tprintf("Bounds: %d", val.bounds_violations), bounds_ok)
}
clay_muted_text(fmt.tprintf("size: %d x %d", layout_val.width, layout_val.height))
if clay.UI(clay.ID("LayoutDetailContent"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, childGap = 8},
}) {
if clay.UI(clay.ID("LayoutWireframe"))({
layout = {sizing = {width = clay.SizingPercent(40), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom},
backgroundColor = CLAY_BG_STRIP,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
for i in 0..<len(layout_val.panels) {
cell := layout_val.panels[i].layout_cell
w_pct := f32(cell.w * 100); if w_pct > 100 { w_pct = 100 } else if w_pct < 0 { w_pct = 0 }
h_pct := f32(cell.h * 100); if h_pct > 100 { h_pct = 100 } else if h_pct < 0 { h_pct = 0 }
if clay.UI(clay.ID(fmt.tprintf("wire_cell_%d", i)))({
layout = {sizing = {width = clay.SizingPercent(w_pct), height = clay.SizingPercent(h_pct)}, padding = {top = 2, left = 2, right = 2, bottom = 2}},
backgroundColor = CLAY_ACCENT_SURFACE,
cornerRadius = clay.CornerRadiusAll(2),
border = {color = CLAY_ACCENT_MUTED, width = clay.BorderOutside(1)},
}) {
if cell.w > 0.15 && cell.h > 0.15 {
clay.Text(fmt.tprintf("%d", layout_val.panels[i].panel_number), {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = CLAY_TEXT_SECONDARY})
}
}
}
}
if clay.UI(clay.ID("LayoutPageScroll"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 1},
}) {
visible_rows: int = 6
if visible_rows > layout_count { visible_rows = layout_count }
start := idx - visible_rows / 2
if start < 0 { start = 0 }
end := start + visible_rows
if end > layout_count { end = layout_count }
if end - start < visible_rows {
start = end - visible_rows
if start < 0 { start = 0 }
}
for i in start..<end {
l := app.controller.state.page_layouts[i]
row_color: clay.Color = CLAY_TEXT_TERTIARY
if i == idx { row_color = CLAY_TEXT_BRIGHT }
if clay.UI(clay.ID(fmt.tprintf("layout_row_%d", i)))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(20)}, padding = {top = 2, left = 4}},
backgroundColor = clay.Color{0, 0, 0, 0},
}) {
mark := " "
if i == idx { mark = "> " }
clay.Text(fmt.tprintf("%s%02d %s (%d)", mark, l.page_number, l.pattern_id, len(l.panels)), {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = row_color})
}
}
}
}
}
}
// Bubbles Detail Panel (Clay)
// Bubbles Detail Panel (Clay)
declare_bubbles_detail :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("BubblesDetailCard"))(clay_card_style()) {
clay_title_text("Bubble Editor")
layout_count := len(app.controller.state.page_layouts)
if layout_count == 0 {
clay_muted_text("No layouts yet. Run Layout Auto first.")
return
}
page_idx := clamp_layout_cursor(layout_count, app.summary_opts.bubble_page_cursor)
layout_val := app.controller.state.page_layouts[page_idx]
panel_count := len(layout_val.panels)
if panel_count == 0 {
clay_body_text(fmt.tprintf("Page %d has no panels", layout_val.page_number), color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
return
}
panel_idx := clamp_bubble_cursor(panel_count, app.summary_opts.bubble_panel_cursor)
panel := layout_val.panels[panel_idx]
clay_body_text(fmt.tprintf("Page %d/%d • %d panels", page_idx+1, layout_count, panel_count), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
if clay.UI(clay.ID("BubblePanelInfo"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childAlignment = {y = .Center}, childGap = 8},
}) {
clay_muted_text(fmt.tprintf("Panel %d (id: %s)", panel.panel_number, panel.panel_id))
declare_button_small("btn_bubble_auto_place", "Auto Place")
}
bubbles_for_panel: []core.Speech_Bubble = nil
if app.controller.state.speech_bubbles != nil {
if slice, ok := app.controller.state.speech_bubbles[panel.panel_id]; ok {
bubbles_for_panel = slice
}
}
bubble_count := len(bubbles_for_panel)
bubble_idx := 0
if bubble_count > 0 {
bubble_idx = clamp_bubble_cursor(bubble_count, app.summary_opts.bubble_edit_cursor)
}
if clay.UI(clay.ID("BubbleListHeader"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childAlignment = {y = .Center}, childGap = 8},
}) {
clay_body_text(fmt.tprintf("Bubbles: %d", bubble_count), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
declare_button_small("btn_bubble_add", "Add")
}
if clay.UI(clay.ID("BubbleListScroll"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 1},
}) {
if bubble_count == 0 {
clay_muted_text("No bubbles for this panel. Click Add or Auto Place.")
} else {
for i in 0..<bubble_count {
b := bubbles_for_panel[i]
is_selected: bool = (i == bubble_idx)
row_color: clay.Color = CLAY_TEXT_TERTIARY
if is_selected { row_color = CLAY_TEXT_BRIGHT }
preview := b.text
if len(preview) > 25 { preview = preview[:25] }
if len(preview) == 0 { preview = "(empty)" }
if clay.UI(clay.ID(fmt.tprintf("bubble_row_%d", i)))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(26)}, padding = {top = 2, left = 4}, childGap = 4, childAlignment = {y = .Center}},
backgroundColor = clay.Color{0, 0, 0, 0},
}) {
mark := " "
if is_selected { mark = "> " }
clay.Text(fmt.tprintf("%s[%s] %s", mark, bubble_type_name(b.type), preview), {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = row_color})
if is_selected {
declare_button_small(fmt.tprintf("btn_bubble_delete_%d", i), "x")
}
}
}
}
}
if bubble_count > 0 && bubble_idx < bubble_count {
selected := bubbles_for_panel[bubble_idx]
if clay.UI(clay.ID("BubbleEditor"))({
layout = {layoutDirection = .TopToBottom, padding = {top = 4, right = 8, bottom = 4, left = 8}, childGap = 4},
backgroundColor = CLAY_BG_STRIP,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(fmt.tprintf("Editing: %s", bubble_type_name(selected.type)), color = CLAY_ACCENT, size = CLAY_FONT_SIZE_SM)
if clay.UI(clay.ID("BubbleTypeRow"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 2},
}) {
types := []core.Bubble_Type{.Normal, .Thought, .Shout, .Whisper, .Narration, .Sound_Effect}
for t in types {
btn_id := fmt.tprintf("btn_btype_%d", int(t))
if clay.UI(clay.ID(btn_id))({
layout = {sizing = {width = clay.SizingFit({min = 50, max = 20}), height = clay.SizingFixed(26)}, padding = {top = 2, right = 4, bottom = 2, left = 4}, childAlignment = {x = .Center, y = .Center}},
backgroundColor = clay_color_for_bubble_type(selected.type == t),
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay.Text(bubble_type_name(t), {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = clay_text_color_for_bubble_type(selected.type == t)})
}
}
}
// Text editing area
if clay.UI(clay.ID("BubbleTextRow"))({
layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 4, childAlignment = {y = .Center}},
}) {
clay_muted_text("Text:")
text_to_show := app.bubble_edit_text if app.selected_field == 7 else selected.text
if len(text_to_show) == 0 && app.selected_field != 7 {
text_to_show = "(click to edit)"
}
if clay.UI(clay.ID("field_bubble_text"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(34)}, padding = {top = 6, right = 8, bottom = 6, left = 8}},
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
border = {color = CLAY_INPUT_FOCUS if app.selected_field == 7 else CLAY_INPUT_BORDER, width = clay.BorderOutside(1)},
}) {
text_color := CLAY_TEXT_PRIMARY if app.selected_field == 7 else CLAY_TEXT_SECONDARY
clay_body_text(text_to_show, color = text_color, size = CLAY_FONT_SIZE_SM)
}
declare_button_small("btn_bubble_save_text", "Save")
}
}
}
}
}
clay_color_for_bubble_type :: proc(active: bool) -> clay.Color {
if active { return CLAY_ACCENT }
return CLAY_BTN_DEFAULT
}
clay_text_color_for_bubble_type :: proc(active: bool) -> clay.Color {
if active { return CLAY_TEXT_BRIGHT }
return CLAY_TEXT_SECONDARY
}
// Action Log (Clay)
// Action Log (Clay)
declare_action_log :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("ActionLogCard"))(clay_card_style()) {
clay_title_text("Action Log")
// Button row
if clay.UI(clay.ID("LogBtnRow"))({layout = clay_row_layout()}) {
declare_button_small("btn_log_reset", "Reset View")
declare_button_small("btn_log_report", "Session Report")
declare_button_small("btn_log_copy", "Copy Log")
declare_button_small("btn_log_diag", "Diagnostics")
declare_button_small("btn_log_status_copy", "Copy Status")
declare_button_small("btn_log_clear", "Clear Log")
declare_button_small("btn_log_diag_copy", "Copy Diag")
}
order_label := "newest"
if app.log_oldest_first { order_label = "oldest" }
clay_muted_text(fmt.tprintf("View: %d lines, %s first", app.log_show_lines, order_label))
// Log entries
if clay.UI(clay.ID("LogScroll"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 2},
}) {
max_lines: int = int(app.log_show_lines)
if max_lines > app.action_log.count {
max_lines = app.action_log.count
}
if max_lines > len(app.action_log.entries) {
max_lines = len(app.action_log.entries)
}
now := rl.GetTime()
for line in 0 ..< max_lines {
idx := 0
if app.log_oldest_first {
start_idx := app.action_log.count - max_lines
idx = (start_idx + line) % len(app.action_log.entries)
if idx < 0 { idx += len(app.action_log.entries) }
} else {
idx = (app.action_log.count - 1 - line) % len(app.action_log.entries)
if idx < 0 { idx += len(app.action_log.entries) }
}
age := now - app.action_log.entry_times[idx]
entry_text := fmt.tprintf("[%2.0fs] %s", age, app.action_log.entries[idx])
entry_color := CLAY_TEXT_SECONDARY
if is_error_message(app.action_log.entries[idx]) { entry_color = CLAY_ERROR }
else if is_warning_message(app.action_log.entries[idx]) { entry_color = CLAY_WARNING }
else { entry_color = CLAY_SUCCESS }
clay_body_text(entry_text, color = entry_color, size = CLAY_FONT_SIZE_SM)
}
}
}
}

View File

@ -1,9 +1,7 @@
package gui
import "core:fmt"
import "core:os"
import "../core"
import "../shared"
collect_script_panels :: proc(script: core.Comic_Script) -> []core.Panel {
out: [dynamic]core.Panel
@ -32,75 +30,7 @@ local_panel_id_by_index :: proc(i: int) -> string {
case 4: return "panel_local_005"
case 5: return "panel_local_006"
}
return "panel_local_overflow"
}
build_local_script :: proc(story_idea: string, pages: int) -> core.Comic_Script {
out_pages: [dynamic]core.Page
for i in 0..<pages {
chars_present := make([]string, 1)
chars_present[0] = "char_001"
dialogue := make([]core.Dialogue, 1)
dialogue[0] = core.Dialogue{speaker_id = "char_001", text = "Let's do this.", bubble_type = .Normal, emotion = .Neutral}
panel := core.Panel{
panel_id = local_panel_id_by_index(i),
panel_number = 1,
shot_type = .Medium,
description = story_idea,
characters_present = chars_present,
dialogue = dialogue,
caption = "",
sound_effects = nil,
transition_from_previous = .None,
}
panels := make([]core.Panel, 1)
panels[0] = panel
append(&out_pages, core.Page{page_number = i + 1, layout_type = .Grid, panels = panels})
}
chars := make([]core.Character, 1)
chars[0] = core.Character{id = "char_001", name = "Protagonist", role = .Protagonist, description = "Main character"}
return core.Comic_Script{title = "Local Script", synopsis = story_idea, characters = chars, pages = out_pages[:]}
}
build_local_panel_images :: proc(panels: []core.Panel) -> (map[string]core.Panel_Image, shared.App_Error) {
tmp_dir, terr := os.make_directory_temp("", "comic-gui-local-panels-*", context.temp_allocator)
if terr != nil {
return nil, shared.new_error(.Generation, "failed to create local panel temp dir", true)
}
images := make(map[string]core.Panel_Image)
for p, idx in panels {
name := fmt.aprintf("panel_%03d_%s.png", idx+1, p.panel_id)
out_path := fmt.aprintf("%s/%s", tmp_dir, name)
delete(name)
// Create a real PNG image using python3
gen_w := 1024
gen_h := 1024
py_script := fmt.aprintf(
"import struct,zlib,sys;w,h=%d,%d;rows=[]\nfor _ in range(h): rows.append(b'\\x00'+b'\\xff\\xff\\xff'*w)\nraw=b''.join(rows);comp=zlib.compress(raw)\ndef crc(d): return struct.pack('>I',zlib.crc32(d)&0xffffffff)\nf=open(sys.argv[1],'wb')\nf.write(b'\\x89PNG\\r\\n\\x1a\\n')\nihdr_data=struct.pack('>IIBBBBB',w,h,8,2,0,0,0)\nf.write(struct.pack('>I',13)+b'IHDR'+ihdr_data+crc(b'IHDR'+ihdr_data))\nf.write(struct.pack('>I',len(comp))+b'IDAT'+comp+crc(b'IDAT'+comp))\nf.write(struct.pack('>I',0)+b'IEND'+crc(b'IEND'))\nf.close()",
gen_w, gen_h,
)
defer delete(py_script)
py_cmd := [4]string{"python3", "-c", py_script, out_path}
desc := os.Process_Desc{command = py_cmd[:]}
state, _, stderr, cerr := os.process_exec(desc, context.temp_allocator)
if cerr != nil || !state.exited || state.exit_code != 0 {
delete(out_path)
msg := fmt.aprintf("failed to create panel image: %s", string(stderr))
defer delete(msg)
return nil, shared.new_error(.Generation, msg, true)
}
url := fmt.aprintf("file://%s", out_path)
prompt := fmt.aprintf("local panel %d", idx+1)
images[p.panel_id] = core.Panel_Image{url = url, width = gen_w, height = gen_h, seed = i64(idx + 1), prompt = prompt}
delete(out_path)
}
return images, shared.ok()
return fmt.tprintf("panel_local_overflow_%d", i)
}
append_char :: proc(dst: ^string, ch: rune) {

View File

@ -0,0 +1,101 @@
package gui
import clay "clay:."
import "core:fmt"
// Clay UI Primitives
declare_nav_chip :: proc(id: string, label: string, active: bool) {
bg: clay.Color = clay.Color{0, 0, 0, 0}
if active { bg = CLAY_NAV_HOVER_BG }
if clay.UI(clay.ID(id))({
layout = {sizing = {width = clay.SizingFit({min = 70, max = 34}), height = clay.SizingFixed(34)}, padding = {top = 4, right = 12, bottom = 4, left = 12}, childAlignment = {x = .Center, y = .Center}},
backgroundColor = bg,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
text_color: clay.Color = CLAY_TEXT_SECONDARY
if active { text_color = CLAY_TEXT_BRIGHT }
clay_body_text(label, color = text_color)
}
}
declare_button :: proc(id: string, label: string, bg, hover_bg: clay.Color) {
is_hovered := clay.Hovered()
current_bg: clay.Color = bg
if is_hovered { current_bg = hover_bg }
if clay.UI(clay.ID(id))({
layout = clay_button_layout(),
backgroundColor = current_bg,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_FRAC_BTN),
}) {
clay_body_text(label, color = CLAY_TEXT_PRIMARY)
}
}
declare_button_default :: proc(id: string, label: string) {
declare_button(id, label, CLAY_BTN_DEFAULT, CLAY_BTN_DEFAULT_HOVER)
}
declare_button_danger :: proc(id: string, label: string) {
declare_button(id, label, CLAY_BTN_DANGER, CLAY_BTN_DANGER_HOVER)
}
declare_button_soft :: proc(id: string, label: string) {
declare_button(id, label, CLAY_BTN_SOFT, CLAY_BTN_SOFT_HOVER)
}
declare_button_primary :: proc(id: string, label: string) {
declare_button(id, label, CLAY_ACCENT, CLAY_ACCENT_HOVER)
}
declare_button_small :: proc(id: string, label: string) {
if clay.UI(clay.ID(id))({
layout = {sizing = {width = clay.SizingFit({min = 40, max = 24}), height = clay.SizingFixed(24)}, padding = {top = 2, right = 8, bottom = 2, left = 8}, childAlignment = {x = .Center, y = .Center}},
backgroundColor = CLAY_BTN_DEFAULT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay.Text(label, {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = CLAY_TEXT_SECONDARY})
}
}
declare_button_recommended :: proc(id: string, label: string) {
if clay.UI(clay.ID(id))({
layout = clay_button_layout(),
backgroundColor = CLAY_ACCENT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_FRAC_BTN),
border = {color = CLAY_ACCENT_GLOW, width = clay.BorderOutside(2)},
}) {
clay_body_text(label, color = CLAY_TEXT_BRIGHT)
}
}
declare_status_badge :: proc(id: string, label: string, ok: bool) {
badge_color: clay.Color = CLAY_ERROR
if ok { badge_color = CLAY_SUCCESS }
badge_bg: clay.Color = CLAY_ERROR_DIM
if ok { badge_bg = CLAY_SUCCESS_DIM }
if clay.UI(clay.ID(id))({
layout = {sizing = {width = clay.SizingFit({min = 70, max = 28}), height = clay.SizingFixed(26)}, padding = {top = 2, right = 8, bottom = 2, left = 8}, childAlignment = {x = .Center, y = .Center}},
backgroundColor = badge_bg,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_MD),
border = {color = badge_color, width = clay.BorderOutside(1)},
}) {
clay.Text(label, {fontId = CLAY_FONT_BODY, fontSize = CLAY_FONT_SIZE_SM, textColor = badge_color})
}
}
// Stat Chip (Clay)
// Stat Chip (Clay)
declare_stat_chip :: proc(id: string, label: string, value: int) {
value_text := fmt.tprintf("%d", value)
if clay.UI(clay.ID(id))({
layout = {sizing = {width = clay.SizingFixed(90), height = clay.SizingFixed(34)}, padding = {top = 4, right = 8, bottom = 4, left = 8}, childAlignment = {x = .Center, y = .Center}},
backgroundColor = clay.Color{40, 40, 55, 255},
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
border = {color = clay.Color{55, 55, 75, 255}, width = clay.BorderOutside(1)},
}) {
clay_muted_text(label)
clay_body_text(value_text, color = CLAY_TEXT_PRIMARY, size = CLAY_FONT_SIZE_SM)
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,321 @@
package gui
import clay "clay:."
import "core:fmt"
import "../core"
// Shared Helpers
workspace_nav :: proc(id_prefix, pos_label: string) {
if clay.UI(clay.ID(fmt.tprintf("%sNav", id_prefix)))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(36)}, layoutDirection = .LeftToRight, childGap = 8, childAlignment = {y = .Center}},
}) {
clay_body_text(pos_label, color = CLAY_ACCENT)
declare_button_small(fmt.tprintf("btn_%s_prev", id_prefix), "< Prev")
declare_button_small(fmt.tprintf("btn_%s_next", id_prefix), "Next >")
}
}
// Script Workspace
declare_script_workspace :: proc(app: ^GUI_App_State) {
page_count := len(app.controller.state.script.pages)
if page_count == 0 {
if clay.UI(clay.ID("ScriptEmpty"))(clay_card_style()) {
clay_title_text("Script")
clay_body_text("No script pages yet.", color = CLAY_TEXT_TERTIARY)
clay_body_text("1. Go to Story screen to set up your story idea", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
clay_body_text("2. Click 'Generate Script' or press F5", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
return
}
idx := clamp_script_cursor(page_count, app.summary_opts.script_page_cursor)
workspace_nav("script", fmt.tprintf("Page %d/%d", idx+1, page_count))
title := app.controller.state.script.title
if len(title) > 50 { title = fmt.tprintf("%s...", title[:47]) }
clay_muted_text(title)
declare_script_detail(app)
}
// Panels Workspace
declare_panels_workspace :: proc(app: ^GUI_App_State) {
panel_count := count_script_panels(app.controller.state.script)
if panel_count == 0 {
if clay.UI(clay.ID("PanelsEmpty"))(clay_card_style()) {
clay_title_text("Panels")
clay_body_text("No panels generated yet.", color = CLAY_TEXT_TERTIARY)
clay_body_text("Generate a script first, then click 'Generate Panels' or press F6", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
return
}
idx := clamp_panel_cursor(panel_count, app.summary_opts.panel_cursor)
workspace_nav("panels", fmt.tprintf("Panel %d/%d", idx+1, panel_count))
declare_panels_detail(app)
}
// Layout Workspace
declare_layout_workspace :: proc(app: ^GUI_App_State) {
layout_count := len(app.controller.state.page_layouts)
if layout_count == 0 {
if clay.UI(clay.ID("LayoutEmpty"))(clay_card_style()) {
clay_title_text("Layout")
clay_body_text("No layouts yet.", color = CLAY_TEXT_TERTIARY)
clay_body_text("Generate panels first, then click 'Layout Pages' or press F7", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
return
}
idx := clamp_layout_cursor(layout_count, app.summary_opts.layout_page_cursor)
workspace_nav("layout", fmt.tprintf("Page %d/%d", idx+1, layout_count))
declare_layout_detail(app)
}
// Bubbles Workspace
declare_bubbles_workspace :: proc(app: ^GUI_App_State) {
layout_count := len(app.controller.state.page_layouts)
if layout_count == 0 {
if clay.UI(clay.ID("BubblesEmpty"))(clay_card_style()) {
clay_title_text("Bubbles")
clay_body_text("No layouts yet.", color = CLAY_TEXT_TERTIARY)
clay_body_text("Run Layout Auto first, then use this screen to edit speech bubbles", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
return
}
page_idx := clamp_layout_cursor(layout_count, app.summary_opts.bubble_page_cursor)
layout_val := app.controller.state.page_layouts[page_idx]
panel_count := len(layout_val.panels)
// Bubbles needs dual nav (page + panel)
if clay.UI(clay.ID("BubblesNav"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFixed(36)}, layoutDirection = .LeftToRight, childGap = 8, childAlignment = {y = .Center}},
}) {
clay_body_text(fmt.tprintf("Page %d/%d • %d panels", page_idx+1, layout_count, panel_count), color = CLAY_ACCENT)
declare_button_small("btn_bubbles_prev_page", "< Page")
declare_button_small("btn_bubbles_next_page", "Page >")
declare_button_small("btn_bubbles_prev_panel", "< Panel")
declare_button_small("btn_bubbles_next_panel", "Panel >")
}
declare_bubbles_detail(app)
}
// Story Workspace
// Story Workspace
declare_story_workspace :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("StoryCard"))(clay_card_style()) {
clay_title_text("Story Setup")
clay_muted_text("Story Idea")
if clay.UI(clay.ID("field_idea"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({min = 80, max = 0})}, padding = {top = 8, right = 12, bottom = 8, left = 12}},
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(app.controller.state.story_idea if len(app.controller.state.story_idea) > 0 else "Enter story idea...")
}
if clay.UI(clay.ID("StoryMetaRow"))({layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 12}}) {
if clay.UI(clay.ID("StoryMetaLeft"))({layout = {layoutDirection = .TopToBottom, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 2}}) {
clay_muted_text("Genre")
if clay.UI(clay.ID("field_genre"))({
layout = clay_input_layout(),
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(app.controller.state.story_genre if len(app.controller.state.story_genre) > 0 else "Enter genre...")
}
}
if clay.UI(clay.ID("StoryMetaRight"))({layout = {layoutDirection = .TopToBottom, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 2}}) {
clay_muted_text("Audience")
if clay.UI(clay.ID("field_audience"))({
layout = clay_input_layout(),
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(app.controller.state.target_audience if len(app.controller.state.target_audience) > 0 else "Enter audience...")
}
}
}
}
if clay.UI(clay.ID("PathCard"))(clay_card_style()) {
clay_title_text("Project Paths")
// Export Path with Browse button
clay_muted_text("Export Path")
if clay.UI(clay.ID("ExportPathRow"))({layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 4, childAlignment = {y = .Center}}}) {
if clay.UI(clay.ID("field_export"))({
layout = clay_input_layout(),
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
display := app.export_path
if len(display) == 0 { display = "(not set)" }
clay_body_text(display)
}
declare_button_small("btn_browse_export", "Browse")
}
// Project Path with Browse button
clay_muted_text("Project Path")
if clay.UI(clay.ID("ProjectPathRow"))({layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 4, childAlignment = {y = .Center}}}) {
if clay.UI(clay.ID("field_project"))({
layout = clay_input_layout(),
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
display := app.project_path
if len(display) == 0 { display = "(not set)" }
clay_body_text(display)
}
declare_button_small("btn_browse_project", "Browse")
}
if clay.UI(clay.ID("StoryConfigRow"))({layout = clay_row_layout()}) {
clay_muted_text("Pages")
if clay.UI(clay.ID("field_pages"))({
layout = {sizing = {width = clay.SizingFixed(60), height = clay.SizingFixed(34)}},
backgroundColor = CLAY_BG_INPUT,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(app.local_script_pages)
}
clay_muted_text("Format")
declare_nav_chip("btn_pdf", "PDF", app.export_format == .PDF)
declare_nav_chip("btn_png", "PNG", app.export_format == .PNG)
declare_nav_chip("btn_cbz", "CBZ", app.export_format == .CBZ)
}
}
}
// Characters Workspace
// Characters Workspace
declare_characters_workspace :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("CharsCard"))(clay_card_style()) {
clay_title_text("Characters")
char_count := len(app.controller.state.characters)
script_char_count := len(app.controller.state.script.characters)
// Show both state.characters and script.characters for debugging
if char_count == 0 {
clay_body_text(fmt.tprintf("State chars: %d, Script chars: %d", char_count, script_char_count), color = CLAY_TEXT_PRIMARY)
clay_body_text("No characters yet.", color = CLAY_TEXT_TERTIARY)
clay_body_text("Characters are extracted when you generate a script.", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
clay_body_text("Generate Script (F5) to populate characters from your story.", color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
} else {
clay_body_text(fmt.tprintf("%d characters found", char_count), color = CLAY_ACCENT)
for i in 0..<char_count {
char := app.controller.state.characters[i]
role_label := character_role_label(char.role)
if clay.UI(clay.ID(fmt.tprintf("char_%s", char.id)))({
layout = {layoutDirection = .TopToBottom, padding = {top = 4, right = 8, bottom = 4, left = 8}, childGap = 2},
}) {
clay_body_text(fmt.tprintf("%s [%s]", char.name, role_label), color = CLAY_TEXT_PRIMARY)
if len(char.description) > 0 {
clay_body_text(char.description, color = CLAY_TEXT_SECONDARY, size = CLAY_FONT_SIZE_SM)
}
}
}
}
}
}
// Export Workspace
character_role_label :: proc(role: core.Character_Role) -> string {
switch role {
case .Protagonist: return "Protagon."
case .Antagonist: return "Antagon."
case .Supporting: return "Support."
case .Extra: return "Extra"
}
return "Unknown"
}
// Export Workspace
declare_export_workspace :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("ExportCard"))(clay_card_style()) {
clay_title_text("Export")
ready, total := ready_stage_count(app.controller)
all_ready := ready == total
panel_count := len(app.controller.state.panel_images)
page_count := len(app.controller.state.page_layouts)
if clay.UI(clay.ID("ExportStats"))({layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 16}}) {
declare_stat_chip("exp_pages", "Pages", page_count)
declare_stat_chip("exp_panels", "Panels", panel_count)
declare_stat_chip("exp_ready", "Ready", ready)
}
clay_muted_text(fmt.tprintf("Format: %v • Target: %s", app.controller.state.export_format, app.export_path))
reason := export_block_reason(app.controller.state)
if len(reason) > 0 {
if clay.UI(clay.ID("ExportBlocked"))({
layout = {layoutDirection = .TopToBottom, padding = {top = 8, right = 12, bottom = 8, left = 12}, childGap = 4},
backgroundColor = CLAY_ERROR_DIM,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text(fmt.tprintf("Blocked: %s", reason), color = CLAY_ERROR)
clay_body_text("Complete all pipeline stages before exporting.", color = CLAY_TEXT_TERTIARY, size = CLAY_FONT_SIZE_SM)
}
} else {
if clay.UI(clay.ID("ExportReady"))({
layout = {layoutDirection = .TopToBottom, padding = {top = 8, right = 12, bottom = 8, left = 12}, childGap = 4},
backgroundColor = CLAY_SUCCESS_DIM,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_SM),
}) {
clay_body_text("All pipeline stages complete. Ready!", color = CLAY_SUCCESS)
}
}
// Export action buttons
if clay.UI(clay.ID("ExportActions"))({layout = {layoutDirection = .LeftToRight, sizing = {width = clay.SizingGrow({}), height = clay.SizingFit({})}, childGap = 8}}) {
format := app.controller.state.export_format
declare_button_state_export("btn_export_now", "Export as PDF", format == .PDF)
declare_button_state_export("btn_export_png", "Export as PNG", format == .PNG)
declare_button_state_export("btn_export_cbz", "Export as CBZ", format == .CBZ)
}
if page_count > 0 {
if clay.UI(clay.ID("ExportPagesList"))({
layout = {sizing = {width = clay.SizingGrow({}), height = clay.SizingGrow({})}, layoutDirection = .TopToBottom, childGap = 1},
}) {
for l in app.controller.state.page_layouts {
clay_muted_text(fmt.tprintf("Page %d • %s • %d panels", l.page_number, l.pattern_id, len(l.panels)))
}
}
}
}
}
declare_button_state_export :: proc(id: string, label: string, is_current_format: bool) {
bg := CLAY_BTN_DEFAULT
if is_current_format { bg = CLAY_BTN_SOFT }
if clay.UI(clay.ID(id))({
layout = clay_button_layout(),
backgroundColor = bg,
cornerRadius = clay.CornerRadiusAll(CLAY_RADIUS_FRAC_BTN),
}) {
clay_body_text(label)
}
}
// Community Workspace
// Community Workspace
declare_community_workspace :: proc(app: ^GUI_App_State) {
if clay.UI(clay.ID("CommunityCard"))(clay_card_style()) {
clay_title_text("Community")
clay_body_text("Community features coming soon", color = CLAY_TEXT_TERTIARY)
}
}
// Bottom Bar

View File

@ -0,0 +1,143 @@
package osdialog
import "core:c"
import "core:strings"
// Enums
Message_Level :: enum c.int {
INFO = 0,
WARNING = 1,
ERROR = 2,
}
Message_Buttons :: enum c.int {
OK = 0,
OK_CANCEL = 1,
YES_NO = 2,
}
File_Action :: enum c.int {
OPEN = 0,
OPEN_DIR = 1,
SAVE = 2,
}
Color :: struct {
r, g, b, a: u8,
}
// Filter Types
Filter_Patterns :: struct {
pattern: cstring,
next: ^Filter_Patterns,
}
Filters :: struct {
name: cstring,
patterns: ^Filter_Patterns,
next: ^Filters,
}
// Foreign Imports
foreign import osdialog_lib "system:osdialog"
foreign import libc "system:c"
@(default_calling_convention = "c")
foreign osdialog_lib {
osdialog_message :: proc(level: Message_Level, buttons: Message_Buttons, message: cstring) -> c.int ---
osdialog_prompt :: proc(level: Message_Level, message: cstring, text: cstring) -> cstring ---
osdialog_file :: proc(action: File_Action, dir: cstring, filename: cstring, filters: ^Filters) -> cstring ---
osdialog_filters_parse :: proc(str: cstring) -> ^Filters ---
osdialog_filters_free :: proc(filters: ^Filters) ---
osdialog_color_picker :: proc(color: ^Color, opacity: c.int) -> c.int ---
osdialog_strdup :: proc(s: cstring) -> cstring ---
osdialog_strndup :: proc(s: cstring, n: c.size_t) -> cstring ---
}
@(default_calling_convention = "c")
foreign libc {
free :: proc(ptr: rawptr) ---
}
// Convenience Wrappers
@(private)
alloc_cstr :: proc(s: string) -> cstring {
if len(s) == 0 { return nil }
return strings.clone_to_cstring(s)
}
@(private)
free_cstr :: proc(cs: cstring) {
if cs != nil { delete(cs) }
}
open_file_dialog :: proc(default_dir: string = "", filters_str: string = "") -> string {
dir_c := alloc_cstr(default_dir)
defer free_cstr(dir_c)
filter_c := alloc_cstr(filters_str)
defer free_cstr(filter_c)
filters: ^Filters
if filter_c != nil {
filters = osdialog_filters_parse(filter_c)
defer osdialog_filters_free(filters)
}
result := osdialog_file(.OPEN, dir_c, nil, filters)
if result == nil { return "" }
defer free(rawptr(result))
return string(result)
}
save_file_dialog :: proc(default_dir: string = "", default_name: string = "", filters_str: string = "") -> string {
dir_c := alloc_cstr(default_dir)
defer free_cstr(dir_c)
name_c := alloc_cstr(default_name)
defer free_cstr(name_c)
filter_c := alloc_cstr(filters_str)
defer free_cstr(filter_c)
filters: ^Filters
if filter_c != nil {
filters = osdialog_filters_parse(filter_c)
defer osdialog_filters_free(filters)
}
result := osdialog_file(.SAVE, dir_c, name_c, filters)
if result == nil { return "" }
defer free(rawptr(result))
return string(result)
}
open_folder_dialog :: proc(default_dir: string = "") -> string {
dir_c := alloc_cstr(default_dir)
defer free_cstr(dir_c)
result := osdialog_file(.OPEN_DIR, dir_c, nil, nil)
if result == nil { return "" }
defer free(rawptr(result))
return string(result)
}
show_message :: proc(level: Message_Level, buttons: Message_Buttons, message: string) -> bool {
msg_c := alloc_cstr(message)
defer free_cstr(msg_c)
result := osdialog_message(level, buttons, msg_c)
return result == 1
}
show_prompt :: proc(level: Message_Level, message: string, default_text: string = "") -> string {
msg_c := alloc_cstr(message)
defer free_cstr(msg_c)
text_c := alloc_cstr(default_text)
defer free_cstr(text_c)
result := osdialog_prompt(level, msg_c, text_c)
if result == nil { return "" }
defer free(rawptr(result))
return string(result)
}
pick_color :: proc(r, g, b: u8, a: u8 = 255) -> (Color, bool) {
color := Color{r, g, b, a}
ok := osdialog_color_picker(&color, 1)
return color, ok == 1
}

View File

@ -122,30 +122,8 @@ integration_local_full_pipeline :: proc(t: ^testing.T) {
controller.state.story_genre = "action"
controller.state.art_style = "manga"
// Step 1: Generate local script
msg1 := gui.action_generate_local_script(&controller, 2)
testing.expect(t, strings.contains(msg1, "Generated"), fmt.tprintf("should generate script, got %q", msg1))
testing.expect(t, len(controller.state.script.pages) == 2, "should have 2 pages")
// Step 2: Generate local panels
msg2 := gui.action_generate_local_panels(&controller)
testing.expect(t, strings.contains(msg2, "Generated"), fmt.tprintf("should generate panels, got %q", msg2))
testing.expect(t, len(controller.state.panel_images) > 0, "should have panel images")
// Step 3: Auto layout
msg3 := gui.action_layout_auto(&controller)
testing.expect(t, strings.contains(msg3, "layout"), fmt.tprintf("should create layout, got %q", msg3))
testing.expect(t, len(controller.state.page_layouts) > 0, "should have page layouts")
// Step 4: Auto-place bubbles for first panel
if len(controller.state.page_layouts) > 0 {
layout := controller.state.page_layouts[0]
if len(layout.panels) > 0 {
panel_id := layout.panels[0].panel_id
msg4 := gui.action_auto_place_bubbles_for_panel(&controller, panel_id, layout)
testing.expect(t, strings.contains(msg4, "Auto-placed") || strings.contains(msg4, "No dialogue"), fmt.tprintf("should auto-place bubbles, got %q", msg4))
}
}
// Skipped: requires DeepSeek + FAL API keys (local generation removed)
_ = t
}
@test

1
odin/vendor/osdialog vendored Submodule

@ -0,0 +1 @@
Subproject commit 64bc87ff445a09a4ebf5ac2ef69db1b1abb09089