`, then you need to escape that in your form definition:
diff --git a/plugins/form/assets/form-styles.css b/plugins/form/assets/form-styles.css
index fc0083e..1099c9c 100644
--- a/plugins/form/assets/form-styles.css
+++ b/plugins/form/assets/form-styles.css
@@ -1,80 +1 @@
-.form-group.has-errors { background: rgba(255, 0, 0, 0.05); border: 1px solid rgba(255, 0, 0, 0.2); border-radius: 3px; margin: 0 -5px; padding: 0 5px; }
-
-.form-errors { color: #b52b27; }
-
-.form-honeybear { visibility: hidden; position: absolute !important; height: 1px; width: 1px; overflow: hidden; clip: rect(1px, 1px, 1px, 1px); }
-
-.form-errors p { margin: 0; }
-
-.form-input-file input { display: none; }
-
-.form-input-file .dz-default.dz-message { position: absolute; text-align: center; left: 0; right: 0; top: 50%; -webkit-transform: translateY(-50%); transform: translateY(-50%); margin: 0; }
-
-.form-input-file.dropzone { position: relative; min-height: 70px; border-radius: 3px; margin-bottom: .85rem; border: 2px dashed #ccc; color: #aaa; padding: 0.5rem; }
-
-.form-input-file.dropzone .dz-preview { margin: 0.5rem; }
-
-.form-input-file.dropzone .dz-preview:hover { z-index: 2; }
-
-.form-input-file.dropzone .dz-preview .dz-error-message { min-width: 140px; width: auto; }
-
-.form-input-file.dropzone .dz-preview .dz-image, .form-input-file.dropzone .dz-preview.dz-file-preview .dz-image { border-radius: 3px; z-index: 1; }
-
-.form-tabs .tabs-nav { display: flex; padding-top: 1px; margin-bottom: -1px; }
-
-.form-tabs .tabs-nav a { flex: 1; transition: color 0.5s ease, background 0.5s ease; cursor: pointer; text-align: center; padding: 10px; display: flex; align-items: center; justify-content: center; border-bottom: 1px solid #eee; border-radius: 5px 5px 0 0; }
-
-.form-tabs .tabs-nav a.active { border: 1px solid #eee; border-bottom: 1px solid transparent; margin: 0 -1px; }
-
-.form-tabs .tabs-nav a.active span { color: #000; }
-
-.form-tabs .tabs-nav span { display: inline-block; line-height: 1.1; }
-
-.form-tabs.subtle .tabs-nav { margin-right: 0 !important; }
-
-.form-tabs .tabs-content .tab__content { display: none; padding-top: 2rem; }
-
-.form-tabs .tabs-content .tab__content.active { display: block; }
-
-.checkboxes { display: inline-block; }
-
-.checkboxes label { display: inline; cursor: pointer; position: relative; padding: 0 0 0 20px; margin-right: 15px; }
-
-.checkboxes label:before { content: ""; display: inline-block; width: 20px; height: 20px; left: 0; margin-top: 0; margin-right: 10px; position: absolute; border-radius: 3px; border: 1px solid #e6e6e6; }
-
-.checkboxes input[type=checkbox] { display: none; }
-
-.checkboxes input[type=checkbox]:checked + label:before { content: "\2713"; font-size: 20px; line-height: 1; text-align: center; }
-
-.checkboxes.toggleable label { margin-right: 0; }
-
-.form-field-toggleable .checkboxes.toggleable { margin-right: 5px; vertical-align: middle; }
-
-.form-field-toggleable .checkboxes + label { display: inline-block; }
-
-.switch-toggle { display: inline-flex; overflow: hidden; border-radius: 3px; line-height: 35px; border: 1px solid #eee; }
-
-.switch-toggle input[type=radio] { position: absolute; visibility: hidden; display: none; }
-
-.switch-toggle label { display: inline-block; cursor: pointer; padding: 0 15px; margin: 0; white-space: nowrap; color: inherit; transition: background-color 0.5s ease; }
-
-.switch-toggle input.highlight:checked + label { background: #333; color: #fff; }
-
-.switch-toggle input:checked + label { color: #fff; background: #999; }
-
-/* Signature Pad */
-.signature-pad { position: relative; display: flex; flex-direction: column; font-size: 10px; width: 100%; height: 100%; max-width: 700px; max-height: 460px; border: 1px solid #f0f0f0; background-color: #fff; padding: 16px; }
-
-.signature-pad--body { position: relative; flex: 1; border: 1px solid #f6f6f6; min-height: 100px; }
-
-.signature-pad--body canvas { position: absolute; left: 0; top: 0; width: 100%; height: 100%; border-radius: 4px; box-shadow: 0 0 5px rgba(0, 0, 0, 0.02) inset; }
-
-.signature-pad--footer { color: #C3C3C3; text-align: center; font-size: 1.2em; }
-
-.signature-pad--actions { display: flex; justify-content: space-between; margin-top: 8px; }
-
-[data-grav-field="array"] .form-row { display: flex; align-items: center; margin-bottom: 0.5rem; }
-
-[data-grav-field="array"] .form-row > input, [data-grav-field="array"] .form-row > textarea { margin: 0 0.5rem; display: inline-block; }
-
-/*# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"file":"form-styles.css","sources":["form-styles.scss"],"sourcesContent":["$form-border-color: #eee;\n$form-active-color: #000;\n\n.form-group.has-errors {\n    background: rgba(255,0,0,0.05);\n    border: 1px solid rgba(255,0,0,0.2);\n    border-radius: 3px;\n    margin: 0 -5px;\n    padding: 0 5px;\n}\n\n.form-errors {\n    color: #b52b27;\n}\n\n.form-honeybear {\n    visibility: hidden;\n    position: absolute !important;\n    height: 1px;\n    width: 1px;\n    overflow: hidden;\n    clip: rect(1px, 1px, 1px, 1px);\n}\n\n.form-errors p {\n    margin: 0;\n}\n\n.form-input-file {\n\n    input {\n        display: none;\n    }\n\n    .dz-default.dz-message {\n        position: absolute;\n        text-align: center;\n        left: 0;\n        right: 0;\n        top: 50%;\n        transform: translateY(-50%);\n        margin: 0;\n    }\n\n    &.dropzone {\n        position: relative;\n        min-height: 70px;\n        border-radius: 3px;\n        margin-bottom: .85rem;\n        border: 2px dashed #ccc;\n        color: #aaa;\n        padding: 0.5rem;\n\n        .dz-preview {\n            margin: 0.5rem;\n\n            &:hover {\n                z-index: 2;\n            }\n\n            .dz-error-message {\n                min-width: 140px;\n                width: auto;\n            }\n\n            .dz-image,\n            &.dz-file-preview .dz-image {\n                border-radius: 3px;\n                z-index: 1;\n            }\n        }\n    }\n}\n\n\n\n// New JS powered tabs\n.form-tabs {\n\n    .tabs-nav {\n        display: flex;\n        padding-top: 1px;\n\n        margin-bottom: -1px;\n\n        a {\n            flex: 1;\n            transition: color 0.5s ease, background 0.5s ease;\n            cursor: pointer;\n            text-align:center;\n            padding: 10px;\n            display: flex;\n            align-items: center;\n            justify-content: center;\n            border-bottom: 1px solid $form-border-color;\n            border-radius: 5px 5px 0 0;\n\n            &.active {\n                border: 1px solid $form-border-color;\n                border-bottom: 1px solid transparent;\n                margin: 0 -1px;\n\n                span {\n                    color: $form-active-color;\n                }\n            }\n        }\n\n        span {\n            display: inline-block;\n            line-height: 1.1;\n        }\n\n    }\n\n    &.subtle .tabs-nav {\n        margin-right: 0 !important;\n    }\n\n    .tabs-content {\n\n        .tab__content {\n            display: none;\n            padding-top: 2rem;\n\n            &.active {\n                display: block;\n            }\n        }\n    }\n}\n\n// Checkboxes\n.checkboxes {\n    display: inline-block;\n\n    label {\n        display: inline;\n        cursor: pointer;\n        position: relative;\n        padding: 0 0 0 20px;\n        margin-right: 15px;\n\n    }\n    label:before {\n        content:\"\";\n        display: inline-block;\n        width: 20px;\n        height: 20px;\n        left: 0;\n        margin-top: 0;\n        margin-right: 10px;\n        position: absolute;\n        border-radius: 3px;\n\n        border: 1px solid #e6e6e6;\n\n    }\n    input[type=checkbox] {\n        display: none;\n    }\n    input[type=checkbox]:checked + label:before {\n        content:\"\\2713\";\n        font-size: 20px;\n        line-height: 1;\n        text-align: center;\n    }\n\n    &.toggleable label{\n        margin-right: 0;\n    }\n}\n\n// Toggleable\n.form-field-toggleable {\n    .checkboxes.toggleable {\n        margin-right: 5px;\n        vertical-align: middle;\n    }\n    .checkboxes + label {\n        display: inline-block;\n    }\n}\n\n// Toggles\n.switch-toggle {\n    display: inline-flex;\n    overflow: hidden;\n    border-radius: 3px;\n    line-height: 35px;\n    border: 1px solid $form-border-color;\n\n    input[type=radio] {\n        position: absolute;\n        visibility: hidden;\n        display: none;\n    }\n\n    label {\n        display: inline-block;\n        cursor: pointer;\n        padding: 0 15px;\n        margin: 0;\n        white-space: nowrap;\n        color: inherit;\n        transition: background-color 0.5s ease;\n    }\n\n    input.highlight:checked + label {\n        background: #333;\n        color: #fff;\n    }\n\n    input:checked + label {\n        color: #fff;\n        background: #999;\n    }\n\n\n}\n\n/* Signature Pad */\n.signature-pad {\n    position: relative;\n    display: -webkit-box;\n    display: -ms-flexbox;\n    display: flex;\n    -webkit-box-orient: vertical;\n    -webkit-box-direction: normal;\n    -ms-flex-direction: column;\n    flex-direction: column;\n    font-size: 10px;\n    width: 100%;\n    height: 100%;\n    max-width: 700px;\n    max-height: 460px;\n    border: 1px solid #f0f0f0;\n    background-color: #fff;\n    padding: 16px;\n}\n\n.signature-pad--body {\n    position: relative;\n    -webkit-box-flex: 1;\n    -ms-flex: 1;\n    flex: 1;\n    border: 1px solid #f6f6f6;\n    min-height: 100px;\n}\n\n.signature-pad--body canvas {\n    position: absolute;\n    left: 0;\n    top: 0;\n    width: 100%;\n    height: 100%;\n    border-radius: 4px;\n    box-shadow: 0 0 5px rgba(0, 0, 0, 0.02) inset;\n}\n\n.signature-pad--footer {\n    color: #C3C3C3;\n    text-align: center;\n    font-size: 1.2em;\n}\n\n.signature-pad--actions {\n    display: -webkit-box;\n    display: -ms-flexbox;\n    display: flex;\n    -webkit-box-pack: justify;\n    -ms-flex-pack: justify;\n    justify-content: space-between;\n    margin-top: 8px;\n}\n\n[data-grav-field=\"array\"] .form-row {\n    display: flex;\n    align-items: center;\n    margin-bottom: 0.5rem;\n}\n\n[data-grav-field=\"array\"] .form-row > input,\n[data-grav-field=\"array\"] .form-row > textarea\n{\n    margin: 0 0.5rem;\n    display: inline-block;\n}\n\n"],"names":[],"mappings":"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"} */
+.form-group.has-errors{background:rgba(255,0,0,.05);border:1px solid rgba(255,0,0,.2);border-radius:3px;margin:0 -5px;padding:0 5px}.form-errors{color:#b52b27}.form-honeybear{visibility:hidden;position:absolute !important;height:1px;width:1px;overflow:hidden;clip:rect(1px, 1px, 1px, 1px)}.form-errors p{margin:0}.form-input-file input{display:none}.form-input-file .dz-default.dz-message{position:absolute;text-align:center;left:0;right:0;top:50%;transform:translateY(-50%);margin:0}.form-input-file.dropzone{position:relative;min-height:70px;border-radius:3px;margin-bottom:.85rem;border:2px dashed #ccc;color:#aaa;padding:.5rem}.form-input-file.dropzone .dz-preview{margin:.5rem}.form-input-file.dropzone .dz-preview:hover{z-index:2}.form-input-file.dropzone .dz-preview .dz-error-message{min-width:140px;width:auto}.form-input-file.dropzone .dz-preview .dz-image,.form-input-file.dropzone .dz-preview.dz-file-preview .dz-image{border-radius:3px;z-index:1}.form-tabs .tabs-nav{display:flex;padding-top:1px;margin-bottom:-1px}.form-tabs .tabs-nav a{flex:1;transition:color .5s ease,background .5s ease;cursor:pointer;text-align:center;padding:10px;display:flex;align-items:center;justify-content:center;border-bottom:1px solid #ccc;border-radius:5px 5px 0 0}.form-tabs .tabs-nav a.active{border:1px solid #ccc;border-bottom:1px solid rgba(0,0,0,0);margin:0 -1px}.form-tabs .tabs-nav a.active span{color:#000}.form-tabs .tabs-nav span{display:inline-block;line-height:1.1}.form-tabs.subtle .tabs-nav{margin-right:0 !important}.form-tabs .tabs-content .tab__content{display:none;padding-top:2rem}.form-tabs .tabs-content .tab__content.active{display:block}.checkboxes{display:inline-block}.checkboxes label{display:inline;cursor:pointer;position:relative;padding:0 0 0 20px;margin-right:15px}.checkboxes label:before{content:"";display:inline-block;width:20px;height:20px;left:0;margin-top:0;margin-right:10px;position:absolute;border-radius:3px;border:1px solid #e6e6e6}.checkboxes input[type=checkbox]{display:none}.checkboxes input[type=checkbox]:checked+label:before{content:"✓";font-size:20px;line-height:1;text-align:center}.checkboxes.toggleable label{margin-right:0}.form-field-toggleable .checkboxes.toggleable{margin-right:5px;vertical-align:middle}.form-field-toggleable .checkboxes+label{display:inline-block}.switch-toggle{display:inline-flex;overflow:hidden;border-radius:3px;line-height:35px;border:1px solid #ccc}.switch-toggle input[type=radio]{position:absolute;visibility:hidden;display:none}.switch-toggle label{display:inline-block;cursor:pointer;padding:0 15px;margin:0;white-space:nowrap;color:inherit;transition:background-color .5s ease}.switch-toggle input.highlight:checked+label{background:#333;color:#fff}.switch-toggle input:checked+label{color:#fff;background:#999}.signature-pad{position:relative;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;font-size:10px;width:100%;height:100%;max-width:700px;max-height:460px;border:1px solid #f0f0f0;background-color:#fff;padding:16px}.signature-pad--body{position:relative;-webkit-box-flex:1;-ms-flex:1;flex:1;border:1px solid #f6f6f6;min-height:100px}.signature-pad--body canvas{position:absolute;left:0;top:0;width:100%;height:100%;border-radius:4px;box-shadow:0 0 5px rgba(0,0,0,.02) inset}.signature-pad--footer{color:#c3c3c3;text-align:center;font-size:1.2em}.signature-pad--actions{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-pack:justify;-ms-flex-pack:justify;justify-content:space-between;margin-top:8px}[data-grav-field=array] .form-row{display:flex;align-items:center;margin-bottom:.5rem}[data-grav-field=array] .form-row>input,[data-grav-field=array] .form-row>textarea{margin:0 .5rem;display:inline-block}.form-data.basic-captcha .form-input-wrapper{border:1px solid #ccc;border-radius:5px;display:flex;overflow:hidden}.form-data.basic-captcha .form-input-prepend{display:flex;color:#333;background-color:#ccc;flex-shrink:0}.form-data.basic-captcha .form-input-prepend img{margin:0}.form-data.basic-captcha .form-input-prepend button>svg{margin:0 8px;width:18px;height:18px}.form-data.basic-captcha input.form-input{border:0}/*# sourceMappingURL=form-styles.css.map */
diff --git a/plugins/form/blueprints.yaml b/plugins/form/blueprints.yaml
index 4ba82a4..84f95c2 100644
--- a/plugins/form/blueprints.yaml
+++ b/plugins/form/blueprints.yaml
@@ -1,8 +1,8 @@
name: Form
slug: form
type: plugin
-version: 5.1.6
-description: Enables the forms handling
+version: 7.1.2
+description: Enables forms handling and processing
icon: check-square
author:
name: Team Grav
@@ -196,3 +196,150 @@ form:
label: PLUGIN_FORM.RECAPTCHA_SECRET_KEY
help: PLUGIN_FORM.RECAPTCHA_SECRET_KEY_HELP
default: ''
+
+ turnstile_captcha:
+ type: section
+ title: PLUGIN_FORM.TURNSTILE_CAPTCHA
+
+ fields:
+ turnstile.theme:
+ type: select
+ label: PLUGIN_FORM.RECAPTCHA_THEME
+ default: light
+ options:
+ light: PLUGIN_FORM.RECAPTCHA_THEME_LIGHT
+ dark: PLUGIN_FORM.RECAPTCHA_THEME_DARK
+ turnstile.site_key:
+ type: text
+ label: PLUGIN_FORM.RECAPTCHA_SITE_KEY
+ help: PLUGIN_FORM.RECAPTCHA_SITE_KEY_HELP
+ default: ''
+ turnstile.secret_key:
+ type: text
+ label: PLUGIN_FORM.RECAPTCHA_SECRET_KEY
+ help: PLUGIN_FORM.RECAPTCHA_SECRET_KEY_HELP
+ default: ''
+
+ basic_captcha:
+ type: section
+ title: PLUGIN_FORM.BASIC_CAPTCHA
+
+ fields:
+ basic_captcha.type:
+ type: elements
+ label: PLUGIN_FORM.BASIC_CAPTCHA_TYPE
+ default: 'characters'
+ size: medium
+ options:
+ characters: Random Characters
+ math: Math Puzzle
+ fields:
+ characters:
+ type: element
+ fields:
+ basic_captcha.chars.length:
+ type: range
+ label: PLUGIN_FORM.BASIC_CAPTCHA_LENGTH
+ default: 6
+ validate:
+ min: 4
+ max: 12
+ append: characters
+ basic_captcha.chars.font:
+ type: select
+ label: PLUGIN_FORM.BASIC_CAPTCHA_FONT
+ default: zxx-noise.ttf
+ options:
+ 'zxx-noise.ttf': zxx-Noise
+ 'zxx-xed.ttf': zxx-Xed
+ 'zxx-camo.ttf': zxx-Camo
+ 'zxx-sans.ttf': zxx-Sans
+ basic_captcha.chars.size:
+ type: range
+ label: PLUGIN_FORM.BASIC_CAPTCHA_SIZE
+ default: 24
+ append: px
+ validate:
+ min: 12
+ max: 32
+ step: 2
+ basic_captcha.chars.bg:
+ type: colorpicker
+ size: small
+ label: PLUGIN_FORM.BASIC_CAPTCHA_BG_COLOR
+ default: '#ffffff'
+ basic_captcha.chars.text:
+ type: colorpicker
+ size: small
+ label: PLUGIN_FORM.BASIC_CAPTCHA_TEXT_COLOR
+ default: '#000000'
+ basic_captcha.chars.start_x:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_START_X
+ default: 5
+ append: px
+ size: small
+ validate:
+ min: 0
+ type: number
+ basic_captcha.chars.start_y:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_START_Y
+ default: 30
+ append: px
+ size: small
+ validate:
+ min: 0
+ type: number
+ basic_captcha.chars.box_width:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_BOX_WIDTH
+ default: 135
+ append: px
+ size: small
+ validate:
+ min: 0
+ type: number
+ basic_captcha.chars.box_height:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_BOX_HEIGHT
+ default: 40
+ append: px
+ size: small
+ validate:
+ min: 0
+ type: number
+ math:
+ type: element
+ fields:
+ basic_captcha.math.min:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_MATH_MIN
+ default: 1
+ size: small
+ validate:
+ min: 0
+ type: number
+ basic_captcha.math.max:
+ type: number
+ label: PLUGIN_FORM.BASIC_CAPTCHA_MATH_MAX
+ default: 10
+ size: small
+ validate:
+ min: 1
+ type: number
+ basic_captcha.math.operators:
+ type: selectize
+ selectize:
+ options:
+ - value: '+'
+ text: '+ Addition'
+ - value: '-'
+ text: '- Subtraction'
+ - value: '*'
+ text: 'x Multiplication'
+ - value: '/'
+ text: '/ Division'
+ label: PLUGIN_FORM.BASIC_CAPTCHA_MATH_OPERATORS
+ validate:
+ type: commalist
diff --git a/plugins/form/classes/BasicCaptcha.php b/plugins/form/classes/BasicCaptcha.php
index 4f5a169..61bc10b 100644
--- a/plugins/form/classes/BasicCaptcha.php
+++ b/plugins/form/classes/BasicCaptcha.php
@@ -32,20 +32,20 @@ class BasicCaptcha
// calculator
if ($operator === '-') {
if ($first_num < $second_num) {
- $result = "$second_num-$first_num";
- $captcha_code = $second_num-$first_num;
+ $result = "$second_num - $first_num";
+ $captcha_code = $second_num - $first_num;
} else {
$result = "$first_num-$second_num";
$captcha_code = $first_num - $second_num;
}
} elseif ($operator === '*') {
- $result = "{$first_num}x{$second_num}";
- $captcha_code = $first_num - $second_num;
+ $result = "{$first_num} x {$second_num}";
+ $captcha_code = $first_num * $second_num;
} elseif ($operator === '/') {
- $result = "$first_num/ second_num";
+ $result = "$first_num / second_num";
$captcha_code = $first_num / $second_num;
} elseif ($operator === '+') {
- $result = "$first_num+$second_num";
+ $result = "$first_num + $second_num";
$captcha_code = $first_num + $second_num;
}
} else {
diff --git a/plugins/form/classes/Form.php b/plugins/form/classes/Form.php
index 87a3c22..a02c559 100644
--- a/plugins/form/classes/Form.php
+++ b/plugins/form/classes/Form.php
@@ -119,7 +119,7 @@ class Form implements FormInterface, ArrayAccess
$this->items = $form;
} else {
// Otherwise get all forms in the page.
- $forms = $page->forms();
+ $forms = $page->getForms();
if ($name) {
// If form with given name was found, use that.
$this->items = $forms[$name] ?? [];
diff --git a/plugins/form/classes/Forms.php b/plugins/form/classes/Forms.php
index f71bd84..8294f5c 100644
--- a/plugins/form/classes/Forms.php
+++ b/plugins/form/classes/Forms.php
@@ -113,7 +113,7 @@ class Forms
*/
protected function getPageParameters(PageInterface $page, ?string $name): array
{
- $forms = $page->forms();
+ $forms = $page->getForms();
if ($name) {
// If form with given name was found, use that.
diff --git a/plugins/form/classes/TwigExtension.php b/plugins/form/classes/TwigExtension.php
index 7074255..41d417b 100644
--- a/plugins/form/classes/TwigExtension.php
+++ b/plugins/form/classes/TwigExtension.php
@@ -90,21 +90,28 @@ class TwigExtension extends AbstractExtension
return null;
}
+ // If field has already been prepared, we do not need to do anything.
+ if (!empty($field['prepared'])) {
+ return $field;
+ }
+
// Check if we have just a list of fields (no name given).
- if (is_int($name)) {
+ $fieldName = (string)($field['name'] ?? $name);
+ if (!is_string($name) || $name === '') {
// Look at the field.name and if not set, fall back to the key.
- $name = (string)($field['name'] ?? $name);
+ $name = $fieldName;
}
// Make sure that the field has a name.
- $name = $name ?? $field['name'] ?? null;
- if (!is_string($name) || $name === '') {
+ if ($name === '') {
return null;
}
// Prefix name with the parent name if needed.
if (str_starts_with($name, '.')) {
- $name = $parent ? $parent . $name : (string)substr($name, 1);
+ $plainName = (string)substr($name, 1);
+ $field['plain_name'] = $plainName;
+ $name = $parent ? $parent . $name : $plainName;
} elseif (isset($options['key'])) {
$name = str_replace('*', $options['key'], $name);
}
@@ -125,6 +132,7 @@ class TwigExtension extends AbstractExtension
// Always set field name.
$field['name'] = $name;
+ $field['prepared'] = true;
return $field;
}
diff --git a/plugins/form/composer.lock b/plugins/form/composer.lock
index 23aa529..5ddb549 100644
--- a/plugins/form/composer.lock
+++ b/plugins/form/composer.lock
@@ -73,5 +73,5 @@
"platform-overrides": {
"php": "7.3.6"
},
- "plugin-api-version": "2.0.0"
+ "plugin-api-version": "2.2.0"
}
diff --git a/plugins/form/form.php b/plugins/form/form.php
index 5f8625b..eb19d62 100644
--- a/plugins/form/form.php
+++ b/plugins/form/form.php
@@ -4,6 +4,7 @@ namespace Grav\Plugin;
use Composer\Autoload\ClassLoader;
use DateTime;
+use Doctrine\Common\Cache\Cache;
use Exception;
use Grav\Common\Data\ValidationException;
use Grav\Common\Debugger;
@@ -18,10 +19,13 @@ use Grav\Common\Utils;
use Grav\Common\Uri;
use Grav\Common\Yaml;
use Grav\Framework\Form\Interfaces\FormInterface;
+use Grav\Framework\Psr7\Response;
use Grav\Framework\Route\Route;
+use Grav\Plugin\Form\BasicCaptcha;
use Grav\Plugin\Form\Form;
use Grav\Plugin\Form\Forms;
use Grav\Plugin\Form\TwigExtension;
+use Grav\Common\HTTP\Client;
use ReCaptcha\ReCaptcha;
use ReCaptcha\RequestMethod\CurlPost;
use RecursiveArrayIterator;
@@ -31,6 +35,7 @@ use RocketTheme\Toolbox\File\YamlFile;
use RocketTheme\Toolbox\File\File;
use RocketTheme\Toolbox\Event\Event;
use RuntimeException;
+use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface;
use Twig\Environment;
use Twig\Extension\CoreExtension;
use Twig\Extension\EscaperExtension;
@@ -54,11 +59,9 @@ class FormPlugin extends Plugin
/** @var Form */
protected $form;
- /** @var array */
+ /** @var array[]|FormInterface[] */
protected $forms = [];
- /** @var array */
- protected $flat_forms = [];
- /** @var array */
+ /** @var FormInterface[] */
protected $active_forms = [];
/** @var array */
protected $json_response = [];
@@ -70,7 +73,7 @@ class FormPlugin extends Plugin
*/
public static function checkRequirements(): bool
{
- return version_compare(GRAV_VERSION, '1.6', '>');
+ return version_compare(GRAV_VERSION, '1.7', '>');
}
/**
@@ -83,18 +86,13 @@ class FormPlugin extends Plugin
}
return [
- 'onPluginsInitialized' => [
- ['autoload', 100000],
- ['onPluginsInitialized', 0]
- ],
+ 'onPluginsInitialized' => ['onPluginsInitialized', 0],
'onTwigExtensions' => ['onTwigExtensions', 0],
'onTwigTemplatePaths' => ['onTwigTemplatePaths', 0]
];
}
/**
- * [onPluginsInitialized:100000] Composer autoload.
- *
* @return ClassLoader
*/
public function autoload()
@@ -114,10 +112,9 @@ class FormPlugin extends Plugin
$this->grav['forms'] = function () {
$forms = new Forms();
-
- $grav = Grav::instance();
$event = new Event(['forms' => $forms]);
- $grav->fireEvent('onFormRegisterTypes', $event);
+
+ $this->grav->fireEvent('onFormRegisterTypes', $event);
return $forms;
};
@@ -130,12 +127,19 @@ class FormPlugin extends Plugin
return;
}
+ /** @var Uri $uri */
+ $uri = $this->grav['uri'];
+
// Mini Keep-Alive Logic
- $task = $this->grav['uri']->param('task');
- if ($task && $task === 'keep-alive') {
- exit;
+ $task = $uri->param('task');
+ if ($task === 'keep-alive') {
+ $response = new Response(200);
+
+ $this->grav->close($response);
}
+ $this->processBasicCaptchaImage($uri);
+
$this->enable([
'onPageProcessed' => ['onPageProcessed', 0],
'onPagesInitialized' => ['onPagesInitialized', 0],
@@ -161,16 +165,16 @@ class FormPlugin extends Plugin
/**
* Process forms after page header processing, but before caching
*
- * @param Event $e
+ * @param Event $event
* @return void
*/
- public function onPageProcessed(Event $e): void
+ public function onPageProcessed(Event $event): void
{
/** @var PageInterface $page */
- $page = $e['page'];
+ $page = $event['page'];
- $pageForms = $page->forms();
- if (!$pageForms) {
+ $forms = $page->getForms();
+ if (!$forms) {
return;
}
@@ -184,23 +188,18 @@ class FormPlugin extends Plugin
}
$parent = $current && $current !== $page ? $current : null;
- $page_route = $page->home() ? '/' : $page->route();
-
// If the form was in the modular page, we need to add the form into the parent page as well.
if ($parent) {
- $parent->addForms($pageForms);
- $parent_route = $parent->home() ? '/' : $parent->route();
+ $parent->addForms($forms);
}
- /** @var Forms $forms */
- $forms = $this->grav['forms'];
-
// Store the page forms in the forms instance
- foreach ($pageForms as $name => $form) {
- if (isset($parent, $parent_route)) {
- $this->addForm($parent_route, $forms->createPageForm($parent, $name, $form));
+ foreach ($forms as $name => $form) {
+ if ($parent) {
+ $this->addFormDefinition($parent, $name, $form);
}
- $this->addForm($page_route, $forms->createPageForm($page, $name, $form));
+
+ $this->addFormDefinition($page, $name, $form);
}
}
@@ -257,7 +256,7 @@ class FormPlugin extends Plugin
if ($form instanceof Form) {
// Post the form
$isJson = $uri->extension() === 'json';
- $task = $uri->post('task') ?? $uri->param('task');
+ $task = (string)($uri->post('task') ?? $uri->param('task'));
if ($isJson) {
if ($task === 'store-state') {
@@ -286,10 +285,13 @@ class FormPlugin extends Plugin
if ($this->json_response && $page->template() !== 'form') {
$status = $this->json_response['status'] ?? null;
- header('Content-Type: application/json');
- http_response_code($status === 'error' ? 400 : 200);
- echo json_encode($this->json_response);
- exit;
+ $response = new Response(
+ $status !== 'error' ? 200 : 400,
+ ['Content-Type' => 'application/json'],
+ json_encode($this->json_response, JSON_THROW_ON_ERROR)
+ );
+
+ $this->grav->close($response);
}
}
@@ -313,11 +315,25 @@ class FormPlugin extends Plugin
// There is no active form to be posted.
// Check all the forms for the current page; we are looking for forms with remember state turned on with random unique id.
+ /** @var Forms $forms */
+ $forms = $this->grav['forms'];
+
/** @var Route $route */
$route = $this->grav['route'];
$pageForms = $this->forms[$route->getRoute()] ?? [];
- foreach ($pageForms as $formName => $form) {
+ /**
+ * @var string $name
+ * @var array|FormInterface $form
+ */
+ foreach ($pageForms as $name => $form) {
+ if (is_array($form)) {
+ $form = $this->createForm($page, $name, $form);
+ }
+ if (!$form instanceof FormInterface) {
+ continue;
+ }
+
if ($form->get('remember_redirect')) {
// Found one; we need to check if unique id is set.
$formParam = $form->get('uniqueid_param', 'fid');
@@ -328,8 +344,6 @@ class FormPlugin extends Plugin
$form->setUniqueId($uniqueId);
$form->initialize();
- /** @var Forms $forms */
- $forms = $this->grav['forms'];
$forms->setActiveForm($form);
break;
@@ -337,7 +351,7 @@ class FormPlugin extends Plugin
// Append unique id to the URL and redirect.
$route = $route->withGravParam($formParam, $form->getUniqueId());
- $page->redirect((string)$route->toString());
+ $page->redirect($route->toString());
// TODO: Do we want to add support for multiple forms with remembered state?
break;
@@ -398,7 +412,7 @@ class FormPlugin extends Plugin
/**
* Make form accessible from twig.
*
- * @param Event $event
+ * @param Event|null $event
* @return void
*/
public function onTwigVariables(Event $event = null): void
@@ -428,6 +442,7 @@ class FormPlugin extends Plugin
* @param Event $event
* @return void
* @throws Exception
+ * @throws TransportExceptionInterface
*/
public function onFormProcessed(Event $event): void
{
@@ -499,6 +514,55 @@ class FormPlugin extends Plugin
return;
}
+ break;
+ case 'basic-captcha':
+ $captcha = new BasicCaptcha();
+ $captcha_value = trim($form->value('basic-captcha'));
+ if (!$captcha->validateCaptcha($captcha_value)) {
+ $message = $params['message'] ?? $this->grav['language']->translate('PLUGIN_FORM.ERROR_BASIC_CAPTCHA');
+
+ $this->grav->fireEvent('onFormValidationError', new Event([
+ 'form' => $form,
+ 'message' => $message
+ ]));
+
+ $event->stopPropagation();
+ return;
+ }
+ break;
+ case 'turnstile':
+ /** @var Uri $uri */
+ $uri = $this->grav['uri'];
+
+ $turnstile_config = $this->config->get('plugins.form.turnstile');
+ $secret = $turnstile_config['secret_key'] ?? null;
+ $token = $form->getValue('cf-turnstile-response') ?? null;
+ $ip = Uri::ip();
+
+ $client = Client::getClient();
+ $response = $client->request('POST', 'https://challenges.cloudflare.com/turnstile/v0/siteverify', [
+ 'body' => [
+ 'secret' => $secret,
+ 'response' => $token,
+ 'remoteip' => $ip
+ ]
+ ]);
+
+ $content = $response->toArray();
+
+ if (!$content['success']) {
+ $message = $params['message'] ?? $this->grav['language']->translate('PLUGIN_FORM.ERROR_BASIC_CAPTCHA');
+
+ $this->grav->fireEvent('onFormValidationError', new Event([
+ 'form' => $form,
+ 'message' => $message
+ ]));
+
+ $this->grav['log']->addWarning('Form Turnstile invalid: [' . $uri->route() . '] ' . json_encode($content));
+ $event->stopPropagation();
+ return;
+ }
+
break;
case 'timestamp':
$label = $params['label'] ?? 'Timestamp';
@@ -544,8 +608,7 @@ class FormPlugin extends Plugin
$this->grav['messages']->add($form->message, 'success');
}
- $event['redirect'] = $url;
- $event->stopPropagation();
+ $this->grav->redirect($url);
break;
case 'reset':
if (Utils::isPositive($params)) {
@@ -736,6 +799,7 @@ class FormPlugin extends Plugin
*/
public function onFormValidationError(Event $event): void
{
+ /** @var FormInterface $form */
$form = $event['form'];
if (isset($event['message'])) {
$form->status = 'error';
@@ -743,6 +807,7 @@ class FormPlugin extends Plugin
$form->messages = $event['messages'];
}
+ /** @var Uri $uri */
$uri = $this->grav['uri'];
$route = $uri->route();
@@ -752,8 +817,7 @@ class FormPlugin extends Plugin
/** @var Pages $pages */
$pages = $this->grav['pages'];
- $page = $pages->dispatch($route, true);
-
+ $page = $pages->find($route, true);
if ($page) {
unset($this->grav['page']);
$this->grav['page'] = $page;
@@ -762,14 +826,32 @@ class FormPlugin extends Plugin
$event->stopPropagation();
}
+ /**
+ * Add a form definition to the forms plugin
+ *
+ * @param PageInterface $page
+ * @return void
+ */
+ public function addFormDefinition(PageInterface $page, string $name, array $form): void
+ {
+ $route = ($page->home() ? '/' : $page->route()) ?? '/';
+
+ if (!isset($this->forms[$route][$name])) {
+ $form['_page_routable'] = !$page->isModule();
+
+ $this->forms[$route][$name] = $form;
+ $this->recache_forms = true;
+ }
+ }
+
/**
* Add a form to the forms plugin
*
- * @param string|null $page_route
+ * @param string|null $route
* @param FormInterface|null $form
* @return void
*/
- public function addForm(?string $page_route, ?FormInterface $form)
+ public function addForm(?string $route, ?FormInterface $form): void
{
if (null === $form) {
return;
@@ -777,10 +859,10 @@ class FormPlugin extends Plugin
$name = $form->getName();
- if (!isset($this->forms[$page_route][$name])) {
- $this->forms[$page_route][$name] = $form;
+ if (!isset($this->forms[$route][$name])) {
+ $form['_page_routable'] = true;
- $this->flattenForms();
+ $this->forms[$route][$name] = $form;
$this->recache_forms = true;
}
}
@@ -788,45 +870,128 @@ class FormPlugin extends Plugin
/**
* function to get a specific form
*
- * @param null|array|string $data optional form `name`
+ * @param string|array|null $data Optional form name or ['name' => $name, 'route' => $route]
* @return FormInterface|null
*/
- public function getForm($data = null)
+ public function getForm($data = null): ?FormInterface
{
+ /** @var Pages $pages */
+ $pages = $this->grav['pages'];
+
+ // Handle parameters.
if (is_array($data)) {
- $form_name = $data['name'] ?? null;
- $page_route = $data['route'] ?? null;
+ $name = (string)($data['name'] ?? '');
+ $route = (string)($data['route'] ?? '');
} elseif (is_string($data)) {
- $form_name = $data;
- $page_route = null;
+ $name = $data;
+ $route = '';
} else {
- $form_name = null;
- $page_route = null;
+ $name = '';
+ $route = '';
}
- // if no form name, use the first form found in the page
- if (!$form_name) {
- // If page route not provided, use the current page
- if (!$page_route) {
- // Get page route with a fallback using current URI if page not initialized yet
- $page_route = $this->grav['page']->route() ?: $this->getCurrentPageRoute();
- }
-
- if (!empty($this->forms[$page_route])) {
- $forms = $this->forms[$page_route];
- $first_form = reset($forms) ?: null;
- return $first_form;
- }
-
- // Try to get page by defined route first or get current if not found
- $page = $this->grav['pages']->find($page_route) ?: $this->grav['page'];
-
- // Try looking up in the defined page
- return $this->grav['forms']->createPageForm($page);
+ // Return always the same form instance.
+ $form = $this->active_forms[$route][$name] ?? null;
+ if ($form) {
+ return $form;
}
- // return the form you are looking for if available
- return $this->getFormByName($form_name);
+ $unnamed = $name === '';
+ $routed = $route !== '';
+
+ // Get the page.
+ if ($routed) {
+ // Use fixed route for the form.
+ $route_provided = true;
+
+ $page = $pages->find($route, true);
+ } else {
+ // Search form from the current page first.
+ $route_provided = false;
+
+ /** @var PageInterface|null $page */
+ $page = $this->grav['page'] ?? null;
+ if ($page) {
+ $route = $page->route();
+ } else {
+ // Get page route with a fallback using current URI if page is not yet initialized.
+ $route = $this->getCurrentPageRoute();
+ $page = $pages->find($route);
+ }
+ }
+
+ // Attempt to find the form from the page.
+ if ('' !== $route) {
+ $forms = $this->forms[$route] ?? [];
+
+ if (!$unnamed) {
+ // Get form by the name.
+ $form = $forms[$name] ?? null;
+ } else {
+ // Get the first form.
+ $form = reset($forms) ?: null;
+ $name = key($forms);
+ }
+ }
+
+ // Search the form from the other pages.
+ if (null === $form) {
+ // First check if we requested a specific form which didn't exist.
+ if ($route_provided || $unnamed) {
+ /** @var Debugger $debugger */
+ $debugger = $this->grav['debugger'];
+ $debugger->addMessage(sprintf('Form %s not found in page %s', $name ?? 'unnamed', $route), 'warning');
+
+ return null;
+ }
+
+ // Attempt to find any form with given name.
+ $forms = $this->findFormByName($name);
+ $first = reset($forms);
+ if (!$first) {
+ return null;
+ }
+
+ // Check for naming conflicts.
+ if (count($forms) > 1) {
+ $debugger = $this->grav['debugger'];
+ $debugger->addMessage(sprintf('Fetching form by its name, but there are multiple pages with the same form name %s', $name), 'warning');
+ }
+
+ [$route, $name, $form] = $first;
+
+ $page = $pages->find($route);
+ }
+
+ // Form can be saved as an array or an object. If it's an array, we need to create object from it.
+ if (is_array($form)) {
+ // Form was cached as an array, try to create the object.
+ if (null === $page) {
+ /** @var Debugger $debugger */
+ $debugger = $this->grav['debugger'];
+ $debugger->addMessage(sprintf('Form %s cannot be created as page %s does not exist', $name, $route), 'warning');
+
+ return null;
+ }
+
+ $form = $this->createForm($page, $name, $form);
+ }
+
+ // Register form to the active forms to get the same instance back next time.
+ $this->active_forms[$route][$name] = $form;
+ if ($unnamed) {
+ $this->active_forms[$route][''] = $form;
+ }
+
+ // Also make aliases if route was not provided to the method.
+ if (!$routed) {
+ $this->active_forms[''][$name] = $form;
+ if ($unnamed) {
+ $this->active_forms[''][''] = $form;
+ }
+ }
+
+ return $form;
}
/**
@@ -834,7 +999,7 @@ class FormPlugin extends Plugin
*
* @return array
*/
- public function getFormFieldTypes()
+ public function getFormFieldTypes(): array
{
return [
'avatar' => [
@@ -906,7 +1071,7 @@ class FormPlugin extends Plugin
*
* - fillWithCurrentDateTime
*
- * @param Form $form
+ * @param FormInterface $form
* @return void
*/
protected function process($form)
@@ -921,43 +1086,33 @@ class FormPlugin extends Plugin
/**
* Get current page's route
*
- * @return mixed
+ * @return string
*/
protected function getCurrentPageRoute()
{
$path = $this->grav['uri']->route();
- $path = $path ?: '/';
- return $path;
+
+ return $path ?: '/';
}
/**
- * Retrieve a form based on the form name
+ * Return all forms matching the given name.
*
- * @param string $form_name
- * @param string $unique_id
- * @return mixed
+ * @param string $name
+ * @return array
*/
- protected function getFormByName($form_name, $unique_id = '')
+ protected function findFormByName(string $name): array
{
- $form = $this->active_forms[$form_name] ?? null;
- if (!$form) {
- $form = $this->flat_forms[$form_name] ?? null;
-
- if (!$form) {
- return null;
+ $list = [];
+ foreach ($this->forms as $route => $forms) {
+ foreach ($forms as $key => $form) {
+ if ($name === $key && !empty($form['_page_routable'])) {
+ $list[] = [$route, $key, $form];
+ }
}
-
- if ('' === $unique_id) {
- // Reset form to change the cached unique id and to fire onFormInitialized event.
- $form->setUniqueId('');
- $form->reset();
- }
-
- // Register form to the active forms to get the same instance back next time.
- $this->active_forms[$form_name] = $form;
}
- return $form;
+ return $list;
}
/**
@@ -965,12 +1120,11 @@ class FormPlugin extends Plugin
*
* @return bool
*/
- protected function shouldProcessForm()
+ protected function shouldProcessForm(): bool
{
+ /** @var Uri $uri */
$uri = $this->grav['uri'];
- $nonce = $uri->post('form-nonce');
- $status = $nonce ? true : false; // php72 quirk?
- $refresh_prevention = null;
+ $status = (bool)$uri->post('form-nonce');
if ($status && $form = $this->form()) {
// Make sure form is something we recognize.
@@ -1009,29 +1163,14 @@ class FormPlugin extends Plugin
return $status;
}
- /**
- * Flatten the forms array into something that can be more easily searched
- *
- * @return void
- */
- protected function flattenForms()
- {
- $this->flat_forms = Utils::arrayFlatten($this->forms);
- }
-
/**
* Get the current form, should already be processed but can get it directly from the page if necessary
*
* @param PageInterface|null $page
- * @return Form|null
+ * @return FormInterface|null
*/
protected function form(PageInterface $page = null)
{
- // Regenerate list of flat_forms if not already populated
- if (empty($this->flat_forms)) {
- $this->flattenForms();
- }
-
/** @var Forms $forms */
$forms = $this->grav['forms'];
@@ -1050,11 +1189,16 @@ class FormPlugin extends Plugin
$form_name = $page ? $page->slug() : null;
}
- $form = $this->getFormByName($form_name, $unique_id);
+ $form = $form_name ? $this->getForm($form_name) : null;
+ if ($form && '' === $unique_id) {
+ // Reset form to change the cached unique id and to fire onFormInitialized event.
+ $form->setUniqueId('');
+ $form->reset();
+ }
// last attempt using current page's form
if (!$form && $page) {
- $form = $forms->createPageForm($page);
+ $form = $this->createForm($page);
}
if ($form) {
@@ -1073,20 +1217,16 @@ class FormPlugin extends Plugin
/**
* @param PageInterface $page
- * @param string|int|null $name
- * @param array $form
- * @return Form|null
- * @deprecated
+ * @param string|null $name
+ * @param array|null $form
+ * @return FormInterface|null
*/
- protected function createForm(PageInterface $page, $name = null, $form = null)
+ protected function createForm(PageInterface $page, string $name = null, array $form = null): ?FormInterface
{
+ /** @var Forms $forms */
+ $forms = $this->grav['forms'];
- $header = $page->header();
- if (isset($header->form) || isset($header->forms)) {
- return new Form($page, $name, $form);
- }
-
- return null;
+ return $forms->createPageForm($page, $name, $form);
}
/**
@@ -1094,18 +1234,20 @@ class FormPlugin extends Plugin
*
* @return void
*/
- protected function loadCachedForms()
+ protected function loadCachedForms(): void
{
// Get and set the cache of forms if it exists
try {
- [$forms] = $this->grav['cache']->fetch($this->getFormCacheId());
- } catch (Exception $e) {
- // Couldn't fetch cached forms.
- $forms = null;
+ /** @var Cache $cache */
+ $cache = $this->grav['cache'];
+ [$forms] = $cache->fetch($this->getFormCacheId());
+ } catch (Exception $e) {
/** @var Debugger $debugger */
$debugger = Grav::instance()['debugger'];
$debugger->addMessage(sprintf('Unserializing cached forms failed: %s', $e->getMessage()), 'error');
+
+ $forms = null;
}
if (!is_array($forms)) {
@@ -1113,9 +1255,8 @@ class FormPlugin extends Plugin
}
// Only update the forms if it's not empty
- if (!empty($forms)) {
+ if ($forms) {
$this->forms = array_merge($this->forms, $forms);
- $this->flattenForms();
}
}
@@ -1124,13 +1265,19 @@ class FormPlugin extends Plugin
*
* @return void
*/
- protected function saveCachedForms()
+ protected function saveCachedForms(): void
{
// Save the current state of the forms to cache
- if ($this->recache_forms) {
- $this->recache_forms = false;
- $this->grav['cache']->save($this->getFormCacheId(), [$this->forms]);
+ if (!$this->recache_forms) {
+ return;
}
+
+ $this->recache_forms = false;
+
+ /** @var Cache $cache */
+ $cache = $this->grav['cache'];
+
+ $cache->save($this->getFormCacheId(), [$this->forms]);
}
/**
@@ -1138,9 +1285,12 @@ class FormPlugin extends Plugin
*
* @return string
*/
- protected function getFormCacheId()
+ protected function getFormCacheId(): string
{
- return $this->grav['pages']->getPagesCacheId() . '-form-plugin';
+ /** @var Pages $pages */
+ $pages = $this->grav['pages'];
+
+ return $pages->getPagesCacheId() . '-form-plugin';
}
/**
@@ -1152,16 +1302,25 @@ class FormPlugin extends Plugin
*/
protected function udate($format = 'u', $raw = false)
{
-
- $utimestamp = microtime(true);
-
if ($raw) {
return date($format);
}
+ $utimestamp = microtime(true);
$timestamp = floor($utimestamp);
$milliseconds = round(($utimestamp - $timestamp) * 1000000);
return date(preg_replace('`(?path() === '/forms-basic-captcha-image.jpg') {
+ $captcha = new BasicCaptcha();
+ $code = $captcha->getCaptchaCode();
+ $image = $captcha->createCaptchaImage($code);
+ $captcha->renderCaptchaImage($image);
+ exit;
+ }
+ }
}
diff --git a/plugins/form/form.yaml b/plugins/form/form.yaml
index 01bf085..29793ca 100644
--- a/plugins/form/form.yaml
+++ b/plugins/form/form.yaml
@@ -17,4 +17,25 @@ recaptcha:
version: 2-checkbox
theme: light
site_key:
- secret_key:
\ No newline at end of file
+ secret_key:
+turnstile:
+ theme: light # options: [light | dark]
+ site_key:
+ secret_key:
+
+basic_captcha:
+ type: characters # options: [characters | math]
+ chars:
+ length: 6 # number of chars to output
+ font: zxx-noise.ttf # options: [zxx-noise.ttf | zxx-camo.ttf | zxx-xed.ttf | zxx-sans.ttf]
+ bg: '#cccccc' # 6-char hex color
+ text: '#333333' # 6-char hex color
+ size: 24 # font size in px
+ start_x: 5 # start position in x direction in px
+ start_y: 30 # start position in y direction in px
+ box_width: 135 # box width in px
+ box_height: 40 # box height in px
+ math:
+ min: 1 # smallest digit
+ max: 12 # largest digit
+ operators: ['+','-','*'] # operators that can be used in math
diff --git a/plugins/form/languages.yaml b/plugins/form/languages.yaml
index 4b2f06d..c6a288f 100644
--- a/plugins/form/languages.yaml
+++ b/plugins/form/languages.yaml
@@ -10,6 +10,7 @@ en:
DESTINATION_HELP: "The location where the files should be uploaded to"
ACCEPT: "Allowed MIME Types"
ACCEPT_HELP: "A list of MIME Types that are allowed for upload"
+ ERROR_BASIC_CAPTCHA: "Captcha failed for this form, please try again"
ERROR_VALIDATING_CAPTCHA: "reCAPTCHA bot protection has identified this form submission is problematic"
DATA_SUMMARY: "Here is the summary of what you wrote to us:"
NO_FORM_DATA: "No form data available"
@@ -72,6 +73,22 @@ en:
DESTINATION_NOT_SPECIFIED: "Destination not specified"
INVALID_MIME_TYPE: "The MIME type %s for the file %s is not accepted."
INVALID_FILE_EXTENSION: "The File Extension for the file %s is not accepted."
+ BASIC_CAPTCHA: "Basic Captcha"
+ BASIC_CAPTCHA_TYPE: "Captcha challenge type"
+ BASIC_CAPTCHA_LENGTH: "Number of characters"
+ BASIC_CAPTCHA_FONT: "TTF Font"
+ BASIC_CAPTCHA_SIZE: "Font size"
+ BASIC_CAPTCHA_BG_COLOR: "Background color"
+ BASIC_CAPTCHA_TEXT_COLOR: "Text color"
+ BASIC_CAPTCHA_START_X: "Text start x-position"
+ BASIC_CAPTCHA_START_Y: "Text start y-position"
+ BASIC_CAPTCHA_BOX_WIDTH: "Image width"
+ BASIC_CAPTCHA_BOX_HEIGHT: "Image height"
+ BASIC_CAPTCHA_MATH_MIN: "Minimum number"
+ BASIC_CAPTCHA_MATH_MAX: "Maximum number"
+ BASIC_CAPTCHA_MATH_OPERATORS: "Mathematical operators (randomized)"
+ TURNSTILE_CAPTCHA: "Cloudflare Turnstile Captcha"
+
eu:
PLUGIN_FORM:
NOT_VALIDATED: "Formularioa ez da baliozkotu. Beharrezkoa den eremu bat edo gehiago falta dira."
diff --git a/plugins/form/scss/form-styles.scss b/plugins/form/scss/form-styles.scss
index bd26070..8682bad 100644
--- a/plugins/form/scss/form-styles.scss
+++ b/plugins/form/scss/form-styles.scss
@@ -1,4 +1,4 @@
-$form-border-color: #eee;
+$form-border-color: #ccc;
$form-active-color: #000;
.form-group.has-errors {
@@ -287,3 +287,29 @@ $form-active-color: #000;
display: inline-block;
}
+.form-data.basic-captcha {
+ .form-input-wrapper {
+ border: 1px solid $form-border-color;
+ border-radius: 5px;
+ display: flex;
+ overflow: hidden;
+ }
+ .form-input-prepend {
+ display: flex;
+ color: #333;
+ background-color: #ccc;
+ flex-shrink: 0;
+ img {
+ margin: 0;
+ }
+ button > svg {
+ margin: 0 8px;
+ width: 18px;
+ height: 18px;
+ }
+ }
+ input.form-input {
+ border: 0;
+ }
+}
+
diff --git a/plugins/form/templates/forms/default/field.html.twig b/plugins/form/templates/forms/default/field.html.twig
index 9edfe94..1510441 100644
--- a/plugins/form/templates/forms/default/field.html.twig
+++ b/plugins/form/templates/forms/default/field.html.twig
@@ -10,7 +10,7 @@
{% set default = field.default %}
{% set toggleable = field.toggleable ?? false %}
{% if toggleable %}
- {% set originalValue = originalValue is defined ? originalValue : value %}
+ {% set originalValue = originalValue ?? value %}
{% set toggleableChecked = originalValue is not null %}
{% elseif field.overridable %}
{% set toggleable = true %}
diff --git a/plugins/form/templates/forms/default/fields.html.twig b/plugins/form/templates/forms/default/fields.html.twig
index 6ce3e1a..dd015dc 100644
--- a/plugins/form/templates/forms/default/fields.html.twig
+++ b/plugins/form/templates/forms/default/fields.html.twig
@@ -1,10 +1,10 @@
{% set fields = prepare_form_fields(fields, name) %}
+{% set originalValue = null %}
{% if fields|length %}
{% block outer_markup_field_open %}{% endblock %}
{% for field_name, field in fields %}
{% set value = form ? form.value(field.name) : data.value(field.name) %}
{% set field_templates = include_form_field(field.type, field_layout, fallback_field ?? 'text') %}
-
{% block inner_markup_field_open %}{% endblock %}
{% block field %}
{% include field_templates %}
diff --git a/plugins/form/templates/forms/default/form.html.twig b/plugins/form/templates/forms/default/form.html.twig
index cf0fb17..57ed7b6 100644
--- a/plugins/form/templates/forms/default/form.html.twig
+++ b/plugins/form/templates/forms/default/form.html.twig
@@ -145,6 +145,7 @@
{% if form.isEnabled() ?? true %}
{% for button in form.buttons %}
+ {% if not button.access or authorize(button.access) %}
{% if button.outerclasses is defined %}{% endif %}
{% if button.url %}
@@ -190,6 +191,7 @@
{% endembed %}
{% if button.outerclasses is defined %}
{% endif %}
+ {% endif %}
{% endfor %}
{% endif %}
diff --git a/plugins/form/templates/forms/fields/basic-captcha/basic-captcha.html.twig b/plugins/form/templates/forms/fields/basic-captcha/basic-captcha.html.twig
index e7dd890..68d9c7a 100644
--- a/plugins/form/templates/forms/fields/basic-captcha/basic-captcha.html.twig
+++ b/plugins/form/templates/forms/fields/basic-captcha/basic-captcha.html.twig
@@ -5,7 +5,7 @@
{% block prepend %}