{# Plugin Configuration Partial - Server-side rendered form #} {# This template is loaded via HTMX when a plugin tab is clicked #} {# ===== MACROS FOR FORM FIELD GENERATION ===== #} {# Render a single form field based on schema type #} {% macro render_field(key, prop, value, prefix='', plugin_id='') %} {% set full_key = (prefix ~ '.' ~ key) if prefix else key %} {% set field_id = (plugin_id ~ '-' ~ full_key)|replace('.', '-')|replace('_', '-') %} {% set label = prop.title if prop.title else key|replace('_', ' ')|title %} {% set description = prop.description if prop.description else '' %} {% set field_type = prop.type if prop.type is string else (prop.type[0] if prop.type is iterable else 'string') %} {# Handle nested objects - check for widget first #} {% if field_type == 'object' %} {% set obj_widget = prop.get('x-widget') or prop.get('x_widget') %} {% if obj_widget == 'schedule-picker' %} {# Schedule picker widget - renders enable/mode/times UI #} {% set obj_value = value if value is not none else {} %}
{{ description }}
{% endif %}{{ description }}
{% endif %}{{ description }}
{% endif %} {# Boolean - check for widget first #} {% if field_type == 'boolean' %} {% set bool_widget = prop.get('x-widget') or prop.get('x_widget') %} {% if bool_widget == 'toggle-switch' %} {# Render toggle-switch widget #} {% else %} {# Default checkbox - value="true" ensures checked sends "true" not "on" #} {% endif %} {# Enum dropdown #} {% elif prop.enum %} {# Number input - check for widget first #} {% elif field_type in ['number', 'integer'] %} {% set num_widget = prop.get('x-widget') or prop.get('x_widget') %} {% if num_widget in ['slider', 'number-input'] %} {# Render slider or number-input widget #} {% else %} {# Default number input #} {% endif %} {# Array - check for file upload widget first (to avoid breaking static-image plugin), then checkbox-group, then array of objects #} {% elif field_type == 'array' %} {% set x_widget = prop.get('x-widget') or prop.get('x_widget') %} {% if x_widget == 'file-upload' %} {# File upload widget for arrays #} {% set upload_config = prop.get('x-upload-config') or {} %} {% set max_files = upload_config.get('max_files', 10) %} {% set allowed_types = upload_config.get('allowed_types', ['image/png', 'image/jpeg', 'image/bmp', 'image/gif']) %} {% set max_size_mb = upload_config.get('max_size_mb', 5) %} {% set plugin_id_from_config = upload_config.get('plugin_id', plugin_id) %} {% set array_value = value if value is not none and value is iterable and value is not string else (prop.default if prop.default is defined and prop.default is iterable and prop.default is not string else []) %}Drag and drop images here or click to browse
Max {{ max_files }} files, {{ max_size_mb }}MB each (PNG, JPG, GIF, BMP)
Remember to save configuration after upload
{{ img.get('original_filename') or img.get('filename', 'Image') }}
{% if img.get('size') %}{{ (img.get('size') / 1024)|round }} KB{% endif %} {% if img.get('uploaded_at') %} • {{ img.get('uploaded_at') }}{% endif %}
{% if has_schedule %}Scheduled
{% endif %}Custom feeds widget requires 'name' and 'url' properties in items schema.
{% else %} {% set max_items = prop.get('maxItems', 50) %} {% set array_value = value if value is not none and value is iterable and value is not string else (prop.default if prop.default is defined and prop.default is iterable and prop.default is not string else []) %}| Name | URL | Logo | Enabled | Actions |
|---|---|---|---|---|
|
{% set logo_value = item.get('logo') or {} %}
{% set logo_path = logo_value.get('path', '') %}
{% if logo_path %}
|
| {{ col_title }} | {% endfor %}Actions |
|---|---|
| {% if col_type == 'boolean' %} {% elif col_type == 'integer' or col_type == 'number' %} {% else %} {% endif %} | {% endfor %}
Separate multiple values with commas
{% endif %} {% endif %} {# String/default field - check for widgets #} {% else %} {% set str_widget = prop.get('x-widget') or prop.get('x_widget') %} {% set str_value = value if value is not none else (prop.default if prop.default is defined else '') %} {% if str_widget in ['text-input', 'textarea', 'select-dropdown', 'toggle-switch', 'radio-group', 'date-picker', 'slider', 'color-picker', 'email-input', 'url-input', 'password-input', 'font-selector'] %} {# Render widget container #} {% else %} {# Default text input #} {% endif %} {% endif %}{{ plugin.description or 'Plugin configuration' }}