** Entry point - index.php
La home page, si occupa di inizializzare l'applicazione ed impostare le route con annesso array [Controller, metodo].
-#+begin_src php :tangle www/pub/index.php
-<?php
-require_once __DIR__."/../vendor/autoload.php";
-
-use app\controllers\AuthController;
-use app\controllers\SiteController;
-use app\core\Application;
-
-$app = new Application(dirname(__DIR__));
-
-$app->router->get("/", [SiteController::class, "home"]);
-
-$app->router->get("/contact", [SiteController::class, "contact"]);
-$app->router->post("/contact", [SiteController::class, "handleContact"]);
-
-# User authentication
-$app->router->get("/login", [AuthController::class, "login"]);
-$app->router->post("/login", [AuthController::class, "login"]);
-
-$app->router->get("/register", [AuthController::class, "register"]);
-$app->router->post("/register", [AuthController::class, "register"]);
-
-$app->run();
-?>
-#+end_src
-
** Core
*** Main application class
La classe principale, si occupa di istanziare:
Gestisce tutti i controller tramite "BaseController".
-#+begin_src php :tangle www/core/Application.php
-<?php
-namespace app\core;
-
-class Application {
- private BaseController $controller;
-
- public Router $router;
- public Request $req;
- public Response $res;
-
- public static Application $app;
- public static string $ROOT_DIR;
-
- public function __construct(string $root) {
- self::$ROOT_DIR = $root;
- self::$app = $this;
-
- $this->req = new Request();
- $this->res = new Response();
- $this->router = new Router($this->req, $this->res);
- }
-
- public function run() {
- echo $this->router->resolve();
- }
-
- public function getController() {
- return $this->controller;
- }
-
- public function setController(BaseController $controller) {
- $this->controller = $controller;
- }
-}
-?>
-#+end_src
-
*** Router class
Una delle classi principali è il Router, gestisce l'array associativo "$routes" e la risoluzione delle varie request effettuate.
+ "loadLayoutContent()": restituisce il contenuto del layout selezionato dal controller della specifica view
+ "loadViewContent()": restituisce il contenuto della specifica view
-#+begin_src php :tangle www/core/Router.php
-<?php
-
-namespace app\core;
-
-class Router
-{
- private array $routes = [];
-
- public Request $req;
- public Response $res;
-
- public function __construct(Request $req, Response $res)
- {
- $this->req = $req;
- $this->res = $res;
- }
-
- public function get($path, $callback)
- {
- $this->routes["get"][$path] = $callback;
- }
-
- public function post($path, $callback)
- {
- $this->routes["post"][$path] = $callback;
- }
-
-
- public function resolve()
- {
- $path = $this->req->getPath();
- $method = $this->req->getMethod();
- $callback = $this->routes[$method][$path] ?? false;
-
- if (is_string($callback)) {
- return $this->renderView($callback);
- } else if (is_array($callback)) {
- Application::$app->setController(new $callback[0]);
- $callback[0] = Application::$app->getController();
- } else {
- $this->res->setStatusCode(404);
- return $this->renderView("404");
- }
-
- return call_user_func($callback, $this->req);
- }
-
- public function renderView(string $view, array $params = [])
- {
- $layoutContent = $this->loadLayoutContent();
- $viewContent = $this->loadViewContent($view, $params);
-
- return str_replace("{{content}}", $viewContent, $layoutContent);
- }
-
- private function loadLayoutContent() {
- $layout = Application::$app->getController()->layout;
- ob_start();
- include_once Application::$ROOT_DIR."/views/layouts/$layout.php";
- return ob_get_clean();
- }
-
- private function loadViewContent(string $view, array $params) {
- # modo epico per creare variabili con lo stesso nome assegnato nell'array!!
- foreach ($params as $key => $value) {
- $$key = $value;
- }
-
- ob_start();
- include_once Application::$ROOT_DIR."/views/$view.php";
- return ob_get_clean();
- }
-}
-?>
-#+end_src
-
*** Request class
Una classe helper, analizza le requests e restituisce l'informazione necessaria al richiedente.
- "getMethod()": restituisce il metodo utilizzato nella request in minuscolo. (Utile a Router resolve())
- "getBody()": restituisce i valori inseriti nella richiesta, sia essa una GET o una POST.
-#+begin_src php :tangle www/core/Request.php
-<?php
-namespace app\core;
-
-class Request {
- public function getPath() {
- $path = $_SERVER["REQUEST_URI"] ?? "/";
- $position = strpos($path, "?");
-
- if (!$position) {
- return $path;
- }
- return substr($path, 0, $position);
- }
-
- public function getMethod() {
- return strtolower($_SERVER["REQUEST_METHOD"]);
- }
-
- public function getBody() {
- $body = [];
-
- if ($this->getMethod() == "get") {
- foreach ($_GET as $key => $value) {
- $body[$key] = filter_input(INPUT_GET, $key, FILTER_SANITIZE_SPECIAL_CHARS);
- }
- }
- else if ($this->getMethod() == "post") {
- foreach ($_POST as $key => $value) {
- $body[$key] = filter_input(INPUT_POST, $key, FILTER_SANITIZE_SPECIAL_CHARS);
- }
- }
-
- return $body;
- }
-}
-?>
-#+end_src
-
-*** Response class
-Semplice helper class con il compiti di gestire dei parametri della response da inviare all'utente.
-
-#+begin_src php :tangle www/core/Response.php
-<?php
-namespace app\core;
-
-class Response {
- public function setStatusCode(int $code) {
- http_response_code($code);
- }
-}
-?>
-#+end_src
-
-*** Base controller class
-Classe controller di base.
-Tutti i controller si basano su questa classe in questo modo hanno tutti la possibiltà di impostare layout diversi per le view richieste dall'utente e la capacità di renderizzarle.
-
-#+begin_src php :tangle www/core/BaseController.php
-<?php
-namespace app\core;
-
-class BaseController {
- public string $layout = "main";
-
- public function render(string $view, array $params = []) {
- return Application::$app->router->renderView($view, $params);
- }
-
- public function setLayout(string $layout) {
- $this->layout = $layout;
- }
-}
-?>
-#+end_src
-
*** Base model class
Classe model di base.
Tutti i model si basano su questa classe, facilmente permettendo:
+ "addError()"
+ "errorMessage()"
-#+begin_src php :tangle www/core/BaseModel.php
-<?php
-namespace app\core;
-
-abstract class BaseModel {
- public const RULE_REQUIRED = "required";
- public const RULE_EMAIL = "email";
- public const RULE_MIN = "min";
- public const RULE_MAX = "max";
- public const RULE_MATCH = "match";
- public const RULE_UNIQUE = "unique";
-
- public array $errors = [];
-
- public function loadData($data) {
- foreach ($data as $key => $value) {
- if (property_exists($this, $key)) {
- $this->{$key} = $value;
- }
- }
- }
-
- abstract public function rules(): array;
-
- public function validate() {
- foreach ($this->rules() as $attribute => $rules) {
- $value = $this->{$attribute};
-
- foreach ($rules as $rule) {
- $ruleName = $rule;
-
- if (!is_string($ruleName)) {
- $ruleName = $rule[0];
- }
-
- if ($ruleName == self::RULE_REQUIRED && !$value) {
- $this->addError($attribute, self::RULE_REQUIRED);
- }
- if ($ruleName == self::RULE_EMAIL && !filter_var($value, FILTER_VALIDATE_EMAIL)) {
- $this->addError($attribute, self::RULE_EMAIL);
- }
- if ($ruleName == self::RULE_MIN && strlen($value) < $rule["min"]) {
- $this->addError($attribute, self::RULE_MIN, $rule);
- }
- if ($ruleName == self::RULE_MAX && strlen($value) > $rule["max"]) {
- $this->addError($attribute, self::RULE_MAX, $rule);
- }
- if ($ruleName == self::RULE_MATCH && $value != $this->{$rule["match"]}) {
- $this->addError($attribute, self::RULE_MATCH, $rule);
- }
- }
- }
-
- return empty($this->errors);
- }
-
- public function addError(string $attribute, string $rule, $params = []) {
- $message = $this->errorMessages()[$rule] ?? "";
- foreach ($params as $key => $value) {
- $message = str_replace("{{$key}}", $value, $message);
- }
- $this->errors[$attribute][] = $message;
- }
-
- public function errorMessages() {
- return [
- self::RULE_REQUIRED => "This field is required",
- self::RULE_EMAIL => "This field must be a valid email address",
- self::RULE_MIN => "Min length of this field must be {min}",
- self::RULE_MAX => "Max length of this field must be {max}",
- self::RULE_MATCH => "This field must be the same as {match}",
- ];
- }
-}
-?>
-#+end_src
-
** Models
Le classi "Model" gestiscono i dati presenti nel database e controlla che i dati seguano delle determinate regole.
Tutte le classi Model derivano dalla classe "BaseModel".
*** Registration model
Il model di registrazione si occupa della gestione dei dati di nuovi utenti e dell'interazione con il database.
-#+begin_src php :tangle www/models/RegisterModel.php
-<?php
-namespace app\models;
+Attraverso l'implementazione del metodo astratto "rules()" si possono impostare le regole che i campi della form dovranno seguire.
+*Attenzione*: i nomi dei parametri, i nomi inseriti nella array resistituito da "rules()" ed i nomi dei [[./www/core/forms/Field.php][Field]] _devono_ essere uguali.
-use app\core\BaseModel;
-
-class RegisterModel extends BaseModel {
- public string $firstname;
- public string $lastname;
- public string $email;
- public string $username;
- public string $pass;
- public string $passConf;
-
- public function register() {
- echo "Creating new user";
- }
-
- public function rules(): array {
- return [
- "firstname" => [self::RULE_REQUIRED],
- "lastname" => [self::RULE_REQUIRED],
- "email" => [self::RULE_REQUIRED, self::RULE_EMAIL],
- "pass" => [self::RULE_REQUIRED, [ self::RULE_MIN, "min" => 20 ], [ self::RULE_MAX, "max" => 100 ]],
- "passConf" => [self::RULE_REQUIRED, [self::RULE_MATCH, "match" => "pass" ]]
- ];
- }
-}
-?>
-#+end_src
+# TODO: register()
** Controllers
Le classi "Controller" svolgono il ruolo di ponte.
Esse permettono ai dati di apparire nella View richiesta dall'utente una volta prelevati dal database o di venirvi inseriti tramite una form.
-*** General controller
-Generico controller, probabilmente da eliminare.
-
-#+begin_src php :tangle www/controllers/SiteController.php
-<?php
-namespace app\controllers;
-
-use app\core\BaseController;
-use app\core\Request;
-
-class SiteController extends BaseController {
- public function home() {
- $params = [
- "name" => "Leonardo"
- ];
-
- return $this->render("home", $params);
- }
-
- public function contact() {
- return $this->render("contact");
- }
-
- public function handleContact(Request $req) {
- $body = $req->getBody();
-
- # $body validation
-
- return "Handling submitted data";
- }
-}
-?>
-#+end_src
-
*** Authentication controller
L'authentication controller ha 2 compiti fondamentali:
-- registrare nuovi utenti
+- registrare nuovi utenti con l'aiuto della classe "RegisterModel"
- permettere l'accesso ad utenti già registrati
-
-#+begin_src php :tangle www/controllers/AuthController.php
-<?php
-namespace app\controllers;
-
-use app\core\BaseController;
-use app\core\Request;
-use app\models\RegisterModel;
-
-class AuthController extends BaseController{
- public function login() {
- // $this->setLayout("auth");
- return $this->render("login");
- }
-
- public function register(Request $req) {
- // $this->setLayout("auth");
- $errors = [];
- $registerModel = new RegisterModel;
-
- if ($req->getMethod() == "post") {
- $registerModel->loadData($req->getBody());
-
- if ($registerModel->validate() && $registerModel->register()) {
- return "Success";
- }
- }
- return $this->render("register", [ "model" => $registerModel ]);
- }
-}
-#+end_src
+# TODO
self::RULE_MATCH => "This field must be the same as {match}",
];
}
+
+ public function hasError(string $attribute) {
+ return $this->errors[$attribute] ?? false;
+ }
+
+ public function getFirstError(string $attribute) {
+ return $this->errors[$attribute][0] ?? false;
+ }
}
?>
--- /dev/null
+<?php
+namespace app\core\forms;
+use app\core\BaseModel;
+
+class Field {
+ public const TYPE_TEXT = "text";
+ public const TYPE_PASSWORD = "password";
+ public const TYPE_NUMBER = "number";
+
+ public BaseModel $model;
+ public string $attribute;
+ public string $type;
+
+ public function __construct(BaseModel $model, string $attribute) {
+ $this->model = $model;
+ $this->attribute = $attribute;
+ $this->type = self::TYPE_TEXT;
+ }
+
+ public function __toString()
+ {
+ return sprintf(
+ '
+<div class="form-group">
+ <label>%s</label>
+ <input name="%s" type="%s" value="%s" class="form-control%s"/>
+ <div class="invalid-feedback">%s</div>
+</div>',
+ $this->attribute,
+ $this->attribute,
+ $this->type,
+ $this->model->{$this->attribute},
+ $this->model->hasError($this->attribute) ? " is-invalid" : "",
+ $this->model->getFirstError($this->attribute)
+ );
+ }
+
+ public function passwordField() {
+ $this->type = self::TYPE_PASSWORD;
+ return $this;
+ }
+}
+?>
--- /dev/null
+<?php
+namespace app\core\forms;
+use app\core\BaseModel;
+
+class Form {
+ public static function begin(string $action, string $method) {
+ echo sprintf('<form action="%s" method="%s">', $action, $method);
+ return new Form();
+ }
+
+ public static function end() {
+ echo "</form>";
+ }
+
+ public function field(BaseModel $model, string $attribute) {
+ return new Field($model, $attribute);
+ }
+}
+?>
use app\core\BaseModel;
class RegisterModel extends BaseModel {
- public string $firstname;
- public string $lastname;
- public string $email;
- public string $username;
- public string $pass;
- public string $passConf;
+ public string $Firstname = "";
+ public string $Lastname = "";
+ public string $Email = "";
+ public string $Username = "";
+ public string $Password = "";
+ public string $Confirm = "";
public function register() {
echo "Creating new user";
public function rules(): array {
return [
- "firstname" => [self::RULE_REQUIRED],
- "lastname" => [self::RULE_REQUIRED],
- "email" => [self::RULE_REQUIRED, self::RULE_EMAIL],
- "pass" => [self::RULE_REQUIRED, [ self::RULE_MIN, "min" => 20 ], [ self::RULE_MAX, "max" => 100 ]],
- "passConf" => [self::RULE_REQUIRED, [self::RULE_MATCH, "match" => "pass" ]]
+ "Firstname" => [self::RULE_REQUIRED],
+ "Lastname" => [self::RULE_REQUIRED],
+ "Username" => [self::RULE_REQUIRED],
+ "Email" => [self::RULE_REQUIRED, self::RULE_EMAIL],
+ "Password" => [self::RULE_REQUIRED, [ self::RULE_MIN, "min" => 20 ], [ self::RULE_MAX, "max" => 100 ]],
+ "Confirm" => [self::RULE_REQUIRED, [self::RULE_MATCH, "match" => "Password" ]]
];
}
}
+++ /dev/null
-<?php
-
-// autoload.php @generated by Composer
-
-require_once __DIR__ . '/composer/autoload_real.php';
-
-return ComposerAutoloaderInit14d0cfce075a17b4705c28ece2e994fc::getLoader();
+++ /dev/null
-<?php
-
-/*
- * This file is part of Composer.
- *
- * (c) Nils Adermann <naderman@naderman.de>
- * Jordi Boggiano <j.boggiano@seld.be>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
-
-namespace Composer\Autoload;
-
-/**
- * ClassLoader implements a PSR-0, PSR-4 and classmap class loader.
- *
- * $loader = new \Composer\Autoload\ClassLoader();
- *
- * // register classes with namespaces
- * $loader->add('Symfony\Component', __DIR__.'/component');
- * $loader->add('Symfony', __DIR__.'/framework');
- *
- * // activate the autoloader
- * $loader->register();
- *
- * // to enable searching the include path (eg. for PEAR packages)
- * $loader->setUseIncludePath(true);
- *
- * In this example, if you try to use a class in the Symfony\Component
- * namespace or one of its children (Symfony\Component\Console for instance),
- * the autoloader will first look for the class under the component/
- * directory, and it will then fallback to the framework/ directory if not
- * found before giving up.
- *
- * This class is loosely based on the Symfony UniversalClassLoader.
- *
- * @author Fabien Potencier <fabien@symfony.com>
- * @author Jordi Boggiano <j.boggiano@seld.be>
- * @see https://www.php-fig.org/psr/psr-0/
- * @see https://www.php-fig.org/psr/psr-4/
- */
-class ClassLoader
-{
- /** @var ?string */
- private $vendorDir;
-
- // PSR-4
- /**
- * @var array[]
- * @psalm-var array<string, array<string, int>>
- */
- private $prefixLengthsPsr4 = array();
- /**
- * @var array[]
- * @psalm-var array<string, array<int, string>>
- */
- private $prefixDirsPsr4 = array();
- /**
- * @var array[]
- * @psalm-var array<string, string>
- */
- private $fallbackDirsPsr4 = array();
-
- // PSR-0
- /**
- * @var array[]
- * @psalm-var array<string, array<string, string[]>>
- */
- private $prefixesPsr0 = array();
- /**
- * @var array[]
- * @psalm-var array<string, string>
- */
- private $fallbackDirsPsr0 = array();
-
- /** @var bool */
- private $useIncludePath = false;
-
- /**
- * @var string[]
- * @psalm-var array<string, string>
- */
- private $classMap = array();
-
- /** @var bool */
- private $classMapAuthoritative = false;
-
- /**
- * @var bool[]
- * @psalm-var array<string, bool>
- */
- private $missingClasses = array();
-
- /** @var ?string */
- private $apcuPrefix;
-
- /**
- * @var self[]
- */
- private static $registeredLoaders = array();
-
- /**
- * @param ?string $vendorDir
- */
- public function __construct($vendorDir = null)
- {
- $this->vendorDir = $vendorDir;
- }
-
- /**
- * @return string[]
- */
- public function getPrefixes()
- {
- if (!empty($this->prefixesPsr0)) {
- return call_user_func_array('array_merge', array_values($this->prefixesPsr0));
- }
-
- return array();
- }
-
- /**
- * @return array[]
- * @psalm-return array<string, array<int, string>>
- */
- public function getPrefixesPsr4()
- {
- return $this->prefixDirsPsr4;
- }
-
- /**
- * @return array[]
- * @psalm-return array<string, string>
- */
- public function getFallbackDirs()
- {
- return $this->fallbackDirsPsr0;
- }
-
- /**
- * @return array[]
- * @psalm-return array<string, string>
- */
- public function getFallbackDirsPsr4()
- {
- return $this->fallbackDirsPsr4;
- }
-
- /**
- * @return string[] Array of classname => path
- * @psalm-return array<string, string>
- */
- public function getClassMap()
- {
- return $this->classMap;
- }
-
- /**
- * @param string[] $classMap Class to filename map
- * @psalm-param array<string, string> $classMap
- *
- * @return void
- */
- public function addClassMap(array $classMap)
- {
- if ($this->classMap) {
- $this->classMap = array_merge($this->classMap, $classMap);
- } else {
- $this->classMap = $classMap;
- }
- }
-
- /**
- * Registers a set of PSR-0 directories for a given prefix, either
- * appending or prepending to the ones previously set for this prefix.
- *
- * @param string $prefix The prefix
- * @param string[]|string $paths The PSR-0 root directories
- * @param bool $prepend Whether to prepend the directories
- *
- * @return void
- */
- public function add($prefix, $paths, $prepend = false)
- {
- if (!$prefix) {
- if ($prepend) {
- $this->fallbackDirsPsr0 = array_merge(
- (array) $paths,
- $this->fallbackDirsPsr0
- );
- } else {
- $this->fallbackDirsPsr0 = array_merge(
- $this->fallbackDirsPsr0,
- (array) $paths
- );
- }
-
- return;
- }
-
- $first = $prefix[0];
- if (!isset($this->prefixesPsr0[$first][$prefix])) {
- $this->prefixesPsr0[$first][$prefix] = (array) $paths;
-
- return;
- }
- if ($prepend) {
- $this->prefixesPsr0[$first][$prefix] = array_merge(
- (array) $paths,
- $this->prefixesPsr0[$first][$prefix]
- );
- } else {
- $this->prefixesPsr0[$first][$prefix] = array_merge(
- $this->prefixesPsr0[$first][$prefix],
- (array) $paths
- );
- }
- }
-
- /**
- * Registers a set of PSR-4 directories for a given namespace, either
- * appending or prepending to the ones previously set for this namespace.
- *
- * @param string $prefix The prefix/namespace, with trailing '\\'
- * @param string[]|string $paths The PSR-4 base directories
- * @param bool $prepend Whether to prepend the directories
- *
- * @throws \InvalidArgumentException
- *
- * @return void
- */
- public function addPsr4($prefix, $paths, $prepend = false)
- {
- if (!$prefix) {
- // Register directories for the root namespace.
- if ($prepend) {
- $this->fallbackDirsPsr4 = array_merge(
- (array) $paths,
- $this->fallbackDirsPsr4
- );
- } else {
- $this->fallbackDirsPsr4 = array_merge(
- $this->fallbackDirsPsr4,
- (array) $paths
- );
- }
- } elseif (!isset($this->prefixDirsPsr4[$prefix])) {
- // Register directories for a new namespace.
- $length = strlen($prefix);
- if ('\\' !== $prefix[$length - 1]) {
- throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
- }
- $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length;
- $this->prefixDirsPsr4[$prefix] = (array) $paths;
- } elseif ($prepend) {
- // Prepend directories for an already registered namespace.
- $this->prefixDirsPsr4[$prefix] = array_merge(
- (array) $paths,
- $this->prefixDirsPsr4[$prefix]
- );
- } else {
- // Append directories for an already registered namespace.
- $this->prefixDirsPsr4[$prefix] = array_merge(
- $this->prefixDirsPsr4[$prefix],
- (array) $paths
- );
- }
- }
-
- /**
- * Registers a set of PSR-0 directories for a given prefix,
- * replacing any others previously set for this prefix.
- *
- * @param string $prefix The prefix
- * @param string[]|string $paths The PSR-0 base directories
- *
- * @return void
- */
- public function set($prefix, $paths)
- {
- if (!$prefix) {
- $this->fallbackDirsPsr0 = (array) $paths;
- } else {
- $this->prefixesPsr0[$prefix[0]][$prefix] = (array) $paths;
- }
- }
-
- /**
- * Registers a set of PSR-4 directories for a given namespace,
- * replacing any others previously set for this namespace.
- *
- * @param string $prefix The prefix/namespace, with trailing '\\'
- * @param string[]|string $paths The PSR-4 base directories
- *
- * @throws \InvalidArgumentException
- *
- * @return void
- */
- public function setPsr4($prefix, $paths)
- {
- if (!$prefix) {
- $this->fallbackDirsPsr4 = (array) $paths;
- } else {
- $length = strlen($prefix);
- if ('\\' !== $prefix[$length - 1]) {
- throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
- }
- $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length;
- $this->prefixDirsPsr4[$prefix] = (array) $paths;
- }
- }
-
- /**
- * Turns on searching the include path for class files.
- *
- * @param bool $useIncludePath
- *
- * @return void
- */
- public function setUseIncludePath($useIncludePath)
- {
- $this->useIncludePath = $useIncludePath;
- }
-
- /**
- * Can be used to check if the autoloader uses the include path to check
- * for classes.
- *
- * @return bool
- */
- public function getUseIncludePath()
- {
- return $this->useIncludePath;
- }
-
- /**
- * Turns off searching the prefix and fallback directories for classes
- * that have not been registered with the class map.
- *
- * @param bool $classMapAuthoritative
- *
- * @return void
- */
- public function setClassMapAuthoritative($classMapAuthoritative)
- {
- $this->classMapAuthoritative = $classMapAuthoritative;
- }
-
- /**
- * Should class lookup fail if not found in the current class map?
- *
- * @return bool
- */
- public function isClassMapAuthoritative()
- {
- return $this->classMapAuthoritative;
- }
-
- /**
- * APCu prefix to use to cache found/not-found classes, if the extension is enabled.
- *
- * @param string|null $apcuPrefix
- *
- * @return void
- */
- public function setApcuPrefix($apcuPrefix)
- {
- $this->apcuPrefix = function_exists('apcu_fetch') && filter_var(ini_get('apc.enabled'), FILTER_VALIDATE_BOOLEAN) ? $apcuPrefix : null;
- }
-
- /**
- * The APCu prefix in use, or null if APCu caching is not enabled.
- *
- * @return string|null
- */
- public function getApcuPrefix()
- {
- return $this->apcuPrefix;
- }
-
- /**
- * Registers this instance as an autoloader.
- *
- * @param bool $prepend Whether to prepend the autoloader or not
- *
- * @return void
- */
- public function register($prepend = false)
- {
- spl_autoload_register(array($this, 'loadClass'), true, $prepend);
-
- if (null === $this->vendorDir) {
- return;
- }
-
- if ($prepend) {
- self::$registeredLoaders = array($this->vendorDir => $this) + self::$registeredLoaders;
- } else {
- unset(self::$registeredLoaders[$this->vendorDir]);
- self::$registeredLoaders[$this->vendorDir] = $this;
- }
- }
-
- /**
- * Unregisters this instance as an autoloader.
- *
- * @return void
- */
- public function unregister()
- {
- spl_autoload_unregister(array($this, 'loadClass'));
-
- if (null !== $this->vendorDir) {
- unset(self::$registeredLoaders[$this->vendorDir]);
- }
- }
-
- /**
- * Loads the given class or interface.
- *
- * @param string $class The name of the class
- * @return true|null True if loaded, null otherwise
- */
- public function loadClass($class)
- {
- if ($file = $this->findFile($class)) {
- includeFile($file);
-
- return true;
- }
-
- return null;
- }
-
- /**
- * Finds the path to the file where the class is defined.
- *
- * @param string $class The name of the class
- *
- * @return string|false The path if found, false otherwise
- */
- public function findFile($class)
- {
- // class map lookup
- if (isset($this->classMap[$class])) {
- return $this->classMap[$class];
- }
- if ($this->classMapAuthoritative || isset($this->missingClasses[$class])) {
- return false;
- }
- if (null !== $this->apcuPrefix) {
- $file = apcu_fetch($this->apcuPrefix.$class, $hit);
- if ($hit) {
- return $file;
- }
- }
-
- $file = $this->findFileWithExtension($class, '.php');
-
- // Search for Hack files if we are running on HHVM
- if (false === $file && defined('HHVM_VERSION')) {
- $file = $this->findFileWithExtension($class, '.hh');
- }
-
- if (null !== $this->apcuPrefix) {
- apcu_add($this->apcuPrefix.$class, $file);
- }
-
- if (false === $file) {
- // Remember that this class does not exist.
- $this->missingClasses[$class] = true;
- }
-
- return $file;
- }
-
- /**
- * Returns the currently registered loaders indexed by their corresponding vendor directories.
- *
- * @return self[]
- */
- public static function getRegisteredLoaders()
- {
- return self::$registeredLoaders;
- }
-
- /**
- * @param string $class
- * @param string $ext
- * @return string|false
- */
- private function findFileWithExtension($class, $ext)
- {
- // PSR-4 lookup
- $logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . $ext;
-
- $first = $class[0];
- if (isset($this->prefixLengthsPsr4[$first])) {
- $subPath = $class;
- while (false !== $lastPos = strrpos($subPath, '\\')) {
- $subPath = substr($subPath, 0, $lastPos);
- $search = $subPath . '\\';
- if (isset($this->prefixDirsPsr4[$search])) {
- $pathEnd = DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $lastPos + 1);
- foreach ($this->prefixDirsPsr4[$search] as $dir) {
- if (file_exists($file = $dir . $pathEnd)) {
- return $file;
- }
- }
- }
- }
- }
-
- // PSR-4 fallback dirs
- foreach ($this->fallbackDirsPsr4 as $dir) {
- if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) {
- return $file;
- }
- }
-
- // PSR-0 lookup
- if (false !== $pos = strrpos($class, '\\')) {
- // namespaced class name
- $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)
- . strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR);
- } else {
- // PEAR-like class name
- $logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . $ext;
- }
-
- if (isset($this->prefixesPsr0[$first])) {
- foreach ($this->prefixesPsr0[$first] as $prefix => $dirs) {
- if (0 === strpos($class, $prefix)) {
- foreach ($dirs as $dir) {
- if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
- return $file;
- }
- }
- }
- }
- }
-
- // PSR-0 fallback dirs
- foreach ($this->fallbackDirsPsr0 as $dir) {
- if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) {
- return $file;
- }
- }
-
- // PSR-0 include paths.
- if ($this->useIncludePath && $file = stream_resolve_include_path($logicalPathPsr0)) {
- return $file;
- }
-
- return false;
- }
-}
-
-/**
- * Scope isolated include.
- *
- * Prevents access to $this/self from included files.
- *
- * @param string $file
- * @return void
- * @private
- */
-function includeFile($file)
-{
- include $file;
-}
+++ /dev/null
-<?php
-
-/*
- * This file is part of Composer.
- *
- * (c) Nils Adermann <naderman@naderman.de>
- * Jordi Boggiano <j.boggiano@seld.be>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
-
-namespace Composer;
-
-use Composer\Autoload\ClassLoader;
-use Composer\Semver\VersionParser;
-
-/**
- * This class is copied in every Composer installed project and available to all
- *
- * See also https://getcomposer.org/doc/07-runtime.md#installed-versions
- *
- * To require its presence, you can require `composer-runtime-api ^2.0`
- */
-class InstalledVersions
-{
- /**
- * @var mixed[]|null
- * @psalm-var array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>}|array{}|null
- */
- private static $installed;
-
- /**
- * @var bool|null
- */
- private static $canGetVendors;
-
- /**
- * @var array[]
- * @psalm-var array<string, array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>}>
- */
- private static $installedByVendor = array();
-
- /**
- * Returns a list of all package names which are present, either by being installed, replaced or provided
- *
- * @return string[]
- * @psalm-return list<string>
- */
- public static function getInstalledPackages()
- {
- $packages = array();
- foreach (self::getInstalled() as $installed) {
- $packages[] = array_keys($installed['versions']);
- }
-
- if (1 === \count($packages)) {
- return $packages[0];
- }
-
- return array_keys(array_flip(\call_user_func_array('array_merge', $packages)));
- }
-
- /**
- * Returns a list of all package names with a specific type e.g. 'library'
- *
- * @param string $type
- * @return string[]
- * @psalm-return list<string>
- */
- public static function getInstalledPackagesByType($type)
- {
- $packagesByType = array();
-
- foreach (self::getInstalled() as $installed) {
- foreach ($installed['versions'] as $name => $package) {
- if (isset($package['type']) && $package['type'] === $type) {
- $packagesByType[] = $name;
- }
- }
- }
-
- return $packagesByType;
- }
-
- /**
- * Checks whether the given package is installed
- *
- * This also returns true if the package name is provided or replaced by another package
- *
- * @param string $packageName
- * @param bool $includeDevRequirements
- * @return bool
- */
- public static function isInstalled($packageName, $includeDevRequirements = true)
- {
- foreach (self::getInstalled() as $installed) {
- if (isset($installed['versions'][$packageName])) {
- return $includeDevRequirements || empty($installed['versions'][$packageName]['dev_requirement']);
- }
- }
-
- return false;
- }
-
- /**
- * Checks whether the given package satisfies a version constraint
- *
- * e.g. If you want to know whether version 2.3+ of package foo/bar is installed, you would call:
- *
- * Composer\InstalledVersions::satisfies(new VersionParser, 'foo/bar', '^2.3')
- *
- * @param VersionParser $parser Install composer/semver to have access to this class and functionality
- * @param string $packageName
- * @param string|null $constraint A version constraint to check for, if you pass one you have to make sure composer/semver is required by your package
- * @return bool
- */
- public static function satisfies(VersionParser $parser, $packageName, $constraint)
- {
- $constraint = $parser->parseConstraints($constraint);
- $provided = $parser->parseConstraints(self::getVersionRanges($packageName));
-
- return $provided->matches($constraint);
- }
-
- /**
- * Returns a version constraint representing all the range(s) which are installed for a given package
- *
- * It is easier to use this via isInstalled() with the $constraint argument if you need to check
- * whether a given version of a package is installed, and not just whether it exists
- *
- * @param string $packageName
- * @return string Version constraint usable with composer/semver
- */
- public static function getVersionRanges($packageName)
- {
- foreach (self::getInstalled() as $installed) {
- if (!isset($installed['versions'][$packageName])) {
- continue;
- }
-
- $ranges = array();
- if (isset($installed['versions'][$packageName]['pretty_version'])) {
- $ranges[] = $installed['versions'][$packageName]['pretty_version'];
- }
- if (array_key_exists('aliases', $installed['versions'][$packageName])) {
- $ranges = array_merge($ranges, $installed['versions'][$packageName]['aliases']);
- }
- if (array_key_exists('replaced', $installed['versions'][$packageName])) {
- $ranges = array_merge($ranges, $installed['versions'][$packageName]['replaced']);
- }
- if (array_key_exists('provided', $installed['versions'][$packageName])) {
- $ranges = array_merge($ranges, $installed['versions'][$packageName]['provided']);
- }
-
- return implode(' || ', $ranges);
- }
-
- throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
- }
-
- /**
- * @param string $packageName
- * @return string|null If the package is being replaced or provided but is not really installed, null will be returned as version, use satisfies or getVersionRanges if you need to know if a given version is present
- */
- public static function getVersion($packageName)
- {
- foreach (self::getInstalled() as $installed) {
- if (!isset($installed['versions'][$packageName])) {
- continue;
- }
-
- if (!isset($installed['versions'][$packageName]['version'])) {
- return null;
- }
-
- return $installed['versions'][$packageName]['version'];
- }
-
- throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
- }
-
- /**
- * @param string $packageName
- * @return string|null If the package is being replaced or provided but is not really installed, null will be returned as version, use satisfies or getVersionRanges if you need to know if a given version is present
- */
- public static function getPrettyVersion($packageName)
- {
- foreach (self::getInstalled() as $installed) {
- if (!isset($installed['versions'][$packageName])) {
- continue;
- }
-
- if (!isset($installed['versions'][$packageName]['pretty_version'])) {
- return null;
- }
-
- return $installed['versions'][$packageName]['pretty_version'];
- }
-
- throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
- }
-
- /**
- * @param string $packageName
- * @return string|null If the package is being replaced or provided but is not really installed, null will be returned as reference
- */
- public static function getReference($packageName)
- {
- foreach (self::getInstalled() as $installed) {
- if (!isset($installed['versions'][$packageName])) {
- continue;
- }
-
- if (!isset($installed['versions'][$packageName]['reference'])) {
- return null;
- }
-
- return $installed['versions'][$packageName]['reference'];
- }
-
- throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
- }
-
- /**
- * @param string $packageName
- * @return string|null If the package is being replaced or provided but is not really installed, null will be returned as install path. Packages of type metapackages also have a null install path.
- */
- public static function getInstallPath($packageName)
- {
- foreach (self::getInstalled() as $installed) {
- if (!isset($installed['versions'][$packageName])) {
- continue;
- }
-
- return isset($installed['versions'][$packageName]['install_path']) ? $installed['versions'][$packageName]['install_path'] : null;
- }
-
- throw new \OutOfBoundsException('Package "' . $packageName . '" is not installed');
- }
-
- /**
- * @return array
- * @psalm-return array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}
- */
- public static function getRootPackage()
- {
- $installed = self::getInstalled();
-
- return $installed[0]['root'];
- }
-
- /**
- * Returns the raw installed.php data for custom implementations
- *
- * @deprecated Use getAllRawData() instead which returns all datasets for all autoloaders present in the process. getRawData only returns the first dataset loaded, which may not be what you expect.
- * @return array[]
- * @psalm-return array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>}
- */
- public static function getRawData()
- {
- @trigger_error('getRawData only returns the first dataset loaded, which may not be what you expect. Use getAllRawData() instead which returns all datasets for all autoloaders present in the process.', E_USER_DEPRECATED);
-
- if (null === self::$installed) {
- // only require the installed.php file if this file is loaded from its dumped location,
- // and not from its source location in the composer/composer package, see https://github.com/composer/composer/issues/9937
- if (substr(__DIR__, -8, 1) !== 'C') {
- self::$installed = include __DIR__ . '/installed.php';
- } else {
- self::$installed = array();
- }
- }
-
- return self::$installed;
- }
-
- /**
- * Returns the raw data of all installed.php which are currently loaded for custom implementations
- *
- * @return array[]
- * @psalm-return list<array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>}>
- */
- public static function getAllRawData()
- {
- return self::getInstalled();
- }
-
- /**
- * Lets you reload the static array from another file
- *
- * This is only useful for complex integrations in which a project needs to use
- * this class but then also needs to execute another project's autoloader in process,
- * and wants to ensure both projects have access to their version of installed.php.
- *
- * A typical case would be PHPUnit, where it would need to make sure it reads all
- * the data it needs from this class, then call reload() with
- * `require $CWD/vendor/composer/installed.php` (or similar) as input to make sure
- * the project in which it runs can then also use this class safely, without
- * interference between PHPUnit's dependencies and the project's dependencies.
- *
- * @param array[] $data A vendor/composer/installed.php data set
- * @return void
- *
- * @psalm-param array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>} $data
- */
- public static function reload($data)
- {
- self::$installed = $data;
- self::$installedByVendor = array();
- }
-
- /**
- * @return array[]
- * @psalm-return list<array{root: array{name: string, version: string, reference: string, pretty_version: string, aliases: string[], dev: bool, install_path: string, type: string}, versions: array<string, array{dev_requirement: bool, pretty_version?: string, version?: string, aliases?: string[], reference?: string, replaced?: string[], provided?: string[], install_path?: string, type?: string}>}>
- */
- private static function getInstalled()
- {
- if (null === self::$canGetVendors) {
- self::$canGetVendors = method_exists('Composer\Autoload\ClassLoader', 'getRegisteredLoaders');
- }
-
- $installed = array();
-
- if (self::$canGetVendors) {
- foreach (ClassLoader::getRegisteredLoaders() as $vendorDir => $loader) {
- if (isset(self::$installedByVendor[$vendorDir])) {
- $installed[] = self::$installedByVendor[$vendorDir];
- } elseif (is_file($vendorDir.'/composer/installed.php')) {
- $installed[] = self::$installedByVendor[$vendorDir] = require $vendorDir.'/composer/installed.php';
- if (null === self::$installed && strtr($vendorDir.'/composer', '\\', '/') === strtr(__DIR__, '\\', '/')) {
- self::$installed = $installed[count($installed) - 1];
- }
- }
- }
- }
-
- if (null === self::$installed) {
- // only require the installed.php file if this file is loaded from its dumped location,
- // and not from its source location in the composer/composer package, see https://github.com/composer/composer/issues/9937
- if (substr(__DIR__, -8, 1) !== 'C') {
- self::$installed = require __DIR__ . '/installed.php';
- } else {
- self::$installed = array();
- }
- }
- $installed[] = self::$installed;
-
- return $installed;
- }
-}
+++ /dev/null
-
-Copyright (c) Nils Adermann, Jordi Boggiano
-
-Permission is hereby granted, free of charge, to any person obtaining a copy
-of this software and associated documentation files (the "Software"), to deal
-in the Software without restriction, including without limitation the rights
-to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-copies of the Software, and to permit persons to whom the Software is furnished
-to do so, subject to the following conditions:
-
-The above copyright notice and this permission notice shall be included in all
-copies or substantial portions of the Software.
-
-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
-THE SOFTWARE.
-
+++ /dev/null
-<?php
-
-// autoload_classmap.php @generated by Composer
-
-$vendorDir = dirname(dirname(__FILE__));
-$baseDir = dirname($vendorDir);
-
-return array(
- 'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php',
-);
+++ /dev/null
-<?php
-
-// autoload_namespaces.php @generated by Composer
-
-$vendorDir = dirname(dirname(__FILE__));
-$baseDir = dirname($vendorDir);
-
-return array(
-);
+++ /dev/null
-<?php
-
-// autoload_psr4.php @generated by Composer
-
-$vendorDir = dirname(dirname(__FILE__));
-$baseDir = dirname($vendorDir);
-
-return array(
- 'app\\' => array($baseDir . '/'),
-);
+++ /dev/null
-<?php
-
-// autoload_real.php @generated by Composer
-
-class ComposerAutoloaderInit14d0cfce075a17b4705c28ece2e994fc
-{
- private static $loader;
-
- public static function loadClassLoader($class)
- {
- if ('Composer\Autoload\ClassLoader' === $class) {
- require __DIR__ . '/ClassLoader.php';
- }
- }
-
- /**
- * @return \Composer\Autoload\ClassLoader
- */
- public static function getLoader()
- {
- if (null !== self::$loader) {
- return self::$loader;
- }
-
- spl_autoload_register(array('ComposerAutoloaderInit14d0cfce075a17b4705c28ece2e994fc', 'loadClassLoader'), true, true);
- self::$loader = $loader = new \Composer\Autoload\ClassLoader(\dirname(\dirname(__FILE__)));
- spl_autoload_unregister(array('ComposerAutoloaderInit14d0cfce075a17b4705c28ece2e994fc', 'loadClassLoader'));
-
- $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
- if ($useStaticLoader) {
- require __DIR__ . '/autoload_static.php';
-
- call_user_func(\Composer\Autoload\ComposerStaticInit14d0cfce075a17b4705c28ece2e994fc::getInitializer($loader));
- } else {
- $map = require __DIR__ . '/autoload_namespaces.php';
- foreach ($map as $namespace => $path) {
- $loader->set($namespace, $path);
- }
-
- $map = require __DIR__ . '/autoload_psr4.php';
- foreach ($map as $namespace => $path) {
- $loader->setPsr4($namespace, $path);
- }
-
- $classMap = require __DIR__ . '/autoload_classmap.php';
- if ($classMap) {
- $loader->addClassMap($classMap);
- }
- }
-
- $loader->register(true);
-
- return $loader;
- }
-}
+++ /dev/null
-<?php
-
-// autoload_static.php @generated by Composer
-
-namespace Composer\Autoload;
-
-class ComposerStaticInit14d0cfce075a17b4705c28ece2e994fc
-{
- public static $prefixLengthsPsr4 = array (
- 'a' =>
- array (
- 'app\\' => 4,
- ),
- );
-
- public static $prefixDirsPsr4 = array (
- 'app\\' =>
- array (
- 0 => __DIR__ . '/../..' . '/',
- ),
- );
-
- public static $classMap = array (
- 'Composer\\InstalledVersions' => __DIR__ . '/..' . '/composer/InstalledVersions.php',
- );
-
- public static function getInitializer(ClassLoader $loader)
- {
- return \Closure::bind(function () use ($loader) {
- $loader->prefixLengthsPsr4 = ComposerStaticInit14d0cfce075a17b4705c28ece2e994fc::$prefixLengthsPsr4;
- $loader->prefixDirsPsr4 = ComposerStaticInit14d0cfce075a17b4705c28ece2e994fc::$prefixDirsPsr4;
- $loader->classMap = ComposerStaticInit14d0cfce075a17b4705c28ece2e994fc::$classMap;
-
- }, null, ClassLoader::class);
- }
-}
+++ /dev/null
-{
- "packages": [],
- "dev": true,
- "dev-package-names": []
-}
+++ /dev/null
-<?php return array(
- 'root' => array(
- 'pretty_version' => 'dev-master',
- 'version' => 'dev-master',
- 'type' => 'library',
- 'install_path' => __DIR__ . '/../../',
- 'aliases' => array(),
- 'reference' => 'c5a8bfb241afda9ba2150372cb238308bb26b94a',
- 'name' => 'leo/weeb-site',
- 'dev' => true,
- ),
- 'versions' => array(
- 'leo/weeb-site' => array(
- 'pretty_version' => 'dev-master',
- 'version' => 'dev-master',
- 'type' => 'library',
- 'install_path' => __DIR__ . '/../../',
- 'aliases' => array(),
- 'reference' => 'c5a8bfb241afda9ba2150372cb238308bb26b94a',
- 'dev_requirement' => false,
- ),
- ),
-);
<h1>Registration page</h1>
<div class="container">
- <form method="POST" class="row g-3">
- <div class="col-md-4">
- <label class="form-label">Firstname</label>
- <input name="firstname" type="text" class="form-control">
+ <?php $form = app\core\forms\Form::begin("", "post"); ?>
+ <div class="row">
+ <div class="col">
+ <?php echo $form->field($model, "Firstname"); ?>
</div>
- <div class="col-md-4">
- <label class="form-label">Lastname</label>
- <input name="lastname" type="text" class="form-control">
+ <div class="col">
+ <?php echo $form->field($model, "Lastname"); ?>
</div>
- <div class="col-md-4">
- <label class="form-label">Email</label>
- <input name="email" type="email" class="form-control">
+ </div>
+ <?php echo $form->field($model, "Email"); ?>
+ <?php echo $form->field($model, "Username"); ?>
+ <div class="row">
+ <div class="col">
+ <?php echo $form->field($model, "Password")->passwordField(); ?>
</div>
- <div class="col-md-12">
- <label class="form-label">Username</label>
- <input name="username" type="text" class="form-control">
+ <div class="col">
+ <?php echo $form->field($model, "Confirm")->passwordField(); ?>
</div>
- <div class="col-md-6">
- <label class="form-label">Password</label>
- <input name="pass" type="password" class="form-control">
- </div>
- <div class="col-md-6">
- <label class="form-label">Password confirm</label>
- <input name="passConf" name="pass" type="password" class="form-control">
- </div>
- <div class="col-md-1">
- <button type="submit" class="btn btn-primary">Submit</button>
- </div>
- </form>
+ </div>
+
+ <button type="submit" class="btn btn-primary">Submit</button>
+ <?php app\core\forms\Form::end(); ?>
</div>