init
This commit is contained in:
130
user/plugins/admin/themes/grav/templates/forms/field.html.twig
Normal file
130
user/plugins/admin/themes/grav/templates/forms/field.html.twig
Normal file
@@ -0,0 +1,130 @@
|
||||
{% if not field.validate.ignore %}
|
||||
|
||||
{% if not blueprints or (blueprints.schema.type(field.type)['input@'] ?? true) is same as(true) %}
|
||||
{% set default = field.default %}
|
||||
{% set toggleable = field.toggleable ?? false %}
|
||||
{% if toggleable %}
|
||||
{% set originalValue = originalValue ?? value %}
|
||||
{% set toggleableChecked = originalValue is not null %}
|
||||
{% endif %}
|
||||
|
||||
{% set has_value = value is not null %}
|
||||
{% if not has_value %}
|
||||
{% set value = default %}
|
||||
{% endif %}
|
||||
|
||||
{% if (field.yaml or field.validate.type == 'yaml') and value is iterable %}
|
||||
{% set value = value|toYaml %}
|
||||
{% endif %}
|
||||
{% else %}
|
||||
{% set toggleable = false %}
|
||||
{% endif %}
|
||||
{% set vertical = field.style == 'vertical' %}
|
||||
{% set field_name = (scope ~ field.name)|fieldName %}
|
||||
{% set show_label = field.label is not same as(false) and field.display_label is not same as(false) %}
|
||||
|
||||
{# DEPRECATED: Needed by old form fields; remove when backwards compatibility breaks are allowed #}
|
||||
{% set isDisabledToggleable = toggleable and not toggleableChecked %}
|
||||
|
||||
{% block field %}
|
||||
<div class="form-field grid{% if vertical %} vertical{% endif %}{% if toggleable %} form-field-toggleable{% endif %} {{ field.outerclasses }} {{ field.field_classes }}">
|
||||
{% block contents %}
|
||||
{% if show_label %}
|
||||
<div class="form-label{% if not vertical %} block size-1-3{% endif %}">
|
||||
{% if toggleable %}
|
||||
<span class="checkboxes toggleable" data-grav-field="toggleable" data-grav-field-name="{{ field_name }}">
|
||||
<input type="checkbox"
|
||||
id="toggleable_{{ field.name }}"
|
||||
{% if toggleableChecked %}value="1"{% endif %}
|
||||
name="toggleable_{{ field_name }}"
|
||||
{% if toggleableChecked %}checked="checked"{% endif %}
|
||||
>
|
||||
<label for="toggleable_{{ field.name }}"></label>
|
||||
</span>
|
||||
{% endif %}
|
||||
<label{{ (toggleable ? ' class="toggleable" for="toggleable_' ~ field.name ~ '"')|raw }}>
|
||||
{% block label %}
|
||||
{% if field.help %}
|
||||
{% if field.markdown %}
|
||||
<span class="hint--bottom" data-hint="{{ field.help|t|markdown(false) }}">{{ field.label|t|markdown(false)|raw }}</span>
|
||||
{% else %}
|
||||
<span class="hint--bottom" data-hint="{{ field.help|t }}">{{ field.label|t|raw }} <i class="hint-icon fa fa-question-circle" aria-hidden="true"></i></span>
|
||||
{% endif %}
|
||||
{% else %}
|
||||
{% if field.markdown %}
|
||||
{{ field.label|t|markdown(false)|raw }}
|
||||
{% else %}
|
||||
{{ field.label|t|raw }}
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
{{ field.validate.required in ['on', 'true', 1] ? '<span class="required">*</span>' }}
|
||||
{% endblock %}
|
||||
</label>
|
||||
{% if field.sublabel %}
|
||||
<div class="form-sublabel">
|
||||
{% if field.markdown %}
|
||||
{{ field.sublabel|t|markdown(false)|raw }}
|
||||
{% else %}
|
||||
{{ field.sublabel|t|raw }}
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
<div class="form-data{% if not vertical %} block size-2-3{% endif %}"
|
||||
{% block global_attributes %}
|
||||
data-grav-field="{{ field.type }}"
|
||||
data-grav-disabled="{{ toggleableChecked }}"
|
||||
data-grav-default="{{ field.default|json_encode|e('html_attr') }}"
|
||||
{% endblock %}
|
||||
>
|
||||
{% block group %}
|
||||
{% block input %}
|
||||
<div class="form-input-wrapper {{ field.size }} {{ field.wrapper_classes }}">
|
||||
{% block prepend %}{% endblock prepend %}
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<input
|
||||
{# required attribute structures #}
|
||||
name="{{ field_name }}"
|
||||
value="{{ input_value }}"
|
||||
{% if field.key %}
|
||||
data-key-observe="{{ (scope ~ field_name)|fieldName }}"
|
||||
{% endif %}
|
||||
{# input attribute structures #}
|
||||
{% block input_attributes %}
|
||||
{% if field.classes is defined %}class="{{ field.classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder|t }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.autocomplete is defined %}autocomplete="{{ field.autocomplete }}"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.validate.pattern %}pattern="{{ field.validate.pattern }}"{% endif %}
|
||||
{% if field.validate.message %}title="{{ field.validate.message|t }}"
|
||||
{% elseif field.title is defined %}title="{{ field.title|t }}" {% endif %}
|
||||
{% endblock %}
|
||||
/>
|
||||
{% block append %}{% endblock append %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
{% endblock %}
|
||||
{% if field.description %}
|
||||
<div class="form-extra-wrapper {{ field.wrapper_classes }}">
|
||||
<span class="form-description">
|
||||
{% if field.markdown %}
|
||||
{{ field.description|t|markdown(false)|raw }}
|
||||
{% else %}
|
||||
{{ field.description|t|raw }}
|
||||
{% endif %}
|
||||
</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
{% endif %}
|
||||
@@ -0,0 +1,134 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
{% set permissions = grav.permissions %}
|
||||
{% set classes = { '': 'status-unchecked', 1: 'status-checked', 0: 'status-indeterminate' } %}
|
||||
{% set states = { '': 0, 1: 1, 0: 2 } %}
|
||||
|
||||
{% if field.data_type == 'access' %}
|
||||
{% set groupsList = [] %}
|
||||
{% for action in permissions %}
|
||||
{% if (action.visible ?? true) %}
|
||||
{% set groupsList = groupsList|merge([{ label: action.label|t, value: action.name }]) %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
{% set optionsList = [] %}
|
||||
{% for action in permissions.instances %}
|
||||
{% if (action.visible ?? true) %}
|
||||
{% set label = (action.params.letter ? action.parent.label|t ~ ' > ') ~ action.label|t %}
|
||||
{% set optionsList = optionsList|merge([{ text: label ~ ' (' ~ action.name ~ ')', value: action.name, optgroup: action.scope }]) %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{% elseif field.data_type == 'permissions' %}
|
||||
{% set groups = grav.flex.directory('user-groups') %}
|
||||
{% set groupsList = [] %}
|
||||
{% set crudp = {
|
||||
create: { letter: 'C', title: 'Create', value: '' },
|
||||
read: { letter: 'R', title: 'Read', value: '' },
|
||||
update: { letter: 'U', title: 'Update', value: '' },
|
||||
delete: { letter: 'D', title: 'Delete', value: '' }
|
||||
} %}
|
||||
|
||||
{% if object.hasFlexFeature('page') %}
|
||||
{% set optionsList = [{text: 'Page Authors (Special)', value: 'authors'}, {text: 'Default ACL (Special)', value: 'defaults'}] %}
|
||||
{% else %}
|
||||
{% set optionsList = [] %}
|
||||
{% endif %}
|
||||
{% for group in groups.index %}
|
||||
{% set optionsList = optionsList|merge([{ text: group.readableName ?? group.groupname, value: group.groupname }]) %}
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
|
||||
<template data-id="acl_picker-{{ field.name }}">
|
||||
<div class="permissions-item" data-field-type="{{ field.data_type }}">
|
||||
<a href="#" class="remove-item"><i class="fa fa-trash"></i></a>
|
||||
<select data-grav-selectize="{{ { options: optionsList, optgroups: groupsList }|json_encode }}"></select>
|
||||
|
||||
{% if field.data_type == 'access' %}
|
||||
<div class="switch-toggle switch-grav medium switch-3">
|
||||
<input type="radio" value="1" id="{{ field.name ~ '_' }}" name="{{ (scope ~ field.name)|fieldName ~ '[]' }}" class="label1" checked>
|
||||
|
||||
<label for="{{ field.name ~ '_' }}">{{ 'PLUGIN_ADMIN.ALLOWED'|t }}</label>
|
||||
|
||||
<input type="radio" value="0" id="{{ field.name ~ '_' }}" name="{{ (scope ~ field.name)|fieldName ~ '[]' }}" class="label0">
|
||||
|
||||
<label for="{{ field.name ~ '_' }}">{{ 'PLUGIN_ADMIN.DENIED'|t }}</label>
|
||||
|
||||
</div>
|
||||
{% elseif field.data_type == 'permissions' %}
|
||||
{% set data_field_name = (scope ~ '_json.' ~ field.name)|fieldName %}
|
||||
<div class="crudp-container" data-field-name="{{ data_field_name ~ '[]' }}">
|
||||
{% for key, button in crudp %}
|
||||
<div>
|
||||
<span class="checkboxes indeterminate toggleable status-unchecked hint--top"
|
||||
data-_check-status="0"
|
||||
data-hint="{{ button.title }}">
|
||||
<input type="checkbox"
|
||||
id="{{ field.name ~ '_' ~ key ~ '_' }}"
|
||||
data-crudp-key="{{ key }}"
|
||||
{# name="{{ (scope ~ field.name)|fieldName ~ '[][' ~ key ~ ']' }}"#}
|
||||
indeterminte="false" value="">
|
||||
<label for="{{ field.name ~ '_' ~ key ~ '_' }}">{{ button.letter }}</label>
|
||||
</span>
|
||||
</div>
|
||||
{% endfor %}
|
||||
|
||||
<input type="hidden" name="{{ data_field_name ~ '[][' ~ key ~ ']' }}">
|
||||
</div>
|
||||
{% endif %}
|
||||
<button class="button add-item"><i class="fa fa-plus"></i></button>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
|
||||
<div class="permissions-container" data-acl_picker_id="{{ field.name }}" data-acl_picker="{{ { options: optionsList, optgroups: groupsList }|json_encode }}">
|
||||
<div class="permissions-item empty-list {{ value|length ? 'hidden' }}">
|
||||
<a href="#" class="button add-item"><i class="fa fa-plus"></i></a>
|
||||
</div>
|
||||
|
||||
{% for key, access in value %}
|
||||
<div class="permissions-item" data-field-type="{{ field.data_type }}">
|
||||
<a href="#" class="remove-item"><i class="fa fa-trash"></i></a>
|
||||
<select data-grav-selectize="{{ { options: optionsList, optgroups: groupsList }|json_encode }}">
|
||||
<option value="{{ key }}" selected>{{ key }}</option>
|
||||
</select>
|
||||
{% if field.data_type == 'access' %}
|
||||
<div class="switch-toggle switch-grav medium switch-3">
|
||||
{% set rnd = random(100) %}
|
||||
<input type="radio" value="1" id="{{ field.name ~ '_' ~ rnd }}" name="{{ (scope ~ field.name)|fieldName ~ '[' ~ key ~ ']' }}" class="label1" {{ access ? 'checked' }}>
|
||||
|
||||
<label for="{{ field.name ~ '_' ~ rnd }}">{{ 'PLUGIN_ADMIN.ALLOWED'|t }}</label>
|
||||
|
||||
{% set rnd = random(100) %}
|
||||
<input type="radio" value="0" id="{{ field.name ~ '_' ~ rnd }}" name="{{ (scope ~ field.name)|fieldName ~ '[' ~ key ~ ']' }}" class="label0" {{ not access ? 'checked' }}>
|
||||
|
||||
<label for="{{ field.name ~ '_' ~ rnd }}">{{ 'PLUGIN_ADMIN.DENIED'|t }}</label>
|
||||
|
||||
</div>
|
||||
{% elseif field.data_type == 'permissions' %}
|
||||
{% set data_field_name = (scope ~ '_json.' ~ field.name)|fieldName %}
|
||||
<div class="crudp-container" data-field-name="{{ data_field_name ~ '[]' }}">
|
||||
{% for crudp_key, button in crudp %}
|
||||
<div>
|
||||
{% set crudp_value = value[key][crudp_key] %}
|
||||
<span class="checkboxes indeterminate toggleable {{ classes[crudp_value] }} hint--top"
|
||||
data-_check-status="{{ states[crudp_value] }}"
|
||||
data-hint="{{ button.title }}">
|
||||
<input type="checkbox"
|
||||
id="{{ field.name ~ '_' ~ crudp_key ~ '_' }}"
|
||||
data-crudp-key="{{ crudp_key }}"
|
||||
{#name="{{ (scope ~ field.name)|fieldName ~ '[' ~ key ~ '][' ~ crudp_key ~ ']' }}"#}
|
||||
indeterminate="false" value="{{ crudp_value }}">
|
||||
<label for="{{ field.name ~ '_' ~ crudp_key ~ '_' }}">{{ button.letter }}</label>
|
||||
</span>
|
||||
</div>
|
||||
{% endfor %}
|
||||
<input type="hidden" name="{{ data_field_name ~ '[' ~ key ~ ']' }}" value="{{ value[key]|default([])|json_encode }}">
|
||||
</div>
|
||||
{% endif %}
|
||||
<button class="button add-item"><i class="fa fa-plus"></i></button>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,96 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% macro renderer(key, text, field, scope) %}
|
||||
|
||||
{% if text is not iterable %}
|
||||
<div class="form-row{% if field.value_only %} array-field-value_only{% endif %}"
|
||||
data-grav-array-type="row">
|
||||
<span data-grav-array-action="sort" class="fa fa-bars"></span>
|
||||
{% if field.value_only != true %}
|
||||
{% if key == '0' and text == '' %}
|
||||
{% set key = '' %}
|
||||
{% endif %}
|
||||
|
||||
<input
|
||||
data-grav-array-type="key"
|
||||
type="text" value="{{ key }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
placeholder="{{ field.placeholder_key|t }}" />
|
||||
{% endif %}
|
||||
|
||||
{% if field.value_type == 'textarea' %}
|
||||
<textarea
|
||||
data-grav-array-type="value"
|
||||
name="{{ ((scope ~ field.name)|fieldName) ~ '[' ~ key ~ ']' }}"
|
||||
placeholder="{{ field.placeholder_value|t }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}>{{ text }}</textarea>
|
||||
{% else %}
|
||||
<input
|
||||
data-grav-array-type="value"
|
||||
type="text"
|
||||
name="{{ ((scope ~ field.name)|fieldName) ~ '[' ~ key ~ ']' }}"
|
||||
placeholder="{{ field.placeholder_value|t }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
value={% if text == 'true' %}true{% elseif text == 'false' %}false{% else %}"{{ text|join(', ') }}"{% endif %} />
|
||||
{% endif %}
|
||||
|
||||
<span data-grav-array-action="rem" class="fa fa-minus"></span>
|
||||
<span data-grav-array-action="add" class="fa fa-plus"></span>
|
||||
</div>
|
||||
{% endif %}
|
||||
{% endmacro %}
|
||||
|
||||
{% import _self as array_field %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-array-name="{{ (scope ~ field.name)|fieldName }}"
|
||||
data-grav-array-keyname="{{ field.placeholder_key|t }}"
|
||||
data-grav-array-valuename="{{ field.placeholder_value|t }}"
|
||||
data-grav-array-textarea="{{ field.value_type == 'textarea' }}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div class="{{ field.size }}" data-grav-array-type="container"{% if field.value_only %} data-grav-array-mode="value_only"{% endif %}{{ value|length <= 1 ? ' class="one-child"' : '' }}>
|
||||
{% if value|length %}
|
||||
{% for key, text in value -%}
|
||||
{% if text is not iterable %}
|
||||
{{ array_field.renderer(key, text, field, scope) }}
|
||||
{% else %}
|
||||
{# Backward compatibility for nested arrays (metas) which are not supported anymore #}
|
||||
{% for subkey, subtext in text -%}
|
||||
{{ array_field.renderer(key ~ '[' ~ subkey ~ ']', subtext, field, scope) }}
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{%- else -%}
|
||||
{# Empty value, mock the entry field#}
|
||||
<div class="form-row" data-grav-array-type="row">
|
||||
<span data-grav-array-action="sort" class="fa fa-bars"></span>
|
||||
{% if field.value_only != true %}
|
||||
<input
|
||||
data-grav-array-type="key"
|
||||
type="text"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
placeholder="{{ field.placeholder_key|t }}" />
|
||||
{% endif %}
|
||||
{% if field.value_type == 'textarea' %}
|
||||
<textarea
|
||||
data-grav-array-type="value"
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
placeholder="{{ field.placeholder_value|t }}"></textarea>
|
||||
{% else %}
|
||||
<input
|
||||
data-grav-array-type="value"
|
||||
type="text"
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
placeholder="{{ field.placeholder_value|t }}" />
|
||||
{% endif %}
|
||||
<span data-grav-array-action="rem" class="fa fa-minus"></span>
|
||||
<span data-grav-array-action="add" class="fa fa-plus"></span>
|
||||
</div>
|
||||
{%- endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,32 @@
|
||||
{% set delete_url = uri.addNonce(base_url_relative ~ "/backup.json/backup:%BACKUP_FILE/task" ~ config.system.param_sep ~ 'backupDelete', 'admin-form', 'admin-nonce') %}
|
||||
<table class="backups-history noflex">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>#</th>
|
||||
<th>{{ "PLUGIN_ADMIN.BACKUP_DATE"|t }}</th>
|
||||
<th>{{ "PLUGIN_ADMIN.NAME"|t }}</th>
|
||||
<th class="right pad">{{ "PLUGIN_ADMIN.SIZE"|t }}</th>
|
||||
<th class="right pad">{{ "PLUGIN_ADMIN.ACTION"|t }}</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% for backup in backups|default([]) %}
|
||||
{% set encoded_name = backup.filename|base64_encode|url_encode %}
|
||||
{% set backup_delete = delete_url|replace({'%BACKUP_FILE': encoded_name}) %}
|
||||
<tr>
|
||||
<td>{{ loop.index }}</td>
|
||||
<td> <i class="fa fa-clock-o"></i> {{ backup.date|date }}</td>
|
||||
<td>{{ backup.title }}</td>
|
||||
<td class="right pad">{{ backup.size|nicefilesize }}</td>
|
||||
<td class="right pad nowrap" >
|
||||
<a class="button button-small hint--bottom" href="{{ grav.backups.getBackupDownloadUrl(backup.filename, admin.base) }}" data-hint="Download"><i class="fa fa-download"></i></a>
|
||||
<span class="button button-small danger hint--bottom" data-hint="Delete" data-backup data-ajax="{{ backup_delete }}"><i class="fa fa-close"></i></span>
|
||||
</td>
|
||||
</tr>
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="5" class="error" style="text-align: center;">{{ "PLUGIN_ADMIN.BACKUPS_NOT_GENERATED"|t }}</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
@@ -0,0 +1,12 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
<input
|
||||
data-grav-field="hidden"
|
||||
data-grav-disabled="false"
|
||||
type="hidden"
|
||||
class="input"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
value="{{ blueprints.name }}" />
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,10 @@
|
||||
{% extends "forms/fields/editor/editor.html.twig" %}
|
||||
{% set theme = config.plugins.admin.whitelabel.codemirror_theme ?? 'paper.css' %}
|
||||
{% set font = config.plugins.admin.whitelabel.codemirror_md_font ?? 'sans' %}
|
||||
{% set codemirrorOptions = {'spellcheck': 'true', 'inputStyle': 'contenteditable', 'mode': 'gfm', 'theme': theme, 'font': font, 'ignore': []}|merge(field.codemirror|default({})) %}
|
||||
|
||||
{# backward compatibility #}
|
||||
{% if field.showPreview %}
|
||||
{% set codemirrorOptions = codemirrorOptions|merge({'ignore': []}) %}
|
||||
{% endif %}
|
||||
{# end backward compatibility #}
|
||||
@@ -0,0 +1,38 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set pattern = '^#([a-fA-F0-9]{6})|(rgba\\(\\s*(0|[1-9]\\d?|1\\d\\d?|2[0-4]\\d|25[0-5])\\s*,\\s*(0|[1-9]\\d?|1\\d\\d?|2[0-4]\\d|25[0-5])\\s*,\\s*(0|[1-9]\\d?|1\\d\\d?|2[0-4]\\d|25[0-5])\\s*,\\s*((0.[0-9]+)|(1.00)|1.0|1)\\s*\\))$' %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-list-wrapper {{ field.size }}" data-type="collection">
|
||||
<div class="g-colorpicker">
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<input
|
||||
data-grav-colorpicker="{{ {update: '.g-colorpicker-preview-wrap .g-colorpicker-preview'}|json_encode|e('html_attr') }}"
|
||||
{# required attribute structures #}
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
value="{{ input_value }}"
|
||||
type="text"
|
||||
{# input attribute structures #}
|
||||
{% block input_attributes %}
|
||||
{% if field.classes is defined %}class="{{ field.classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.autocomplete in ['on', 'off'] %}autocomplete="{{ field.autocomplete }}"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
pattern="{{ field.validate.pattern|default(pattern)|raw }}"
|
||||
{% if field.validate.message %}title="{{ field.validate.message|t }}"
|
||||
{% elseif field.title is defined %}title="{{ field.title|t }}" {% endif %}
|
||||
{% endblock %}
|
||||
/>
|
||||
<div class="g-colorpicker-preview-wrap">
|
||||
<div class="g-colorpicker-preview" style="background-color: {{ value }}"></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
@@ -0,0 +1,40 @@
|
||||
{% set originalValue = originalValue is defined ? originalValue : value %}
|
||||
{% set value = (value is null ? field.default : value) %}
|
||||
{% set classes = field.classes %}
|
||||
|
||||
{% block field %}
|
||||
{% block contents %}
|
||||
{% block group %}
|
||||
{% block input %}
|
||||
<div class="colorbar-element-container g-colorpicker">
|
||||
<div class="colorbar-element light-border" style="background-color: {{value}}">
|
||||
<div class="colorbar-title">{{ field.help|t }}</div>
|
||||
|
||||
<input
|
||||
data-grav-colorpicker="{{ {update: '.colorbar-element', offset: {x: -2, y: 20}} |json_encode|e('html_attr') }}"
|
||||
{# required attribute structures #}
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
value="{{ value }}"
|
||||
autocomplete="off"
|
||||
{# input attribute structures #}
|
||||
{% block input_attributes %}
|
||||
{% if classes %}class="{{ classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.validate.pattern %}pattern="{{ field.validate.pattern }}"{% endif %}
|
||||
{% if field.validate.message %}title="{{ field.validate.message|t }}"
|
||||
{% elseif field.title is defined %}title="{{ field.title|t }}" {% endif %}
|
||||
{% endblock %}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
{% endblock %}
|
||||
{% endblock %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,12 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
<div class="colorbar current-scheme">
|
||||
{% include 'forms/default/fields.html.twig' with {name: field.name, fields: field.fields} %}
|
||||
</div>
|
||||
|
||||
{% endblock %}
|
||||
|
||||
|
||||
|
||||
|
||||
@@ -0,0 +1,8 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
{% embed 'forms/default/fields.html.twig' with {name: name, fields: field.fields} %}
|
||||
{% block outer_markup_field_open %}<div class="form-column block pure-u-1-{{ cols }}">{% endblock %}
|
||||
{% block outer_markup_field_close %}</div>{% endblock %}
|
||||
{% endembed %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,8 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
<div class="form-columns grid pure-g">
|
||||
{% set cols = field.fields|length %}
|
||||
{% include 'forms/default/fields.html.twig' with {name: field.name|parent_field, fields: field.fields, fallback_field: 'column', cols: cols} %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,7 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<input type="hidden" class="input" name="{{ (scope ~ field.name)|fieldName }}" value="{{ input_value }}" />
|
||||
<div class="cron-selector"></div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,71 @@
|
||||
{% set jobs = grav.scheduler.getAllJobs() %}
|
||||
{% set job_states = grav.scheduler.getJobStates().content() %}
|
||||
|
||||
<table class="cron-status noflex">
|
||||
<thead>
|
||||
<tr>
|
||||
<th style="flex:3;">Job ID</th>
|
||||
<th style="flex:3;">Run</th>
|
||||
<th>Status</th>
|
||||
<th class="right pad">State</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% for job in jobs %}
|
||||
{% set job_status = attribute(data.status,job.id) %}
|
||||
{% set job_enabled = job_status is defined and job_status != 'enabled' ? 0 : 1 %}
|
||||
{% set job_id = job.id %}
|
||||
{% set job_id_md5 = job_id|md5 %}
|
||||
{% set job_state = attribute(job_states, job_id) %}
|
||||
{% set job_at = job.getAt|default('* * * * *') %}
|
||||
{% set job_backlink = job.backlink %}
|
||||
<tr>
|
||||
<td style="flex:3;overflow:hidden;">
|
||||
{% if job_backlink %}
|
||||
<a href="{{ admin_route(job_backlink) }}"><kbd>{{ job.id }}</kbd></a>
|
||||
{% else %}
|
||||
<kbd>{{ job.id }}</kbd>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="flex:3;" class="cron-at">
|
||||
{% if job_enabled %}
|
||||
<span class="hint--bottom" data-hint="next run: {{ cron(job_at).getNextRunDate()|date(config.date_format.default) }}">{{ job_at|nicecron }}</span>
|
||||
{% else %}
|
||||
{{ job_at|nicecron }}
|
||||
{% endif %}
|
||||
</td>
|
||||
<td>
|
||||
{% if job_state.state == 'failure' %}
|
||||
{% set run_type = 'error' %}
|
||||
{% set run_hint = job_state.error %}
|
||||
{% set run_text = "<i class=\"fa fa-warning\"></i> Failure" %}
|
||||
{% else %}
|
||||
{% set run_type = 'info' %}
|
||||
{% if job_state.state is not defined %}
|
||||
{% set run_hint = "not run yet" %}
|
||||
{% set run_text = "<i class=\"fa fa-check\"></i> Ready" %}
|
||||
{% else %}
|
||||
{% set run_hint = "last run: " ~ attribute(job_state,'last-run')|date(config.date_format.default) %}
|
||||
{% set run_text = "<i class=\"fa fa-check\"></i> Success" %}
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
<span class="hint--bottom" data-hint="{{ run_hint }}">
|
||||
<span class="badge {{ run_type }}">{{ run_text|raw }}</span>
|
||||
</span>
|
||||
</td>
|
||||
<td class="right pad">
|
||||
<div class="form-data" data-grav-field="toggle" data-grav-disabled="" data-grav-default="null" data-grav-field-name="data[status][{{ job_id }}]">
|
||||
<div class="switch-toggle switch-grav switch-2 ">
|
||||
<input type="radio" value="enabled" id="toggle_status.{{ job_id_md5 }}1" name="data[status][{{ job_id }}]" class="highlight" {% if job_enabled %}checked="checked"{% endif %}>
|
||||
<label for="toggle_status.{{ job_id_md5 }}1">Enabled</label>
|
||||
|
||||
<input type="radio" value="disabled" id="toggle_status.{{ job_id_md5 }}0" name="data[status][{{ job_id }}]" class="" {% if not job_enabled %}checked="checked"{% endif %}>
|
||||
<label for="toggle_status.{{ job_id_md5 }}0">Disabled</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
@@ -0,0 +1,24 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-selectize="{{ (field.selectize is defined ? field.selectize : {})|json_encode()|e('html_attr') }}"
|
||||
data-grav-field="select"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-select-wrapper {{ field.size }}">
|
||||
<select class="{{ field.classes }}" name="{{ (scope ~ field.name)|fieldName ~ (field.multiple ? '[]' : '') }}"
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.multiple in ['on', 'true', 1] %}multiple="multiple"{% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
{% if field.form %}form="{{ field.form }}"{% endif %}
|
||||
>
|
||||
{% for key, text in field.options %}
|
||||
<option {% if key == value or text in value %}selected="selected"{% endif %} value="{{ field.multiple ? text : key }}">{{ key is empty ? text : "now"|date(key) }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,27 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set value = (value is null ? field.default : value) %}
|
||||
{% set default_php_dateformat = admin.guessDateFormat(value) %}
|
||||
{% set php_dateformat = field.format ?: (form ? form.object.dateformat : admin.page.dateformat) ?: config.system.pages.dateformat.default ?: default_php_dateformat %}
|
||||
{% set js_dateformat = admin.dateformatToMomentJS(php_dateformat) %}
|
||||
{% set value = (value is null ? value : value|date(php_dateformat)) %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-input-wrapper datetime-picker-wrapper {{ field.size }}">
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<input
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
value="{{ input_value }}"
|
||||
{% block input_attributes %}
|
||||
type="text"
|
||||
data-grav-datetime="{{ {'format': js_dateformat} | json_encode | e('html_attr') }}"
|
||||
{% if field.validate.min %}min="{{ (field.validate.min is null ? field.validate.min : field.validate.min|date(php_dateformat)) }}"{% endif %}
|
||||
{% if field.validate.max %}max="{{ (field.validate.max is null ? field.validate.max : field.validate.max|date(php_dateformat)) }}"{% endif %}
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
/>
|
||||
<span class="field-icons">
|
||||
<i class="fa fa-fw fa-calendar"></i>
|
||||
</span>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,62 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set whitelabel = config.plugins.admin.whitelabel %}
|
||||
{% set theme = whitelabel.codemirror_theme ?? 'paper.css' %}
|
||||
{% set fontsize = whitelabel.codemirror_fontsize ?? 'lg' %}
|
||||
|
||||
{% set value = value ?? field.default|t %}
|
||||
{% if not codemirrorOptions %}
|
||||
{% set codemirrorOptions = {'mode': 'gfm', 'theme': theme, 'ignore': ['code', 'preview'], 'font': 'mono'}|merge(field.codemirror|default({})) %}
|
||||
{% endif %}
|
||||
|
||||
{% if codemirrorOptions.theme %}
|
||||
{% do assets.addCss(theme_url ~ '/css/codemirror/themes/' ~ codemirrorOptions.theme ~ '.css') %}
|
||||
{% endif %}
|
||||
|
||||
{% block field %}
|
||||
{% block label %}
|
||||
{% if field.label %}
|
||||
{% if field.help %}
|
||||
{% set hint = 'data-hint="' ~ field.help|t|raw ~ '"' %}
|
||||
{% endif %}
|
||||
<div class="form-label form-field hint--bottom" {{ hint|raw }}>
|
||||
{{ field.label|t|raw }}
|
||||
{{ field.validate.required in ['on', 'true', 1] ? '<span class="required">*</span>' }}
|
||||
</div>
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
<div class="form-field {{ field.classes|default('') }}">
|
||||
<div class="form-data grav-editor">
|
||||
<div class="grav-editor-content is-active fontsize-{{ codemirrorOptions.font_size|default(fontsize) }} fontfamily-{{ codemirrorOptions.font }}">
|
||||
{% set input_value = value is iterable ? value|join("\n") : value|string %}
|
||||
<textarea
|
||||
data-grav-editor="{{ {'codemirror': codemirrorOptions} | json_encode|e('html_attr') }}"
|
||||
data-grav-editor-mode="editor"
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
{% if field.classes is defined %}class="{{ field.classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder|t }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if 'preview' not in codemirrorOptions.ignore %}data-grav-urlpreview="{{ base_url }}/media/{{ admin.route|trim('/') }}.json"{% endif %}
|
||||
>{{ input_value }}</textarea>
|
||||
</div>
|
||||
{% if field.resizer is not defined or field.resizer not in ['off', 'false', 0] %}<div class="grav-editor-resizer"></div>{% endif %}
|
||||
{% if field.description %}
|
||||
<div class="form-extra-wrapper {{ field.size }} {{ field.wrapper_classes }}">
|
||||
<span class="form-description">
|
||||
{% if field.markdown %}
|
||||
{{ field.description|t|markdown(false)|raw }}
|
||||
{% else %}
|
||||
{{ field.description|t|raw }}
|
||||
{% endif %}
|
||||
</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,23 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
{% set parent = parent_name|parent_field %}
|
||||
{% set plain_name = (field.plain_name ?? field.name)|string %}
|
||||
{% set name = parent ~ '.' ~ plain_name %}
|
||||
{% if field.field %}
|
||||
{% set fields = prepare_form_fields({(name): field.field}) %}
|
||||
{% else %}
|
||||
{% set fields = prepare_form_fields(field.fields, name) %}
|
||||
{% endif %}
|
||||
|
||||
{% embed 'forms/default/fields.html.twig' with {name: name, fields: fields} %}
|
||||
{% set initial_state = plain_name is not same as (parent_value|string) ? 'display: none;' %}
|
||||
{% block outer_markup_field_open %}
|
||||
<div id="{{ parent_name ~ '__' ~ plain_name }}" class="form-element" style="{{ initial_state }}">
|
||||
{% endblock %}
|
||||
|
||||
{% block outer_markup_field_close %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
{% endembed %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,20 @@
|
||||
{% extends "forms/fields/select/select.html.twig" %}
|
||||
{% set field = field|merge({ autocomplete: 'off' }) %}
|
||||
|
||||
{% if grav.admin is not defined %}
|
||||
{# load some frontend CSS/JS #}
|
||||
{% endif %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-elements="{{ field.name }}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block field %}
|
||||
{{ parent() }}
|
||||
{% set parent_name = field.name %}
|
||||
{% set parent_value = value %}
|
||||
{% if fields|length %}
|
||||
{% include 'forms/default/fields.html.twig' with {name: parent_name, fields: field.fields} %}
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,81 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
<div class="form-fieldset{% if vertical %} vertical{% endif %}{% if field.classes is defined %} {{ field.classes }}{% endif %}">
|
||||
{% block contents %}
|
||||
<input type="checkbox" class="hidden" id="fieldset_collapsible_{{ field.name }}"{% if not field.collapsible or not field.collapsed %} checked="checked"{% endif %} />
|
||||
|
||||
<div class="form-label form-fieldset--label">
|
||||
<h2>
|
||||
<label{% if field.collapsible %} for="fieldset_collapsible_{{ field.name }}" class="form-fieldset--cursor"{% endif %}>
|
||||
{% if field.help %}
|
||||
<span class="hint--bottom" data-hint="{{ field.help|t }}">
|
||||
{% endif %}
|
||||
{% block label %}
|
||||
{% if field.icon %}
|
||||
<i class="fa fa-fw fa-{{ field.icon }}"></i>
|
||||
{% endif %}
|
||||
{{ field.title|t }}
|
||||
{{ field.validate.required in ['on', 'true', 1] ? '<span class="required">*</span>' }}
|
||||
{% if field.info %}
|
||||
<span class="form-fieldset--info">{{ field.info|t }}</span>
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
{% if field.help %}
|
||||
</span>
|
||||
{% endif %}
|
||||
|
||||
{# Actions panel #}
|
||||
<span class="actions">
|
||||
{% block actions %}
|
||||
{% if field.collapsible %}
|
||||
<span class="form-fieldset--collapsible">
|
||||
<i class="fa fa-chevron-down open"></i>
|
||||
<i class="fa fa-chevron-up close"></i>
|
||||
</span>
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
</span>
|
||||
</label>
|
||||
</h2>
|
||||
</div>
|
||||
<div class="form-data"
|
||||
{#{% block global_attributes %}
|
||||
data-grav-field="{{ field.type }}"
|
||||
data-grav-disabled="{{ originalValue is null ? 'true' : 'false' }}"
|
||||
data-grav-default="{{ field.default|json_encode()|e('html_attr') }}"
|
||||
{% endblock %}#}
|
||||
>
|
||||
|
||||
{% block group %}
|
||||
{% if field.text %}
|
||||
{{ field.markdown ? field.text|t|markdown : ('<p>' ~ field.text|t ~ '</p>')|raw }}
|
||||
{% endif %}
|
||||
|
||||
{% if field.fields %}
|
||||
{% for child_name, child in field.fields %}
|
||||
{% set child = prepare_form_field(child, child_name, field.name, {key: key}) %}
|
||||
{% if child %}
|
||||
{% set default_layout = 'text' %}
|
||||
{% if child.type == 'key' or child.key == true %}
|
||||
{# Special handling for the key field #}
|
||||
{% set default_layout = 'key' %}
|
||||
{% set child_value = key %}
|
||||
{% elseif child.name == 'value' %}
|
||||
{# Special handling for the value field #}
|
||||
{% set child = child|merge({ name: field.name }) %}
|
||||
{% set child_value = value %}
|
||||
{% else %}
|
||||
{% set child_value = form ? form.value(child.name) : data.value(child.name) %}
|
||||
{% endif %}
|
||||
|
||||
{% set field_templates = include_form_field(child.type, field_layout, default_layout) %}
|
||||
{% include field_templates with { field: child, value: child_value, originalValue: null } %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,118 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% macro bytesToSize(bytes) -%}
|
||||
{% spaceless %}
|
||||
{% set kilobyte = 1024 %}
|
||||
{% set megabyte = kilobyte * 1024 %}
|
||||
{% set gigabyte = megabyte * 1024 %}
|
||||
{% set terabyte = gigabyte * 1024 %}
|
||||
|
||||
{% if bytes < kilobyte %}
|
||||
{{ bytes ~ ' B' }}
|
||||
{% elseif bytes < megabyte %}
|
||||
{{ (bytes / kilobyte)|number_format(2, '.') ~ ' KB' }}
|
||||
{% elseif bytes < gigabyte %}
|
||||
{{ (bytes / megabyte)|number_format(2, '.') ~ ' MB' }}
|
||||
{% elseif bytes < terabyte %}
|
||||
{{ (bytes / gigabyte)|number_format(2, '.') ~ ' GB' }}
|
||||
{% else %}
|
||||
{{ (bytes / terabyte)|number_format(2, '.') ~ ' TB' }}
|
||||
{% endif %}
|
||||
{% endspaceless %}
|
||||
{%- endmacro %}
|
||||
|
||||
{% macro preview(path, value, global) %}
|
||||
{% if value %}
|
||||
{% set uri = global.grav.uri %}
|
||||
{% set files = global.files %}
|
||||
{% set config = global.grav.config %}
|
||||
{% set route = global.context.route() %}
|
||||
|
||||
{% set type = global.blueprint_type ? global.blueprint_type : global.admin.location ? global.admin.location : 'config' %}
|
||||
|
||||
{% set blueprint_name = global.blueprints.getFilename %}
|
||||
{% set real_path = path %}
|
||||
|
||||
{% if type == 'pages' %}
|
||||
{% set blueprint_name = type ~ '/' ~ blueprint_name %}
|
||||
{% set real_path = (value.thumb ?? global.context.media[path].relativePath ?? global.form.getPagePathFromToken(path) ?? global.admin.getPagePathFromToken(path))|ltrim('/') %}
|
||||
{% endif %}
|
||||
{% set blueprint = blueprint_name|base64_encode %}
|
||||
|
||||
|
||||
{% set remove = global.file_task_remove ? global.file_url_remove : uri.addNonce(
|
||||
global.file_url_remove ~
|
||||
'/media.json' ~
|
||||
'/task' ~ config.system.param_sep ~ 'removeFileFromBlueprint' ~
|
||||
'/proute' ~ config.system.param_sep ~ (route|base64_encode) ~
|
||||
'/blueprint' ~ config.system.param_sep ~ blueprint ~
|
||||
'/type' ~ config.system.param_sep ~ type ~
|
||||
'/field' ~ config.system.param_sep ~ files.name ~
|
||||
'/path' ~ config.system.param_sep ~ (value.path|base64_encode), 'admin-form', 'admin-nonce') %}
|
||||
|
||||
{% set file = value|merge({remove: remove, path: value.thumb_url ?? (uri.rootUrl == '/' ? '/' : uri.rootUrl ~ '/' ~ real_path) }) %}
|
||||
<div class="hidden" data-file="{{ file|json_encode|e('html_attr') }}"></div>
|
||||
{% endif %}
|
||||
{% endmacro %}
|
||||
|
||||
{% import _self as macro %}
|
||||
|
||||
{% set defaults = config.plugins.form %}
|
||||
{% set files = defaults.files|merge(field|default([])) %}
|
||||
{% set limit = not field.multiple ? 1 : files.limit %}
|
||||
|
||||
{% block input %}
|
||||
{% set page_can_upload = exists or (type == 'page' and not exists and not (field.destination starts with '@self' or field.destination starts with 'self@')) %}
|
||||
{% if form or (type is not defined or page_can_upload) %}
|
||||
|
||||
{% block prepend %}{% endblock %}
|
||||
{% set settings = {name: field.name, paramName: (scope ~ field.name)|fieldName ~ (files.multiple ? '[]' : ''), limit: limit, filesize: form_max_filesize, accept: files.accept, resolution: files.resolution, resizeWidth: files.resizeWidth, resizeHeight: files.resizeHeight, resizeQuality: files.resizeQuality } %}
|
||||
{% set dropzoneSettings = field.dropzone %}
|
||||
{% if form.getMediaTaskRoute() %}
|
||||
{% set file_url_add = base_url_relative ~ form.getMediaTaskRoute({}, 'json') %}
|
||||
{% set file_task_add = {task: 'media.upload', name: field.name, '__form-name__': form.name, '__unique_form_id__': form.uniqueid} %}
|
||||
|
||||
{% set file_url_remove = base_url_relative ~ form.getMediaTaskRoute({}, 'json') %}
|
||||
{% set file_task_remove = {task: 'media.delete', name: field.name, '__form-name__': form.name, '__unique_form_id__': form.uniqueid} %}
|
||||
{% else %}
|
||||
{% set file_url_remove = file_url_remove ?: base_url_relative %}
|
||||
{% endif %}
|
||||
|
||||
<div class="{{ form_field_wrapper_classes ?: 'form-input-wrapper' }} {{ field.classes }} dropzone files-upload {% if field.fancy is not same as(false) %}form-input-file{% endif %} {{ field.size }}"
|
||||
data-grav-file-settings="{{ settings|json_encode|e('html_attr') }}"
|
||||
data-dropzone-options="{{ dropzoneSettings|json_encode|e('html_attr') }}"
|
||||
{% if file_task_add and file_task_remove %}
|
||||
data-file-post-add="{{ file_task_add|json_encode|e('html_attr') }}"
|
||||
data-file-post-remove="{{ file_task_remove|json_encode|e('html_attr') }}"
|
||||
data-file-url-add="{{ file_url_add }}"
|
||||
data-file-url-remove="{{ file_url_remove }}"
|
||||
{% else %}
|
||||
{% if file_url_add %}data-file-url-add="{{ file_url_add }}"{% endif %}
|
||||
{% if file_url_remove %}data-file-url-remove="{{ file_url_remove }}"{% endif %}
|
||||
{% endif %}
|
||||
>
|
||||
|
||||
{% block file_extras %}{% endblock %}
|
||||
<input
|
||||
{# required attribute structures #}
|
||||
{% block input_attributes %}
|
||||
type="file"
|
||||
{% if files.multiple %}multiple="multiple"{% endif %}
|
||||
{% if files.accept %}accept="{{ files.accept|join(',') }}"{% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
{% if field.random_name %}random="true"{% endif %}
|
||||
{% if required %}required="required"{% endif %}
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
/>
|
||||
|
||||
{% for path, file in value %}
|
||||
{{ macro.preview(path, file, _context) }}
|
||||
{% endfor %}
|
||||
{% include 'forms/fields/hidden/hidden.html.twig' with {field: {name: '_json.' ~ field.name}, value: (value ?? [])|json_encode} %}
|
||||
</div>
|
||||
|
||||
{% else %}
|
||||
<span class="note">{{ "PLUGIN_ADMIN.CANNOT_ADD_FILES_PAGE_NOT_SAVED"|t|raw }}</span>
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,27 @@
|
||||
{% embed "forms/fields/select/select.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
{% if not field.options and value %}
|
||||
{% if value is iterable %}
|
||||
{% set options = {} %}
|
||||
{% for val in value %}
|
||||
{% set options = options|merge({(val): val}) %}
|
||||
{% endfor %}
|
||||
{% else %}
|
||||
{% set options = {(value): value} %}
|
||||
{% endif %}
|
||||
{% set field = field|merge({options: options}) %}
|
||||
{% endif %}
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-filepicker
|
||||
data-name="{{ field.name }}"
|
||||
{% if field.preview_images %}data-preview-images{% endif %}
|
||||
{% if field.on_demand %}data-ondemand{% endif %}
|
||||
data-value="{{ input_value }}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
{% endembed %}
|
||||
@@ -0,0 +1,8 @@
|
||||
{% extends "forms/fields/text/text.html.twig" %}
|
||||
{% set field = field|merge({'wrapper_classes': 'form-input-addon-wrapper'}) %}
|
||||
|
||||
{% block append %}
|
||||
<div class="form-input-addon form-input-append hint--top" data-hint="{{ 'PLUGIN_ADMIN.AUTOREGENERATE_FOLDER_SLUG'|t }}" data-regenerate='[name="{{ (scope ~ field.name)|fieldName }}"]'>
|
||||
<i class="fa fa-refresh"></i>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,10 @@
|
||||
{# Deprecated field. Alias to editor/editor.html.twig #}
|
||||
|
||||
{% extends "forms/fields/editor/editor.html.twig" %}
|
||||
{% set codemirrorOptions = {'mode': 'yaml', 'indentUnit': 4, ignore: ['code'], 'autofocus': true, 'indentWithTabs': false, 'lineNumbers': true, 'gutters': ["CodeMirror-lint-markers"]}|merge(field.codemirror|default({})) %}
|
||||
|
||||
{# backward compatibility #}
|
||||
{% if field.showPreview %}
|
||||
{% set codemirrorOptions = codemirrorOptions|merge({'ignore': []}) %}
|
||||
{% endif %}
|
||||
{# end backward compatibility #}
|
||||
@@ -0,0 +1,44 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-iconpicker
|
||||
data-name="{{field.name}}"
|
||||
data-value="{{value}}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-list-wrapper {{ field.size }}" data-type="collection">
|
||||
<div class="icon-picker" data-pickerid="fa" data-iconsets='{"fa":"FontAwesome Icons"}'>
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<input
|
||||
{# required attribute structures #}
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
data-format="{{ field.format == 'short' ? 'short' : 'long' }}"
|
||||
value="{{ input_value }}"
|
||||
type="text"
|
||||
{# input attribute structures #}
|
||||
{% block input_attributes %}
|
||||
{% if field.classes is defined %}class="{{ field.classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.autocomplete in ['on', 'off'] %}autocomplete="{{ field.autocomplete }}"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% endblock %}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="fa-set icon-set">
|
||||
<ul>
|
||||
{% for icon in fa_icons %}
|
||||
<li data-code="{{icon.unicode}}" data-class="fa fa-{{icon.id}}" class="fa fa-{{icon.id}}"></li>
|
||||
{% endfor %}
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,188 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set value = (value is null ? field.default : value) %}
|
||||
{% set name = field.name %}
|
||||
{% set btnLabel = field.btnLabel is defined ? field.btnLabel : "PLUGIN_ADMIN.ADD_ITEM" %}
|
||||
{% set btnSortLabel = field.btnSortLabel is defined ? field.btnSortLabel : "PLUGIN_ADMIN.SORT_BY" %}
|
||||
{% set fieldControls = field.controls|default('bottom') %}
|
||||
|
||||
{% block contents %}
|
||||
<div class="form-label{% if not vertical %} block size-1-3 pure-u-1-3{% endif %}">
|
||||
{% if field.toggleable %}
|
||||
<span class="checkboxes toggleable" data-grav-field="toggleable" data-grav-field-name="{{ (scope ~ field.name)|fieldName }}">
|
||||
<input type="checkbox"
|
||||
id="toggleable_{{ field.name }}"
|
||||
{% if toggleableChecked %}value="1"{% endif %}
|
||||
name="toggleable_{{ (scope ~ field.name)|fieldName }}"
|
||||
{% if toggleableChecked %}checked="checked"{% endif %}
|
||||
>
|
||||
<label for="toggleable_{{ field.name }}"></label>
|
||||
</span>
|
||||
{% endif %}
|
||||
<label{{ (field.toggleable ? ' class="toggleable" for="toggleable_' ~ field.name ~ '"')|raw }}>
|
||||
{% if field.help %}
|
||||
<span class="hint--bottom" data-hint="{{ field.help|t }}">{{ field.label|t }}</span>
|
||||
{% else %}
|
||||
{{ field.label|t }}
|
||||
{% endif %}
|
||||
{{ field.validate.required in ['on', 'true', 1] ? '<span class="required">*</span>' }}
|
||||
</label>
|
||||
</div>
|
||||
<div class="form-data{% if not vertical %} block size-2-3 pure-u-2-3{% endif %}"
|
||||
{% block global_attributes %}
|
||||
data-grav-field="{{ field.type }}"
|
||||
data-grav-disabled="{{ toggleableChecked }}"
|
||||
data-grav-default="{{ field.default|json_encode|e('html_attr') }}"
|
||||
{% endblock %}
|
||||
>
|
||||
|
||||
<div class="form-list-wrapper {{ field.size }}" data-type="collection"
|
||||
{% if field.selectunique %}
|
||||
data-select-unique="{{ field.selectunique|json_encode|e('html_attr') }}"
|
||||
data-max="{{ field.selectunique|length }}"
|
||||
{% endif %}
|
||||
{% if field.min is defined %}data-min="{{ field.min }}"{% endif %}
|
||||
{% if field.max is defined and not field.selectunique %}data-max="{{ field.max }}"{% endif %}
|
||||
>
|
||||
{% if fieldControls in ['top', 'both'] %}
|
||||
<div class="collection-actions">
|
||||
{% if collapsible %}
|
||||
<button class="button" type="button" data-action="expand_all"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-chevron-circle-down"></i> {{ "PLUGIN_ADMIN.EXPAND_ALL"|t }}</button>
|
||||
<button class="button" type="button" data-action="collapse_all"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-chevron-circle-right"></i> {{ "PLUGIN_ADMIN.COLLAPSE_ALL"|t }}</button>
|
||||
{% endif %}
|
||||
{% if field.sortby %}
|
||||
<button class="button{{ not value|length ? ' hidden' : '' }}" type="button" data-action="sort" data-action-sort="{{ field.sortby }}" data-action-sort-dir="{{ field.sortby_dir|default('asc') }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-sort-amount-{{ field.sortby_dir|default('asc') }}"></i> {{ btnSortLabel|t }} '{{ field.sortby }}'</button>
|
||||
{% endif %}
|
||||
<button class="button" type="button" data-action="add"
|
||||
data-action-add="{{ field.placement is same as('position') ? 'top' : field.placement|default('bottom') }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
><i class="fa fa-plus"></i> {{ btnLabel|t }}</button>
|
||||
</div>
|
||||
{% endif %}
|
||||
<ul {% if field.classes is defined %}class="{{ field.classes }}"{% endif %} data-collection-holder="{{ name }}"
|
||||
{% if field.sort is same as(false) %}
|
||||
data-collection-nosort
|
||||
{% endif %}>
|
||||
{% if field.fields %}
|
||||
{% set collapsible = field.fields|length > 1 and (field.collapsible is not defined or field.collapsible) %}
|
||||
{% for key, val in value %}
|
||||
{% set item_name = name ? name ~ '.' ~ key : key %}
|
||||
<li data-collection-item="{{ item_name }}"
|
||||
data-collection-key="{{ key }}"
|
||||
class="{{ collapsible and field.collapsed ? 'collection-collapsed' : '' }}"
|
||||
{% if field.min_height %}style="min-height:{{ field.min_height }};"{% endif %}>
|
||||
<div class="collection-sort"><i class="fa fa-fw fa-bars"></i></div>
|
||||
{%- for child_name, child in field.fields -%}
|
||||
{% set child = prepare_form_field(child, child_name, item_name) %}
|
||||
{% if child %}
|
||||
{% set child = child|merge({ '_list_index': item_name }) %}
|
||||
{% set default_layout = 'text' %}
|
||||
{% if child.type == 'key' or (child.key == true and child.type != 'list') %}
|
||||
{# Special handling for the key field #}
|
||||
{% set default_layout = 'key' %}
|
||||
{% set child_value = key %}
|
||||
{% elseif child.name == 'value' %}
|
||||
{# Special handling for the value field #}
|
||||
{% set child = child|merge({ name: item_name }) %}
|
||||
{% set child_value = val %}
|
||||
{% else %}
|
||||
{% set child_value = form ? form.value(child.name) : data.value(child.name) %}
|
||||
{# Look for a default value for that field #}
|
||||
{% if child_value is null and val[child_name|trim('.', 'left')] is defined %}
|
||||
{% set child_value = val[child_name|trim('.', 'left')] %}
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
|
||||
{% set field_templates = include_form_field(child.type, field_layout, default_layout) %}
|
||||
{% set template_data = { field: child, value: child_value, originalValue: null } %}
|
||||
{% if default_layout != 'key' %}
|
||||
{% if child.type == 'fieldset' %}
|
||||
{% set template_data = template_data|merge({val: child_value}) %}
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
|
||||
{%- include field_templates with template_data -%}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
<div class="item-actions">
|
||||
{% if collapsible %}
|
||||
<i class="fa fa-chevron-circle-{{ field.collapsed ? 'right' : 'down' }}" data-action="{{ field.collapsed ? 'expand' : 'collapse' }}"></i>
|
||||
<br />
|
||||
{% endif %}
|
||||
<i class="fa fa-trash-o" data-action="confirm"></i>
|
||||
<div class="list-confirm-deletion button danger hidden" data-action="delete">
|
||||
<i class="fa fa-fw text-primary fa-check"></i>
|
||||
<span>{{ 'PLUGIN_ADMIN.DELETE'|t }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</li>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
</ul>
|
||||
{% if fieldControls in ['bottom', 'both'] %}
|
||||
<div class="collection-actions">
|
||||
{% if collapsible %}
|
||||
<button class="button" type="button" data-action="expand_all"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-chevron-circle-down"></i> {{ "PLUGIN_ADMIN.EXPAND_ALL"|t }}</button>
|
||||
<button class="button" type="button" data-action="collapse_all"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-chevron-circle-right"></i> {{ "PLUGIN_ADMIN.COLLAPSE_ALL"|t }}</button>
|
||||
{% endif %}
|
||||
{% if field.sortby %}
|
||||
<button class="button{{ not value|length ? ' hidden' : '' }}" type="button" data-action="sort" data-action-sort="{{ field.sortby }}" data-action-sort-dir="{{ field.sortby_dir|default('asc') }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}><i class="fa fa-sort-amount-{{ field.sortby_dir|default('asc') }}"></i> {{ btnSortLabel|t }} '{{ field.sortby }}'</button>
|
||||
{% endif %}
|
||||
<button class="button" type="button" data-action="add"
|
||||
data-action-add="{{ field.placement is same as('position') ? 'bottom' : field.placement|default('bottom') }}"
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
><i class="fa fa-plus"></i> {{ btnLabel|t }}</button>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{% set template %}
|
||||
{%- set item_name = name ? name ~ '.*' : '*' -%}
|
||||
<li data-collection-item="{{ item_name }}">
|
||||
{% if field.sort is not same as(false) %}
|
||||
<div class="collection-sort"><i class="fa fa-fw fa-bars"></i></div>
|
||||
{% endif %}
|
||||
{%- if field.fields -%}
|
||||
{%- for child_name, child in field.fields -%}
|
||||
{% set child = prepare_form_field(child, child_name, item_name) %}
|
||||
{% if child %}
|
||||
{% set child = child|merge({ '_list_index': item_name }) %}
|
||||
{% set default_layout = 'text' %}
|
||||
{% if child.type == 'key' or child.key == true %}
|
||||
{# Special handling for the key field #}
|
||||
{% set default_layout = 'key' %}
|
||||
{% elseif child.name == 'value' %}
|
||||
{# Special handling for the value field #}
|
||||
{% set child = child|merge({ name: item_name }) %}
|
||||
{% endif %}
|
||||
|
||||
{% set field_templates = include_form_field(child.type, field_layout, default_layout) %}
|
||||
{% include field_templates with { field: child, value: null } %}
|
||||
{% endif %}
|
||||
{%- endfor %}
|
||||
<div class="item-actions">
|
||||
{% if collapsible %}
|
||||
<i class="fa fa-chevron-circle-down" data-action="collapse"></i>
|
||||
<br />
|
||||
{% endif %}
|
||||
<i class="fa fa-trash-o" data-action="confirm"></i>
|
||||
<div class="list-confirm-deletion button danger hidden" data-action="delete">
|
||||
<i class="fa fa-fw text-primary fa-check"></i>
|
||||
<span>{{ 'PLUGIN_ADMIN.DELETE'|t }}</span>
|
||||
</div>
|
||||
</div>
|
||||
{%- endif -%}
|
||||
</li>
|
||||
{% endset %}
|
||||
<div style="display: none;" data-collection-template="new"
|
||||
data-collection-template-html="{{ template|regex_replace('/([ \r\n]+)/', ' ')|e('html_attr') }}"></div>
|
||||
<div style="display: none;" data-collection-config="{{ name }}"></div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
@@ -0,0 +1 @@
|
||||
{% extends "forms/fields/codemirror/codemirror.html.twig" %}
|
||||
@@ -0,0 +1,17 @@
|
||||
{% extends "forms/fields/text/text.html.twig" %}
|
||||
|
||||
{% set unique_identifier = random_string() %}
|
||||
|
||||
{% block global_attributes %}
|
||||
{{ parent() }}
|
||||
data-mediapicker-modal-trigger
|
||||
data-grav-mediapicker-unique-identifier="{{ unique_identifier }}"
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block contents %}
|
||||
{{ parent() }}
|
||||
<div class="remodal remodal-mediapicker" data-remodal-mediapicker data-remodal-unique-identifier="{{ unique_identifier }}" data-remodal-options="hashTracking: false">
|
||||
{% include 'partials/media-list-wrapper.html.twig' with { is_modal: true } %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,121 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% macro renderer(key, content, field, scope, level, parent_key, up_level) %}
|
||||
{% import _self as self %}
|
||||
|
||||
{% macro field(value, key, level, globalvars, disable_name, hidden) %}
|
||||
{% set name = 'data[' ~ globalvars.field.name|replace({'.': ']['}) ~ ']' ~ key %}
|
||||
<div class="form-row array-field-value_only js__multilevel-field {{ level == 0 ? 'top' : '' }}"
|
||||
data-grav-array-type="row" {% if (hidden|default(false) == true) %}style="display: none"{% endif %}>
|
||||
{% set marginDir = not language_codes.rtl(grav.user.language) ? 'margin-left' : 'margin-right' %}
|
||||
<input
|
||||
type="text"
|
||||
{% if (disable_name != true) %}name="{{ name }}"{% endif %}
|
||||
data-attr-name="{{ name }}"
|
||||
placeholder="{{ field.placeholder_value|t }}"
|
||||
style="{{ marginDir }}: {{ level * 50 }}px"
|
||||
value="{{ value }}" />
|
||||
|
||||
<span class="fa fa-minus js__remove-item"></span>
|
||||
<span class="fa fa-plus js__add-sibling hidden" data-level="{{level}}"></span>
|
||||
<span class="fa fa-plus-circle js__add-children hidden" data-level="{{level}}"></span>
|
||||
</div>
|
||||
{% endmacro %}
|
||||
|
||||
{% if level == 0 %}
|
||||
|
||||
{{ self.field(key, '', level, _context, true, (is_numeric(key) ? true : false)) }}
|
||||
|
||||
{% if content is not iterable %}
|
||||
{% set level2 = level + 1 %}
|
||||
|
||||
<div class="children-wrapper">
|
||||
<div class="element-wrapper">
|
||||
{{ self.field(content, '[' ~ key ~ ']', level2, _context) }}
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
|
||||
{% if up_level %}
|
||||
{% set level = level + 1 %}
|
||||
{% endif %}
|
||||
<div class="children-wrapper">
|
||||
{% set unique_child = (is_array(content) and content.length > 1) ? true : false %}
|
||||
|
||||
{% for inner_key, inner_content in content -%}
|
||||
<div class="element-wrapper">
|
||||
{% if not is_numeric(inner_key) %}
|
||||
{% if (content|length > 1) %}
|
||||
{{ self.field(inner_key, parent_key, level, _context, true) }}
|
||||
{% else %}
|
||||
{{ self.field(inner_key, parent_key, level, _context) }}
|
||||
{% endif %}
|
||||
{% set level2 = level + 1 %}
|
||||
{% set up_level = true %}
|
||||
{% else %}
|
||||
{% set up_level = false %}
|
||||
{% set level2 = level %}
|
||||
{% endif %}
|
||||
|
||||
{% if inner_content is not iterable %}
|
||||
|
||||
{% if not is_numeric(inner_key) %}
|
||||
<div class="children-wrapper">
|
||||
<div class="element-wrapper">
|
||||
{% endif %}
|
||||
|
||||
{% set last_key = (is_numeric(inner_key)) ? '' : inner_key %}
|
||||
{{ self.field(inner_content, parent_key ~ '[' ~ inner_key ~ ']', level2, _context) }}
|
||||
|
||||
{% if not is_numeric(inner_key) %}
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
{% else %}
|
||||
|
||||
{% set inner_parent_key = parent_key ~ '[' ~ inner_key ~ ']' %}
|
||||
{{ self.renderer(inner_key, inner_content, field, scope, level, inner_parent_key, up_level) }}
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endmacro %}
|
||||
|
||||
{% import _self as macro %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-array-name="{{ (scope ~ field.name)|fieldName }}"
|
||||
data-grav-array-keyname="{{ field.placeholder_key|t }}"
|
||||
data-grav-array-valuename="{{ field.placeholder_value|t }}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div data-id="{{random_string()}}" data-grav-multilevel-field data-grav-array-type="container" data-grav-array-mode="value_only"{{ value|length <= 1 ? ' class="one-child"' : '' }}>
|
||||
{% if value|length %}
|
||||
{% for key, content in value -%}
|
||||
<div class="element-wrapper">
|
||||
{{ macro.renderer(key, content, field, scope, 0, '[' ~ key ~ ']', true) }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
{%- else -%}
|
||||
{# Empty value, mock the entry field#}
|
||||
<div class="element-wrapper">
|
||||
<div class="form-row array-field-value_only js__multilevel-field"
|
||||
data-grav-array-type="row">
|
||||
|
||||
<input
|
||||
type="text"
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
placeholder="Enter value"
|
||||
value="" />
|
||||
|
||||
<span class="fa fa-minus js__remove-item"></span>
|
||||
<span class="fa fa-plus js__add-sibling hidden" data-level="0" ></span>
|
||||
<span class="fa fa-plus-circle js__add-children hidden" data-level="0"></span>
|
||||
</div>
|
||||
</div>
|
||||
{%- endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,68 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set value = (value is null ? field.default : value) %}
|
||||
{% if form %}
|
||||
{% set siblings = form.object.parent.children.collection %}
|
||||
{% set canOrder = form.object.order %}
|
||||
{% if not form.object.exists %}
|
||||
{% do siblings.add(form.object) %}
|
||||
{% endif %}
|
||||
{% else %}
|
||||
{% do admin.enablePages() %}
|
||||
{% set siblings = context.parent.children %}
|
||||
{% set canOrder = context.order %}
|
||||
{% endif %}
|
||||
{% set vertical = field.style == 'vertical' %}
|
||||
|
||||
{% block field %}
|
||||
<div class="form-field grid pure-g{% if vertical %} vertical{% endif %}">
|
||||
<div class="form-label{% if not vertical %} block size-1-3 pure-u-1-3{% endif %}">
|
||||
<label>
|
||||
{% if field.help %}
|
||||
<span class="tooltip" data-asTooltip-position="w" title="{{ field.help|t }}">{{ field.label|t }}</span>
|
||||
{% else %}
|
||||
{{ field.label|t }}
|
||||
{% endif %}
|
||||
{{ field.validate.required in ['on', 'true', 1] ? '<span class="required">*</span>' }}
|
||||
</label>
|
||||
</div>
|
||||
<div class="form-data{% if not vertical %} block size-2-3 pure-u-2-3{% endif %}">
|
||||
<div class="form-order-wrapper {{ field.size }}">
|
||||
|
||||
<input
|
||||
type="hidden"
|
||||
data-order
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
value="" />
|
||||
{% if not canOrder %}
|
||||
<div class="notice">{{ "PLUGIN_ADMIN.ORDERING_DISABLED_BECAUSE_PAGE_NO_PREFIX"|t|raw }}</div>
|
||||
{% endif %}
|
||||
|
||||
{% if siblings|length < 200 %}
|
||||
{% set sortable_count = 0 %}
|
||||
<ul id="ordering" class="orderable {{ field.classes }}">
|
||||
{% for page in siblings %}
|
||||
{% if page.order %}
|
||||
<li class="drag-handle" data-id="{{ page.slug }}" {{ page.slug == data.slug ? 'data-active-id' : ''}}><span class="page-order">{{ page.order|int }}.</span> {{ page.title ?: 'NEW' }} <a href="{{ getPageUrl(page) }}"><i class="fa fa-external-link"></i></a></li>
|
||||
{% set sortable_count = loop.index %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</ul>
|
||||
{% if sortable_count < siblings|length %}
|
||||
<label>{{ "PLUGIN_ADMIN.UNSORTABLE_PAGES"|t }}</label>
|
||||
<ul class="orderable disabled">
|
||||
{% for page in siblings %}
|
||||
{% if not page.order %}
|
||||
<li {{ page.slug == data.slug ? 'data-active-id' : ''}}>{{ page.title ?: 'NEW' }} <a href="{{ getPageUrl(page) }}"><i class="fa fa-external-link"></i></a></li>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</ul>
|
||||
{% endif %}
|
||||
{% else %}
|
||||
<div class="notice">{{ "PLUGIN_ADMIN.ORDERING_DISABLED_BECAUSE_TOO_MANY_SIBLINGS"|t }}</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,65 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set value = (value is null ? field.default : value) %}
|
||||
|
||||
{% block field %}
|
||||
{% if context.folderExists %}
|
||||
{% set pagemedia = config.get('plugins.admin.pagemedia') %}
|
||||
{% set pagemedia_settings = {
|
||||
resolution: {
|
||||
min: {
|
||||
width: pagemedia.res_min_width ?: null,
|
||||
height: pagemedia.res_min_height ?: null
|
||||
},
|
||||
max: {
|
||||
width: pagemedia.res_max_width ?: null,
|
||||
height: pagemedia.res_max_height ?: null
|
||||
}
|
||||
},
|
||||
resizeWidth: pagemedia.resize_width ?: null,
|
||||
resizeHeight: pagemedia.resize_height ?: null,
|
||||
resizeQuality: pagemedia.resize_quality ?: 0.8
|
||||
} %}
|
||||
|
||||
{% set media_url = form.getMediaTaskRoute() ?: '/media/' ~ admin.route|trim('/') ~ '.json' %}
|
||||
{% set media_local = form.getMediaRoute() ?: base_url_relative_frontend|rtrim('/') ~ '/' ~ admin.route|trim('/') %}
|
||||
{% set media_path = url(context.relativePagePath) %}
|
||||
{% set media_uri = context.mediaUri() %}
|
||||
{% set dropzone_settings = { maxFilesize: form_max_filesize }|merge(pagemedia_settings) %}
|
||||
{% set pageMediaStore = get_cookie('grav-admin-pagemedia')|default('{"width":200,"collapsed":false}')|json_decode %}
|
||||
|
||||
<div class="pagemedia-field form-field grid vertical {% if field.classes is defined %}{{ field.classes }}{% endif %}">
|
||||
<div class="form-label">
|
||||
<label class="media-collapser">
|
||||
<i class="fa fa-fw small fa-chevron-{{ pageMediaStore.collapsed ? 'right' : 'down' }}"></i>
|
||||
{{ field.label|t }} <span data-pagemedia-count>({{ admin.page.media|length }})</span>
|
||||
</label>
|
||||
<div class="{{ pageMediaStore.collapsed ? 'hidden' : '' }}">
|
||||
<input type="range" min="70" step="10" max="200" value="{{ pageMediaStore.width }}" class="media-resizer">
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-data form-uploads-wrapper" style="{{ pageMediaStore.collapsed ? 'display: none;' : '' }}">
|
||||
<div id="grav-dropzone"
|
||||
class="dropzone"
|
||||
data-media-url="{{ (base_url ~ media_url)|e('html_attr') }}"
|
||||
data-media-local="{{ media_local|e('html_attr') }}"
|
||||
data-media-path="{{ media_path|e('html_attr') }}"
|
||||
data-media-uri="{{ media_uri|e('html_attr') }}"
|
||||
data-dropzone-options="{{ dropzone_settings|json_encode|e('html_attr') }}"
|
||||
data-dropzone-field="{{ (scope ~ field.name)|fieldName }}"></div>
|
||||
|
||||
{% if admin.session.expert == '0' or not user.authorize('admin.super') %}
|
||||
<input type="hidden" name="{{ (scope ~ field.name)|fieldName }}" value="{{ value }}" />
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
{% else %}
|
||||
<div class="form-tab">
|
||||
<div class="form-field">
|
||||
<div class="form-label">
|
||||
{{ "PLUGIN_ADMIN.CANNOT_ADD_MEDIA_FILES_PAGE_NOT_SAVED"|t }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1 @@
|
||||
{% extends "forms/fields/filepicker/filepicker.html.twig" %}
|
||||
@@ -0,0 +1,52 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% macro page_options(globals, pages_list) %}
|
||||
{% set field = globals.field %}
|
||||
{% set value = globals.value %}
|
||||
{% if field.options and depth == 0 %}
|
||||
{% for key, value in field.options %}
|
||||
<option value="{{ key|e('html_attr') }}">{{ value|t }}</option>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
|
||||
{% for page_route, option in pages_list %}
|
||||
<option {% if page_route == value or (field.multiple and page_route in value) %}selected="selected"{% endif %} value="{{ page_route }}">{{ option|raw }}</option>
|
||||
{% endfor %}
|
||||
{% endmacro %}
|
||||
|
||||
{% import _self as macro %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-selectize="{{ (field.selectize is defined ? field.selectize : {})|json_encode|e('html_attr') }}"
|
||||
data-grav-field="select"
|
||||
data-grav-disabled="{{ originalValue is null ? 'true' : 'false' }}"
|
||||
data-grav-default="{{ field.default|json_encode|e('html_attr') }}"
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
{% do admin.enablePages %}
|
||||
{% set start_page = field.start_route ? pages.find(field.start_route) : null %}
|
||||
{% set show_all = field.show_all is same as(false) ? false : true %}
|
||||
{% set show_fullpath = field.show_fullpath is same as(true) ? true : false %}
|
||||
{% set show_slug = field.show_slug is same as(true) ? true : false %}
|
||||
{% set show_modular = field.show_modular is same as(true) ? true : false %}
|
||||
{% set limit_levels = field.limit_levels ?: false %}
|
||||
|
||||
{% set page_list = grav.pages.getList(start_page, 0, true, show_all, show_fullpath, show_slug, show_modular, limit_levels) %}
|
||||
|
||||
<div class="form-select-wrapper {{ field.size }}">
|
||||
<select class="{{ field.classes }}" name="{{ (scope ~ field.name)|fieldName ~ (field.multiple ? '[]' : '') }}"
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.multiple in ['on', 'true', 1] %}multiple="multiple"{% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
>
|
||||
{% if field.show_root %}
|
||||
<option value="/">/ (root)</option>
|
||||
{% endif %}
|
||||
{{ macro.page_options(_context, page_list) }}
|
||||
</select>
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
@@ -0,0 +1,58 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
{#
|
||||
{% set defaults = {
|
||||
show_root: true,
|
||||
show_all: show_all_val,
|
||||
show_modular: show_modular_val,
|
||||
show_slug: show_slug_val,
|
||||
show_fullpath: show_fullpath_val,
|
||||
default: last_page_route,
|
||||
limit_levels: limit_levels_val
|
||||
} %}
|
||||
{% set field = field|merge(defaults) %}
|
||||
#}
|
||||
|
||||
{% set name = (scope ~ field.name)|fieldName %}
|
||||
{% if form %}
|
||||
{% set parent = form.object.parent.title %}
|
||||
{% elseif grav['flex_objects'].hasDirectory('pages') %}
|
||||
{% set directory = grav['flex_objects'].getDirectory('pages') %}
|
||||
{% set parent = value ? directory.getObject(value|trim('/')).title : '<root>' %}
|
||||
{% else %}
|
||||
{% do admin.enablePages() %}
|
||||
{% set parent = page.find(value).title %}
|
||||
{% endif %}
|
||||
<div class="parents-wrapper">
|
||||
<div class="form-input-wrapper" data-parents="{{ name }}" data-remodal-target="parents">
|
||||
<div class="parent-title" data-parents-field-name="{{ name }}">{{ parent }}</div>
|
||||
<span><i class="fa fa-folder-o"></i> <span data-parents-field-label="{{ name }}">{{ value|default('/') }}</span></span>
|
||||
</div>
|
||||
|
||||
<input type="hidden" class="input" name="{{ name }}" data-field-name="{{ field.name }}" value="{{ value|join(', ') }}" />
|
||||
</div>
|
||||
|
||||
{#{% set last_page_route = admin.page.getLastPageRoute %}
|
||||
{% set show_slug_val = true %}
|
||||
{% set show_fullpath_val = false %}
|
||||
{% set show_all_val = true %}
|
||||
|
||||
{% set show_parents = config.get('plugins.admin.pages.show_parents') %}
|
||||
{% if show_parents == 'folder' %}
|
||||
{% set show_slug_val = false %}
|
||||
{% elseif show_parents == 'fullpath' %}
|
||||
{% set show_fullpath_val = true %}
|
||||
{% endif %}
|
||||
|
||||
{% set limit_levels_val = config.get('plugins.admin.pages.parents_levels') %}
|
||||
|
||||
{% set show_modular_val = config.get('plugins.admin.pages.show_modular', true) %}
|
||||
{% if show_modular_val == false %}
|
||||
{% set show_all_val = false %}
|
||||
{% endif %}
|
||||
|
||||
{% set defaults = {show_root:true, show_all:show_all_val, show_modular:show_modular_val, show_slug:show_slug_val, show_fullpath:show_fullpath_val, default:last_page_route, limit_levels:limit_levels_val} %}
|
||||
{% set field = field|merge(defaults) %}
|
||||
{{ parent() }}#}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,161 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% macro spanToggle(input, length) %}
|
||||
{% set space = repeat(' ', (length - input|length) / 2) %}
|
||||
{{ (space ~ input ~ space)|raw }}
|
||||
{% endmacro %}
|
||||
|
||||
{% macro section(section, context, depth) %}
|
||||
{% import _self as macro %}
|
||||
|
||||
{% set section_label = (section.label ?? section.name)|t %}
|
||||
|
||||
{# Sub sections can have top-level toggle.. needs to #}
|
||||
{% if depth > 0 %}
|
||||
|
||||
{% set section_value = context.access.get(section.name)|string %}
|
||||
{% set params = {
|
||||
context: context,
|
||||
action_label: section_label,
|
||||
action_value: section_value,
|
||||
action_name: section.name,
|
||||
action_class: 'parent-section'
|
||||
}
|
||||
%}
|
||||
|
||||
{{ macro.action_row(params) }}
|
||||
{% endif %}
|
||||
|
||||
<fieldset>
|
||||
{% if depth == 0 %}
|
||||
<legend>{{ section_label }}</legend>
|
||||
{% endif %}
|
||||
|
||||
{% for action in section %}
|
||||
{% if action.visible %}
|
||||
{% if action.count %}
|
||||
{{ macro.section(action, context, depth + 1) }}
|
||||
{% else %}
|
||||
{{ macro.action(action, context) }}
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
</fieldset>
|
||||
|
||||
{% endmacro %}
|
||||
|
||||
{% macro action(action, context) %}
|
||||
{% import _self as macro %}
|
||||
|
||||
{% set action_label = action.label ?? action.name %}
|
||||
{% set action_value = context.access.get(action.name)|string %}
|
||||
|
||||
{% set params = {
|
||||
context: context,
|
||||
action_label: action_label,
|
||||
action_value: action_value,
|
||||
action_name: action.name,
|
||||
action_parent: action.getParent().name
|
||||
}
|
||||
%}
|
||||
|
||||
{{ macro.action_row(params) }}
|
||||
{% endmacro %}
|
||||
|
||||
{% macro action_row(data) %}
|
||||
{% import _self as macro %}
|
||||
|
||||
{% set context = data.context %}
|
||||
{% set field = context.field %}
|
||||
|
||||
<div class="permission-container {{ data.action_class }}">
|
||||
<div class="permission-name">
|
||||
<span>{{ data.action_label|t }}</span>
|
||||
{% if context.auth_badges %}
|
||||
{% set auth = context.object.authorize(data.action_name, 'test') ?? context.super %}
|
||||
{% if context.super and auth %}
|
||||
<span class="badge badge-super"><i class="icon-super"></i></span>
|
||||
{% elseif auth %}
|
||||
<span class="badge badge-access"><i class="fa fa-check"></i></span>
|
||||
{% else %}
|
||||
<span class="badge badge-denied"><i class="fa fa-ban"></i></span>
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
<div class="switch-toggle switch-grav medium switch-3">
|
||||
{% for key, text in context.options %}
|
||||
{% set parent_id = data.action_parent ? "toggle_" ~ field.name ~ "." ~ data.action_parent %}
|
||||
{% set id = "toggle_" ~ field.name ~ "." ~ data.action_name ~ key %}
|
||||
{% set translation = text|t|trim %}
|
||||
|
||||
<input type="radio"
|
||||
value="{{ key }}"
|
||||
id="{{ id }}"
|
||||
{% if parent_id %}
|
||||
data-parent-id="{{ parent_id }}"
|
||||
{% endif %}
|
||||
name="{{ (context.scope ~ field.name)|fieldName }}[{{ data.action_name }}]"
|
||||
class="label{{ key }}"
|
||||
{% if key|fieldName == '' ~ data.action_value|fieldName %}
|
||||
checked="checked"
|
||||
{% endif %}
|
||||
{% if field.validate.required in ['yes', 'on', 'true', 1, true] %}required="required"{% endif %}
|
||||
/>
|
||||
|
||||
<label for="{{ id }}">{{ macro.spanToggle(translation, context.maxLen)|trim|raw }}</label>
|
||||
{% endfor %}
|
||||
<a></a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{% endmacro %}
|
||||
|
||||
{% import _self as macro %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-disabled="{{ originalValue is null ? 'true' : 'false' }}"
|
||||
data-grav-default="{{ field.default|json_encode()|e('html_attr') }}"
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
{% set options = { '1': 'PLUGIN_ADMIN.ALLOWED', '0': 'PLUGIN_ADMIN.DENIED', '': 'PLUGIN_ADMIN.NOT_SET' } %}
|
||||
{% set maxLen = 0 %}
|
||||
{% for text in options %}
|
||||
{% set maxLen = max(text|t|trim|length, maxLen) %}
|
||||
{% endfor %}
|
||||
|
||||
{% set permissions = grav.permissions %}
|
||||
{% set access = permissions.access(value) %}
|
||||
{% if object and field.check_authorize %}
|
||||
{% set auth_badges = true %}
|
||||
{% set super = object.authorize('admin.super', 'test') %}
|
||||
{% endif %}
|
||||
|
||||
<div class="permissions-container">
|
||||
{% for section in permissions %}
|
||||
{% if section.count %}
|
||||
{{ macro.section(section, _context, 0) }}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
{# Look for missing actions #}
|
||||
{% set unknown %}
|
||||
{% for key,val in access.getAllActions() %}
|
||||
{% if not permissions.getAction(key) %}
|
||||
{{ macro.action({name: key}, _context) }}
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{% endset %}
|
||||
|
||||
{% if unknown|trim %}
|
||||
<fieldset>
|
||||
<legend>Unknown Permissions</legend>
|
||||
{{ unknown|raw }}
|
||||
</fieldset>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
@@ -0,0 +1,28 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input_attributes %}
|
||||
type="range"
|
||||
class="rangefield {% if field.classes is defined %}{{ field.classes }} {% endif %}"
|
||||
{% if field.validate.min %}min="{{ field.validate.min }}"{% endif %}
|
||||
{% if field.validate.max %}max="{{ field.validate.max }}"{% endif %}
|
||||
{% if field.validate.step %}step="{{ field.validate.step }}"{% endif %}
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
{% block append %}
|
||||
<input
|
||||
type="number"
|
||||
class="rangefield {% if field.classes is defined %}{{ field.classes }} {% endif %}"
|
||||
{% if field.validate.min %}min="{{ field.validate.min }}"{% endif %}
|
||||
{% if field.validate.max %}max="{{ field.validate.max }}"{% endif %}
|
||||
{% if field.validate.step %}step="{{ field.validate.step }}"{% endif %}
|
||||
{% if value is defined %}
|
||||
value="{{ value }}"
|
||||
{% elseif field.default is defined %}
|
||||
value="{{ field.default }}"
|
||||
{% else %}
|
||||
value="0"
|
||||
{% endif %}
|
||||
/>
|
||||
<span class="range-append">{{ field.append }}</span>
|
||||
|
||||
{% endblock append %}
|
||||
@@ -0,0 +1,24 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
{% if field.security is empty or authorize(array(field.security)) %}
|
||||
|
||||
{% if field.title or field.underline %}
|
||||
<h1 class="{{ field.classes }} {{ field.underline ?: 'no_underline' }}">{{ field.title|t }}</h1>
|
||||
{% endif %}
|
||||
|
||||
{% if field.text %}
|
||||
<p>{{ field.text|t|markdown|raw }}</p>
|
||||
{% endif %}
|
||||
|
||||
{% embed 'forms/default/fields.html.twig' with {name: field.name, fields: field.fields} %}
|
||||
{% block outer_markup_field_open %}
|
||||
<div class="form-section {{ field.field_classes }} {{ field.outer_classes }}">
|
||||
{% endblock %}
|
||||
{% block outer_markup_field_close %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
{% endembed %}
|
||||
|
||||
{% endif %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,19 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block global_attributes %}
|
||||
{% if field.selectize is defined %}
|
||||
{% set fieldSelectize = {create: field.selectize.create ?? true}|merge(field.selectize ?? {}) %}
|
||||
{% if field.merge_items %}
|
||||
{% set fieldSelectize = fieldSelectize|merge({'items':value}) %}
|
||||
{% endif %}
|
||||
{% else %}
|
||||
{% set fieldSelectize = {'create': true} %}
|
||||
{% endif %}
|
||||
data-grav-selectize="{{ fieldSelectize|json_encode()|e('html_attr') }}"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input_attributes %}
|
||||
type="text"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,43 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-selectize="{{ (field.selectize is defined ? field.selectize : {})|json_encode()|e('html_attr') }}"
|
||||
data-grav-field="select"
|
||||
{{ parent() }}
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-input-wrapper {{ field.size }}">
|
||||
{% set input_value = value is iterable ? value|join(',') : value|string %}
|
||||
<select
|
||||
value="{{ input_value }}"
|
||||
name="{{ (scope ~ field.name)|fieldName }}"
|
||||
data-select-observe
|
||||
{% block input_attributes %}
|
||||
{% if field.classes is defined %}class="{{ field.classes }}" {% endif %}
|
||||
{% if field.id is defined %}id="{{ field.id }}" {% endif %}
|
||||
{% if field.style is defined %}style="{{ field.style }}" {% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}
|
||||
{% if field.placeholder %}placeholder="{{ field.placeholder }}"{% endif %}
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.readonly in ['on', 'true', 1] %}readonly="readonly"{% endif %}
|
||||
{% if field.autocomplete in ['on', 'off'] %}autocomplete="{{ field.autocomplete }}"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.validate.pattern %}pattern="{{ field.validate.pattern }}"{% endif %}
|
||||
{% if field.validate.message %}title="{{ field.validate.message|t }}"
|
||||
{% elseif field.title is defined %}title="{{ field.title|t }}" {% endif %}
|
||||
{% endblock %}
|
||||
>
|
||||
{% if field.placeholder %}<option value="" disabled selected>{{ field.placeholder|t|raw }}</option>{% endif %}
|
||||
|
||||
{% if value is defined %}
|
||||
<option selected="selected" value="{{ value }}">{{ value }}</option>
|
||||
{% endif %}
|
||||
</select>
|
||||
|
||||
{% if field.default %}
|
||||
<div><p class="notice warning">It is advised not to use a <strong>default</strong> option with the selectunique field. Currently default value set to `{{ field.default }}`</p></div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,39 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
{% set object = form.object %}
|
||||
{% set taxonomies = taxonomies ?? field.taxonomies ?? admin.data('config/site').taxonomies ?? [] %}
|
||||
{% set parentname = field.name %}
|
||||
{% set options = field.options %}
|
||||
{% set default = field.default %}
|
||||
|
||||
{% for name in taxonomies %}
|
||||
{% set field_name = parentname ~ '.' ~ name %}
|
||||
{% set value = form ? form.value(field_name) : data.value(field_name) %}
|
||||
{% set value = (value ?? default[name] ?? [])|array %}
|
||||
{% if object %}
|
||||
{% set can_translate = can_translate ?? (admin.multilang and object.hasFlexFeature('flex-translate')) %}
|
||||
{% set index = object.getFlexDirectory().getIndex() %}
|
||||
{% set translated = can_translate ? index.withTranslated(admin.language) : index %}
|
||||
{% set sub_taxonomies = translated.getDistinctValues(field_name) %}
|
||||
{% else %}
|
||||
{% set sub_taxonomies = (attribute(grav.taxonomy.taxonomy, name) ?? [])|keys %}
|
||||
{% endif %}
|
||||
{% set list = (options[name] ?? [])|merge(sub_taxonomies)|merge(value)|array_unique %}
|
||||
|
||||
{% set field = {
|
||||
type: 'select',
|
||||
classes: 'fancy create',
|
||||
label: name|capitalize,
|
||||
name: field_name,
|
||||
multiple: true,
|
||||
options: list,
|
||||
style: field.style,
|
||||
selectize: {
|
||||
create: true
|
||||
}
|
||||
} %}
|
||||
|
||||
{% include 'forms/fields/select/select.html.twig' %}
|
||||
{% endfor %}
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,106 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block input %}
|
||||
<div class="jquery-horizontal-scroll-wrap">
|
||||
<div class="jquery-horizontal-scroll">
|
||||
<div class="holder" style="white-space: nowrap;">
|
||||
{% for id,preset in whitelabel_presets %}
|
||||
<style>
|
||||
.pid-{{ id }} .admin-preview {
|
||||
background: {{ preset.colors['page-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-sidebar {
|
||||
background: {{ preset.colors['nav-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-logo {
|
||||
background: {{ preset.colors['logo-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-logo::after {
|
||||
background: {{ preset.colors['logo-link'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-text, .pid-{{ id }} .ap-text::before, .pid-{{ id }} .ap-text::after {
|
||||
background: {{ preset.colors['nav-link'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-active {
|
||||
background: {{ preset.colors['nav-selected-bg'] }};
|
||||
border-left-color: {{ preset.colors['button-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-active::after {
|
||||
background: {{ preset.colors['nav-selected-link'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-toolbar {
|
||||
background: {{ preset.colors['toolbar-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-toolbar::after {
|
||||
background: {{ preset.colors['toolbar-text'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-button {
|
||||
background: {{ preset.colors['button-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-content {
|
||||
background: {{ preset.colors['content-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-title {
|
||||
background: {{ preset.colors['content-header'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-content .ap-text {
|
||||
background: {{ preset.colors['content-link'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-content .ap-text::before, .pid-{{ id }} .ap-content .ap-text::after {
|
||||
background: {{ preset.colors['content-text'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-update {
|
||||
background: {{ preset.colors['update-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-update::after {
|
||||
background: {{ preset.colors['update-text'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-notice {
|
||||
background: {{ preset.colors['notice-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-notice::after {
|
||||
background: {{ preset.colors['notice-text'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-critical {
|
||||
background: {{ preset.colors['critical-bg'] }};
|
||||
}
|
||||
.pid-{{ id }} .ap-critical::after {
|
||||
background: {{ preset.colors['critical-text'] }};
|
||||
}
|
||||
</style>
|
||||
<div class="pid-{{ id }} admin-preview-wrapper" data-preset-values="{{ preset|json_encode|e('html_attr') }}">
|
||||
<div class="admin-preview form-border">
|
||||
<div class="ap-overlay"><b><i class="fa fa-upload"></i> {{ "PLUGIN_ADMIN.LOAD_PRESET"|t }}</b></div>
|
||||
<div class="ap-sidebar">
|
||||
<div class="ap-logo"></div>
|
||||
<div class="ap-nav">
|
||||
<span class="ap-text"></span>
|
||||
<span class="ap-active"></span>
|
||||
<span class="ap-text"></span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="ap-toolbar">
|
||||
<span class="ap-button"></span>
|
||||
</div>
|
||||
<div class="ap-page">
|
||||
<div class="ap-content">
|
||||
<span class="ap-update"></span>
|
||||
<span class="ap-title"></span>
|
||||
<span class="ap-text"></span>
|
||||
<span class="ap-notice"></span>
|
||||
<span class="ap-text"></span>
|
||||
<span class="ap-critical"></span>
|
||||
<span class="ap-text"></span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="admin-preview-title">
|
||||
{{ preset.name }}
|
||||
</div>
|
||||
<input type="hidden" value="{{ preset|json_encode }}" />
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,31 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% set options = {} %}
|
||||
{% for slug, package in admin.themes %}
|
||||
{% set option = {(slug): package.toArray().name} %}
|
||||
{% set options = options|merge(option) %}
|
||||
{% endfor %}
|
||||
|
||||
{% block global_attributes %}
|
||||
data-grav-selectize="{{ (field.selectize is defined ? field.selectize : {})|json_encode()|e('html_attr') }}"
|
||||
data-grav-field="select"
|
||||
data-grav-disabled="{{ originalValue is null ? 'true' : 'false' }}"
|
||||
data-grav-default="{{ field.default|json_encode()|e('html_attr') }}"
|
||||
{% endblock %}
|
||||
|
||||
{% block input %}
|
||||
<div class="form-select-wrapper {{ field.size }}">
|
||||
<select class="{{ field.classes }}" name="{{ (scope ~ field.name)|fieldName ~ (field.multiple ? '[]' : '') }}"
|
||||
{% if field.autofocus in ['on', 'true', 1] %}autofocus="autofocus"{% endif %}
|
||||
{% if field.novalidate in ['on', 'true', 1] %}novalidate="novalidate"{% endif %}
|
||||
{% if field.validate.required in ['on', 'true', 1] %}required="required"{% endif %}
|
||||
{% if field.multiple in ['on', 'true', 1] %}multiple="multiple"{% endif %}
|
||||
{% if field.disabled or isDisabledToggleable %}disabled="disabled"{% endif %}>
|
||||
{% for key, text in options %}
|
||||
<option {% if key == value or text in value %}selected="selected"{% endif %} value="{{ field.multiple ? text : key }}">{{ text }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
@@ -0,0 +1,19 @@
|
||||
{% extends "forms/field.html.twig" %}
|
||||
|
||||
{% block field %}
|
||||
<div class="form-field grid user-details">
|
||||
<div class="form-label block size-1-3">
|
||||
{% include 'partials/userinfo-avatar.html.twig' %}
|
||||
</div>
|
||||
<div class="form-data block size-2-3">
|
||||
<h2>{{ data.fullname }}</h2>
|
||||
<h5>
|
||||
<a href="mailto:{{ data.email }}">{{ data.email }}</a>
|
||||
{% if data.title %}
|
||||
- {{ data.title }}
|
||||
{% endif %}
|
||||
</h5>
|
||||
{% include 'partials/userinfo-avatar-credit.html.twig' %}
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
@@ -0,0 +1,61 @@
|
||||
{#{% extends "forms/field.html.twig" %}#}
|
||||
|
||||
{% import _self as macro %}
|
||||
|
||||
{% macro spanToggle(input, length) %}
|
||||
{% set space = repeat(' ', (length - input|length) / 2) %}
|
||||
{{ (space ~ input ~ space)|raw }}
|
||||
{% endmacro %}
|
||||
|
||||
{% macro outputWidget(widget) %}
|
||||
|
||||
{% endmacro %}
|
||||
|
||||
{% set widgets = grav.twig.plugins_hooked_dashboard_widgets_top|array|merge(grav.twig.plugins_hooked_dashboard_widgets_main|array)|merge(grav.twig.plugins_hooked_dashboard_widgets_bottom|array) %}
|
||||
{% set widgets_display = grav.config.plugins.admin.widgets_display %}
|
||||
|
||||
{% for widget in widgets %}
|
||||
{% set widget_name = widget.name|default(widget.template|titleize) %}
|
||||
{% set widget_id = widget.template %}
|
||||
{% set widget_value = widgets_display[widget_id]|string in ['1', 'true'] ? 'true' : 'false' %}
|
||||
{% set name = 'data['~field.name~']['~widget_id~']' %}
|
||||
|
||||
<div class="block block-toggle">
|
||||
<div class="form-field grid">
|
||||
<div class="form-label block size-1-3">
|
||||
<label>
|
||||
<span>{{ widget_name }} Widget</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="form-data block size-2-3" data-grav-field="toggle" data-grav-default="1">
|
||||
<div class="switch-toggle switch-grav medium switch-2">
|
||||
|
||||
{% set options = { true: 'PLUGIN_ADMIN.ENABLED', false: 'PLUGIN_ADMIN.DISABLED' } %}
|
||||
|
||||
{% set maxLen = 0 %}
|
||||
{% for value, text in options %}
|
||||
{% set translation = text|t %}
|
||||
{% set maxLen = max(translation|length, maxLen) %}
|
||||
{% endfor %}
|
||||
|
||||
{% for key, text in options %}
|
||||
{% set id = "toggle_" ~ field.name ~ "." ~ widget_id ~ '.' ~ key %}
|
||||
{% set translation = text|t|trim %}
|
||||
|
||||
<input type="radio"
|
||||
value="{{ key }}"
|
||||
id="{{ id }}"
|
||||
name="{{ name }}"
|
||||
class="{{ 'true' == '' ~ key ? 'highlight' : '' }}"
|
||||
{% if key|string == widget_value|string %}
|
||||
checked="checked"
|
||||
{% endif %}
|
||||
/>
|
||||
|
||||
<label for="{{ id }}">{{ (macro.spanToggle(translation, maxLen)|trim)|raw }}</label>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endfor %}
|
||||
@@ -0,0 +1,6 @@
|
||||
{% set xss_header = (form ? form.value('header') : data.value('header'))|array %}
|
||||
{% set xss_content = form ? form.value('content') : data.value('content') %}
|
||||
{% set xss_status = xss({header: xss_header, content: xss_content}) %}
|
||||
{% if xss_status is not empty %}
|
||||
<div class="notice alert">{{ "PLUGIN_ADMIN.XSS_ISSUE"|t(xss_status)|raw }}</div>
|
||||
{% endif %}
|
||||
Reference in New Issue
Block a user