Completed
Push — master ( 4d5898...2a5654 )
by Gabor
03:30
created

AbstractForm::setNodes()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 6
Code Lines 3

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 2

Importance

Changes 0
Metric Value
dl 0
loc 6
ccs 0
cts 3
cp 0
rs 9.4285
c 0
b 0
f 0
cc 1
eloc 3
nc 1
nop 1
crap 2
1
<?php
2
/**
3
 * WebHemi.
4
 *
5
 * PHP version 5.6
6
 *
7
 * @copyright 2012 - 2016 Gixx-web (http://www.gixx-web.com)
8
 * @license   https://opensource.org/licenses/MIT The MIT License (MIT)
9
 *
10
 * @link      http://www.gixx-web.com
11
 */
12
namespace WebHemi\Form;
13
14
use Iterator;
15
use WebHemi\Form\Element\NestedElementInterface;
16
use WebHemi\Form\Element\Traits\IteratorTrait;
17
18
/**
19
 * Class AbstractForm
20
 */
21
abstract class AbstractForm implements FormInterface, Iterator
22
{
23
    /** @var NestedElementInterface */
24
    protected $form;
25
    /** @var string */
26
    protected $name;
27
    /** @var string */
28
    protected $salt;
29
30
    // The implementation of the Iterator interface.
31
    use IteratorTrait;
32
33
    /**
34
     * AbstractForm constructor.
35
     *
36
     * @param string $name
37
     * @param string $action
38
     * @param string $method
39
     */
40 5
    final public function __construct($name, $action = '', $method = 'POST')
41
    {
42 5
        $this->form = $this->getFormContainer();
43 5
        $this->form->setName($name)
44 5
            ->setAttributes(
45
                [
46 5
                    'action' => $action,
47 5
                    'method' => strtoupper($method)
48
                ]
49
            );
50
51
        // For simplicity in rendering (twig macro), we store it in an array.
52 5
        $this->nodes[0] =& $this->form;
53
        // Set a default salt for the form name. If the AutoComplete attribute is 'off', it will be added to the form's
54
        // name attribute. The default salt will change every hour.
55 5
        $this->salt = md5(gmdate('YmdH'));
56
57 5
        $this->initForm();
58 5
    }
59
60
    /**
61
     * Returns the form container element. E.g.: for HTML forms it is the <form> tag.
62
     *
63
     * @return NestedElementInterface
64
     */
65
    abstract protected function getFormContainer();
66
67
    /**
68
     * Initialize form.
69
     *
70
     * @return void
71
     */
72
    abstract protected function initForm();
73
74
    /**
75
     * Gets form name.
76
     *
77
     * @return string
78
     */
79 2
    public function getName()
80
    {
81 2
        return $this->form->getName();
82
    }
83
84
    /**
85
     * Set unique identifier for the form.
86
     *
87
     * @param string $salt
88
     * @return AbstractForm
89
     */
90 2
    public function setNameSalt($salt)
91
    {
92 2
        $this->salt = md5($salt);
93
94 2
        return $this;
95
    }
96
97
    /**
98
     * Sets form action.
99
     *
100
     * @param string $action
101
     * @return AbstractForm
102
     */
103 1
    final public function setAction($action)
104
    {
105 1
        $this->setAttribute('action', $action);
106
107 1
        return $this;
108
    }
109
110
    /**
111
     * Sets form submit.
112
     *
113
     * @param string $method
114
     * @return AbstractForm
115
     */
116 1
    final public function setMethod($method = 'POST')
117
    {
118 1
        $this->setAttribute('method', $method);
119
120 1
        return $this;
121
    }
122
123
    /**
124
     * Sets form encoding type.
125
     *
126
     * @param string $encodingType
127
     * @return AbstractForm
128
     */
129 1
    final protected function setEnctype($encodingType = 'application/x-www-form-urlencoded')
130
    {
131 1
        $this->setAttribute('enctype', $encodingType);
132
133 1
        return $this;
134
    }
135
136
    /**
137
     * Sets form auto-complete option.
138
     *
139
     * @param bool $autoComplete
140
     * @return AbstractForm
141
     */
142 1
    final public function setAutoComplete($autoComplete = true)
143
    {
144 1
        $name = $this->form->getName(false);
145 1
        $md5Match = [];
146
147
        // Search for the unique form prefix.
148 1
        preg_match('/^[a-z0-9\_\-\[\]]+\_(?P<md5>[a-f0-9]{32}).*$/', $name, $md5Match);
149
150
        // When it's necessary, add/remove the salt to/from the name
151 1
        if ($autoComplete && !empty($md5Match)) {
152 1
            $name = str_replace('_'.$md5Match['md5'], '', $name);
153 1
        } elseif (!$autoComplete && empty($md5Match)) {
154 1
            $name = $name.'_'.$this->salt;
155
        }
156
157 1
        $this->form->setName($name);
158
159 1
        $this->setAttribute('autocomplete', $autoComplete);
160
161 1
        return $this;
162
    }
163
164
    /**
165
     * Sets specific form attributes.
166
     *
167
     * @param $name
168
     * @param $value
169
     */
170 2
    private function setAttribute($name, $value)
171
    {
172 2
        $attributes = $this->form->getAttributes();
173 2
        $attributes[$name] = $value;
174
175 2
        $this->form->setAttributes($attributes);
176 2
    }
177
178
    /**
179
     * Adds a form element to the form.
180
     *
181
     * @param array<FormElementInterface> $nodes
182
     * @return AbstractForm
183
     */
184
    protected function setNodes(array $nodes)
185
    {
186
        $this->form->setNodes($nodes);
187
188
        return $this;
189
    }
190
191
    /**
192
     * Gets the form elements.
193
     *
194
     * @return array<FormElementInterface>
195
     */
196
    public function getNodes()
197
    {
198
        return $this->form->getNodes();
199
    }
200
201
    /**
202
     * Validates the form.
203
     *
204
     * @return boolean
205
     */
206 1
    public function isValid()
207
    {
208 1
        return $this->form->isValid();
209
    }
210
211
    /**
212
     * Sets form data.
213
     *
214
     * @param array $data
215
     * @return FormInterface
216
     */
217
    public function setData(array $data)
218
    {
219
        if (isset($data[$this->form->getName()])) {
220
            $data = $data[$this->form->getName()];
221
        }
222
223
        $this->form->setValue($data);
224
225
        return $this;
226
    }
227
228
    /**
229
     * Returns the form data.
230
     *
231
     * @return array
232
     */
233
    public function getData()
234
    {
235
        return $this->form->getValue();
236
    }
237
}
238