From 89c6d6d867e2e31a9f02a00a2e57518980a01bd6 Mon Sep 17 00:00:00 2001 From: Brandon Shipley Date: Tue, 16 Apr 2024 02:00:10 -0700 Subject: [PATCH] initial commit --- .gitignore | 9 + README.md | 11 + composer.json | 24 ++ config/app.example.php | 9 + config/paginator-templates.php | 24 ++ phpunit.xml.dist | 30 +++ src/CakeHtmxPlugin.php | 93 +++++++ src/Command/Bake/TableElementCommand.php | 269 +++++++++++++++++++++ src/Controller/AppController.php | 10 + src/Controller/Component/HtmxComponent.php | 140 +++++++++++ templates/bake/Template/Element/table.twig | 78 ++++++ templates/bake/Template/add.twig | 35 +++ templates/bake/Template/edit.twig | 35 +++ templates/bake/Template/index.twig | 28 +++ templates/bake/Template/view.twig | 141 +++++++++++ tests/bootstrap.php | 55 +++++ tests/schema.sql | 1 + webroot/.gitkeep | 0 18 files changed, 992 insertions(+) create mode 100644 .gitignore create mode 100644 README.md create mode 100644 composer.json create mode 100644 config/app.example.php create mode 100644 config/paginator-templates.php create mode 100644 phpunit.xml.dist create mode 100644 src/CakeHtmxPlugin.php create mode 100644 src/Command/Bake/TableElementCommand.php create mode 100644 src/Controller/AppController.php create mode 100644 src/Controller/Component/HtmxComponent.php create mode 100644 templates/bake/Template/Element/table.twig create mode 100644 templates/bake/Template/add.twig create mode 100644 templates/bake/Template/edit.twig create mode 100644 templates/bake/Template/index.twig create mode 100644 templates/bake/Template/view.twig create mode 100644 tests/bootstrap.php create mode 100644 tests/schema.sql create mode 100644 webroot/.gitkeep diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..77c68a6 --- /dev/null +++ b/.gitignore @@ -0,0 +1,9 @@ +/composer.lock +/composer.phar +/phpunit.xml +/.phpunit.result.cache +/.phpunit.cache +/phpunit.phar +/config/Migrations/schema-dump-default.lock +/vendor/ +/.idea/ diff --git a/README.md b/README.md new file mode 100644 index 0000000..8de18a0 --- /dev/null +++ b/README.md @@ -0,0 +1,11 @@ +# CakeHtmx plugin for CakePHP + +## Installation + +You can install this plugin into your CakePHP application using [composer](https://getcomposer.org). + +The recommended way to install composer packages is: + +``` +composer require your-name-here/cake-htmx +``` diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..418fca6 --- /dev/null +++ b/composer.json @@ -0,0 +1,24 @@ +{ + "name": "hi-powered-dev/cake-htmx", + "description": "CakeHtmx plugin for CakePHP", + "type": "cakephp-plugin", + "license": "MIT", + "require": { + "php": ">=8.1", + "cakephp/cakephp": "^5.0.1" + }, + "require-dev": { + "phpunit/phpunit": "^10.1" + }, + "autoload": { + "psr-4": { + "CakeHtmx\\": "src/" + } + }, + "autoload-dev": { + "psr-4": { + "CakeHtmx\\Test\\": "tests/", + "Cake\\Test\\": "vendor/cakephp/cakephp/tests/" + } + } +} diff --git a/config/app.example.php b/config/app.example.php new file mode 100644 index 0000000..d11ca48 --- /dev/null +++ b/config/app.example.php @@ -0,0 +1,9 @@ + [ + 'boostLinks' => true, + 'usePaginator' => true, + ], +]; diff --git a/config/paginator-templates.php b/config/paginator-templates.php new file mode 100644 index 0000000..f8bd207 --- /dev/null +++ b/config/paginator-templates.php @@ -0,0 +1,24 @@ + '', + 'nextDisabled' => '', + 'prevActive' => '', + 'prevDisabled' => '', + 'counterRange' => '{{start}} - {{end}} of {{count}}', + 'counterPages' => '{{page}} of {{pages}}', + 'first' => '
  • {{text}}
  • ', + 'last' => '
  • {{text}}
  • ', + 'number' => '
  • {{text}}
  • ', + 'current' => '
  • {{text}}
  • ', + 'ellipsis' => '
  • ', + 'sort' => '{{text}}', + 'sortAsc' => '{{text}}', + 'sortDesc' => '{{text}}', + 'sortAscLocked' => '{{text}}', + 'sortDescLocked' => '{{text}}', +]; diff --git a/phpunit.xml.dist b/phpunit.xml.dist new file mode 100644 index 0000000..70bab82 --- /dev/null +++ b/phpunit.xml.dist @@ -0,0 +1,30 @@ + + + + + + + + + + + tests/TestCase/ + + + + + + + + + + + src/ + + + diff --git a/src/CakeHtmxPlugin.php b/src/CakeHtmxPlugin.php new file mode 100644 index 0000000..5c1622c --- /dev/null +++ b/src/CakeHtmxPlugin.php @@ -0,0 +1,93 @@ +plugin( + 'CakeHtmx', + ['path' => '/cake-htmx'], + function (RouteBuilder $builder) { + // Add custom routes here + + $builder->fallbacks(); + } + ); + parent::routes($routes); + } + + /** + * Add middleware for the plugin. + * + * @param \Cake\Http\MiddlewareQueue $middlewareQueue The middleware queue to update. + * @return \Cake\Http\MiddlewareQueue + */ + public function middleware(MiddlewareQueue $middlewareQueue): MiddlewareQueue + { + // Add your middlewares here + + return $middlewareQueue; + } + + /** + * Add commands for the plugin. + * + * @param \Cake\Console\CommandCollection $commands The command collection to update. + * @return \Cake\Console\CommandCollection + */ + public function console(CommandCollection $commands): CommandCollection + { + // Add your commands here + + $commands = parent::console($commands); + + return $commands; + } + + /** + * Register application container services. + * + * @param \Cake\Core\ContainerInterface $container The Container to update. + * @return void + * @link https://book.cakephp.org/4/en/development/dependency-injection.html#dependency-injection + */ + public function services(ContainerInterface $container): void + { + // Add your services here + } +} diff --git a/src/Command/Bake/TableElementCommand.php b/src/Command/Bake/TableElementCommand.php new file mode 100644 index 0000000..adcf193 --- /dev/null +++ b/src/Command/Bake/TableElementCommand.php @@ -0,0 +1,269 @@ + + */ + public array $scaffoldActions = ['index']; + /** + * Execute the command. + * + * @param \Cake\Console\Arguments $args The command arguments. + * @param \Cake\Console\ConsoleIo $io The console io + * @return int|null The exit code or null for success + */ + public function execute(Arguments $args, ConsoleIo $io): ?int + { + $this->extractCommonProperties($args); + $name = $args->getArgument('name') ?? ''; + $name = $this->_getName($name); + + if (empty($name)) { + $io->out('Possible tables to bake view templates for based on your current database:'); + /** @var \Cake\Database\Connection $connection */ + $connection = ConnectionManager::get($this->connection); + $scanner = new TableScanner($connection); + foreach ($scanner->listUnskipped() as $table) { + $io->out('- ' . $this->_camelize($table)); + } + + return static::CODE_SUCCESS; + } + $action = $args->getArgument('action'); + $template = 'index'; + $controller = $args->getOption('controller'); + $this->controller($args, $name, $controller); + $this->model($name); + + if ($template && $action === null) { + $action = $template; + } + $this->bake($args, $io, $template, true, $action); // update index file + $this->bakeElementFile($args, $io, $template, true, $action); // add/update element table file + + return static::CODE_SUCCESS; + } + + /** + * Assembles and writes bakes the view file. + * + * @param \Cake\Console\Arguments $args CLI arguments + * @param \Cake\Console\ConsoleIo $io Console io + * @param string $template Template file to use. + * @param string|true $content Content to write. + * @param ?string $outputFile The output file to create. If null will use `$template` + * @return void + */ + public function bake( + Arguments $args, + ConsoleIo $io, + string $template, + string|bool $content = '', + ?string $outputFile = null + ): void { + if ($outputFile === null) { + $outputFile = $template; + } + if ($content === true) { + $content = $this->getContent($args, $io, $template); + } + if (empty($content)) { + // phpcs:ignore Generic.Files.LineLength + $io->err("No generated content for '{$template}.{$this->ext}', not generating template."); + + return; + } + $path = $this->getTemplatePath($args); + $filename = $path . Inflector::underscore($outputFile) . '.' . $this->ext; + + $io->out("\n" . sprintf('Baking `%s` view template file...', $outputFile), 1, ConsoleIo::NORMAL); + $io->createFile($filename, $content, $this->force); + } + + /** + * Get the path base for view templates. + * + * @param \Cake\Console\Arguments $args The arguments + * @param string|null $container Unused. + * @return string + */ + public function getElementPath(Arguments $args, ?string $container = null): string + { + $path = BakeCommand::getTemplatePath($args, $container) . 'element' . DS; + $path .= $this->controllerName . DS . 'index' . DS; + + return $path; + } + + /** + * Assembles and writes bakes the view file. + * + * @param \Cake\Console\Arguments $args CLI arguments + * @param \Cake\Console\ConsoleIo $io Console io + * @param string $template Template file to use. + * @param string|true $content Content to write. + * @param ?string $outputFile The output file to create. If null will use `$template` + * @return void + */ + public function bakeElementFile( + Arguments $args, + ConsoleIo $io, + string $template, + string|bool $content = '', + ?string $outputFile = null + ): void { + if ($outputFile === null) { + $outputFile = $template; + } + if ($content === true) { + $content = $this->getElementContent($args, $io, $template); + } + if (empty($content)) { + // phpcs:ignore Generic.Files.LineLength + $io->err("No generated content for '{$template}.{$this->ext}', not generating template."); + + return; + } + $path = $this->getElementPath($args); + $filename = $path . 'table.' . $this->ext; + + $io->out("\n" . sprintf('Baking `%s` view template file...', $outputFile), 1, ConsoleIo::NORMAL); + $io->createFile($filename, $content, $this->force); + } + + public function getElementContent(Arguments $args, ConsoleIo $io, string $action, ?array $vars = null): string + { + if (!$vars) { + $vars = $this->_loadController($io); + } + + if (empty($vars['primaryKey'])) { + $io->error('Cannot generate views for models with no primary key'); + $this->abort(); + } + + if (in_array($action, $this->excludeHiddenActions)) { + $vars['fields'] = array_diff($vars['fields'], $vars['hidden']); + } + + $renderer = $this->createTemplateRenderer() + ->set('action', $action) + ->set('plugin', $this->plugin) + ->set($vars); + + $indexColumns = 0; + if ($action === 'index' && $args->getOption('index-columns') !== null) { + $indexColumns = $args->getOption('index-columns'); + } + $renderer->set('indexColumns', $indexColumns); + + return $renderer->generate('Bake.Template/Element/table'); + } + + /** + * Loads Controller and sets variables for the template + * Available template variables: + * + * - 'modelObject' + * - 'modelClass' + * - 'entityClass' + * - 'primaryKey' + * - 'displayField' + * - 'singularVar' + * - 'pluralVar' + * - 'singularHumanName' + * - 'pluralHumanName' + * - 'fields' + * - 'keyFields' + * - 'schema' + * + * @param \Cake\Console\ConsoleIo $io Instance of the ConsoleIO + * @return array Returns variables to be made available to a view template + */ + protected function _loadController(ConsoleIo $io): array + { + if ($this->getTableLocator()->exists($this->modelName)) { + $modelObject = $this->getTableLocator()->get($this->modelName); + } else { + $modelObject = $this->getTableLocator()->get($this->modelName, [ + 'connectionName' => $this->connection, + ]); + } + + $namespace = Configure::read('App.namespace'); + + $primaryKey = $displayField = $singularVar = $singularHumanName = null; + $schema = $fields = $hidden = $modelClass = null; + try { + $primaryKey = (array)$modelObject->getPrimaryKey(); + $displayField = $modelObject->getDisplayField(); + $singularVar = $this->_singularName($this->controllerName); + $singularHumanName = $this->_singularHumanName($this->controllerName); + $schema = $modelObject->getSchema(); + $fields = $schema->columns(); + $hidden = $modelObject->newEmptyEntity()->getHidden() ?: ['token', 'password', 'passwd']; + $modelClass = $this->modelName; + } catch (Exception $exception) { + $io->error($exception->getMessage()); + $this->abort(); + } + + [, $entityClass] = namespaceSplit($this->_entityName($this->modelName)); + $entityClass = sprintf('%s\Model\Entity\%s', $namespace, $entityClass); + if (!class_exists($entityClass)) { + $entityClass = EntityInterface::class; + } + $associations = $this->_filteredAssociations($modelObject); + $keyFields = []; + if (!empty($associations['BelongsTo'])) { + foreach ($associations['BelongsTo'] as $assoc) { + $keyFields[$assoc['foreignKey']] = $assoc['variable']; + } + } + + $pluralVar = Inflector::variable($this->controllerName); + $pluralHumanName = $this->_pluralHumanName($this->controllerName); + $controllerName = $this->controllerName; + return compact( + 'modelObject', + 'modelClass', + 'entityClass', + 'schema', + 'primaryKey', + 'displayField', + 'singularVar', + 'pluralVar', + 'singularHumanName', + 'pluralHumanName', + 'fields', + 'hidden', + 'associations', + 'keyFields', + 'namespace', + 'controllerName' + ); + } + +} diff --git a/src/Controller/AppController.php b/src/Controller/AppController.php new file mode 100644 index 0000000..208c1dd --- /dev/null +++ b/src/Controller/AppController.php @@ -0,0 +1,10 @@ + + */ + protected array $_defaultConfig = []; + + /** + * @var \Cake\Controller\Controller + */ + protected $controller; + + /** + * @param array $config + * @return void + */ + public function initialize(array $config): void { + parent::initialize($config); + + $this->controller = $this->getController(); + } + + /** + * Convenience method to check if request is from HTMX + * + * @return bool + */ + public function isHtmx() + { + return $this->controller->getRequest()->getHeaderLine('HX-Request') === 'true'; + } + + /** + * Convenience method to check if request is Boosted + * + * @return bool + */ + public function isBoosted() + { + return $this->controller->getRequest()->getHeaderLine('HX-Boosted') === 'true'; + } + + /** + * Get HTMX target id + * + * @return string|null + */ + public function getHtmxTarget() + { + $target = $this->controller->getRequest()->getHeaderLine('HX-Target'); + + return $target ?: null; + } + + /** + * Get HTMX trigger id + * + * @return string|null + */ + public function getHtmxTrigger() + { + $trigger = $this->controller->getRequest()->getHeaderLine('HX-Trigger'); + + return $trigger ?: null; + } + + /** + * Get HTMX trigger name + * + * @return string|null + */ + public function getHtmxTriggerName() + { + $trigger = $this->controller->getRequest()->getHeaderLine('HX-Trigger-Name'); + + return $trigger ?: null; + } + + /** + * Set headers to cache this request. + * Opposite of Controller::disableCache() + * + * @param array|string $redirectTo + * @param bool $full if client side redirect should be a full page reload or not + * + * @return void + */ + public function clientSideRedirect(array|string $redirectTo, bool $full = false): void + { + $response = $this->controller->getResponse(); + if (is_array($redirectTo)) { + $redirectTo = Router::url($redirectTo); + } + $header = $full ? 'HX-Redirect' : 'HX-Location'; + $response = $response->withHeader($header, $redirectTo); + + $this->controller->setResponse($response); + } + + /** + * Set headers to cache this request. + * Opposite of Controller::disableCache() + * + * @param array|string $redirectTo + * @param bool $full if client side redirect should be a full page reload or not + * + * @return void + */ + public function clientSideRefresh(): void + { + $response = $this->controller->getResponse(); + $response = $response->withHeader('HX-Refresh', 'true'); + + $this->controller->setResponse($response); + } + + /** + * @return void + */ + public function indexHtmx() + { + if ($this->isHtmx() && $this->getHtmxTarget() == 'table-container') { + return $this->render('/element/' . $this->request->getParam('controller') . '/index/table', 'ajax'); + } + } +} diff --git a/templates/bake/Template/Element/table.twig b/templates/bake/Template/Element/table.twig new file mode 100644 index 0000000..4b0a6c2 --- /dev/null +++ b/templates/bake/Template/Element/table.twig @@ -0,0 +1,78 @@ +{# +/** + * CakePHP(tm) : Rapid Development Framework (https://cakephp.org) + * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * + * Licensed under The MIT License + * For full copyright and license information, please see the LICENSE.txt + * Redistributions of files must retain the above copyright notice. + * + * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * @link https://cakephp.org CakePHP(tm) Project + * @since 2.0.0 + * @license https://www.opensource.org/licenses/mit-license.php MIT License + */ +#} + ${{ pluralVar }} + */ +?> +
    +
    + + + +{% for field in fields %} + +{% endfor %} + + + + + + +{% for field in fields %} + {% set isKey = false %} +{% if associations.BelongsTo is defined %} +{% for alias, details in associations.BelongsTo %} +{% if field == details.foreignKey %} + {% set isKey = true %} + +{% endif %} +{% endfor %} +{% endif %} +{% if isKey is not same as(true) %} + {% set columnData = Bake.columnData(field, schema) %} +{% if columnData.type not in ['integer', 'float', 'decimal', 'biginteger', 'smallinteger', 'tinyinteger'] %} + +{% elseif columnData.null %} + +{% else %} + +{% endif %} +{% endif %} +{% endfor %} +{% set pk = '$' ~ singularVar ~ '->' ~ primaryKey[0] %} + + + + +
    Paginator->sort('{{ field }}') ?>
    hasValue('{{ details.property }}') ? $this->Html->link(${{ singularVar }}->{{ details.property }}->{{ details.displayField }}, ['controller' => '{{ details.controller }}', 'action' => 'view', ${{ singularVar }}->{{ details.property }}->{{ details.primaryKey[0] }}]) : '' ?>{{ field }}) ?>{{ field }} === null ? '' : $this->Number->format(${{ singularVar }}->{{ field }}) ?>Number->format(${{ singularVar }}->{{ field }}) ?> + Html->link(__('View'), ['action' => 'view', {{ pk|raw }}]) ?> + Html->link(__('Edit'), ['action' => 'edit', {{ pk|raw }}]) ?> + Form->postLink(__('Delete'), ['action' => 'delete', {{ pk|raw }}], ['confirm' => __('Are you sure you want to delete # {0}?', {{ pk|raw }})]) ?> +
    +
    +
    +
      + Paginator->first('<< ' . __('first')) ?> + Paginator->prev('< ' . __('previous')) ?> + Paginator->numbers() ?> + Paginator->next(__('next') . ' >') ?> + Paginator->last(__('last') . ' >>') ?> +
    +

    Paginator->counter(__('Page {{ '{{' }}page{{ '}}' }} of {{ '{{' }}pages{{ '}}' }}, showing {{ '{{' }}current{{ '}}' }} record(s) out of {{ '{{' }}count{{ '}}' }} total')) ?>

    +
    +
    diff --git a/templates/bake/Template/add.twig b/templates/bake/Template/add.twig new file mode 100644 index 0000000..c21d7c1 --- /dev/null +++ b/templates/bake/Template/add.twig @@ -0,0 +1,35 @@ +{# +/** + * CakePHP(tm) : Rapid Development Framework (https://cakephp.org) + * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * + * Licensed under The MIT License + * For full copyright and license information, please see the LICENSE.txt + * Redistributions of files must retain the above copyright notice. + * + * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * @link https://cakephp.org CakePHP(tm) Project + * @since 2.0.0 + * @license https://www.opensource.org/licenses/mit-license.php MIT License + */ +#} + +{{ element('Bake.form') }} \ No newline at end of file diff --git a/templates/bake/Template/edit.twig b/templates/bake/Template/edit.twig new file mode 100644 index 0000000..90fed4e --- /dev/null +++ b/templates/bake/Template/edit.twig @@ -0,0 +1,35 @@ +{# +/** + * CakePHP(tm) : Rapid Development Framework (https://cakephp.org) + * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * + * Licensed under The MIT License + * For full copyright and license information, please see the LICENSE.txt + * Redistributions of files must retain the above copyright notice. + * + * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * @link https://cakephp.org CakePHP(tm) Project + * @since 2.0.0 + * @license https://www.opensource.org/licenses/mit-license.php MIT License + */ +#} + +{{ element('Bake.form') }} \ No newline at end of file diff --git a/templates/bake/Template/index.twig b/templates/bake/Template/index.twig new file mode 100644 index 0000000..ee247d5 --- /dev/null +++ b/templates/bake/Template/index.twig @@ -0,0 +1,28 @@ +{# +/** + * CakePHP(tm) : Rapid Development Framework (https://cakephp.org) + * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * + * Licensed under The MIT License + * For full copyright and license information, please see the LICENSE.txt + * Redistributions of files must retain the above copyright notice. + * + * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * @link https://cakephp.org CakePHP(tm) Project + * @since 2.0.0 + * @license https://www.opensource.org/licenses/mit-license.php MIT License + */ +#} + ${{ pluralVar }} + */ +?> +
    +{% set fields = Bake.filterFields(fields, schema, modelObject, indexColumns, ['binary', 'text']) %} + Html->link(__('New {{ singularHumanName }}'), ['action' => 'add'], ['class' => 'button float-right']) ?> +{% set done = [] %} +

    + element('{{ controllerName }}/index/table'); ?> +
    diff --git a/templates/bake/Template/view.twig b/templates/bake/Template/view.twig new file mode 100644 index 0000000..6f8f255 --- /dev/null +++ b/templates/bake/Template/view.twig @@ -0,0 +1,141 @@ +{# +/** + * CakePHP(tm) : Rapid Development Framework (https://cakephp.org) + * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * + * Licensed under The MIT License + * For full copyright and license information, please see the LICENSE.txt + * Redistributions of files must retain the above copyright notice. + * + * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org) + * @link https://cakephp.org CakePHP(tm) Project + * @since 2.0.0 + * @license https://www.opensource.org/licenses/mit-license.php MIT License + */ +#} + +{% set associations = {'BelongsTo': [], 'HasOne': [], 'HasMany': [], 'BelongsToMany': []}|merge(associations) %} +{% set fieldsData = Bake.getViewFieldsData(fields, schema, associations) %} +{% set associationFields = fieldsData.associationFields %} +{% set groupedFields = fieldsData.groupedFields %} +{% set pK = '$' ~ singularVar ~ '->' ~ primaryKey[0] %} +
    + +
    +
    +

    {{ displayField }}) ?>

    + +{% if groupedFields['string'] %} +{% for field in groupedFields['string'] %} +{% if associationFields[field] is defined %} +{% set details = associationFields[field] %} + + + + +{% else %} + + + + +{% endif %} +{% endfor %} +{% endif %} +{% if associations.HasOne %} +{% for alias, details in associations.HasOne %} + + + + +{% endfor %} +{% endif %} +{% if groupedFields.number %} +{% for field in groupedFields.number %} + + +{% set columnData = Bake.columnData(field, schema) %} +{% if columnData.null %} + +{% else %} + +{% endif %} + +{% endfor %} +{% endif %} +{% if groupedFields.date %} +{% for field in groupedFields.date %} + + + + +{% endfor %} +{% endif %} +{% if groupedFields.boolean %} +{% for field in groupedFields.boolean %} + + + + +{% endfor %} +{% endif %} +
    hasValue('{{ details.property }}') ? $this->Html->link(${{ singularVar }}->{{ details.property }}->{{ details.displayField }}, ['controller' => '{{ details.controller }}', 'action' => 'view', ${{ singularVar }}->{{ details.property }}->{{ details.primaryKey[0] }}]) : '' ?>
    {{ field }}) ?>
    hasValue('{{ details.property }}') ? $this->Html->link(${{ singularVar }}->{{ details.property }}->{{ details.displayField }}, ['controller' => '{{ details.controller }}', 'action' => 'view', ${{ singularVar }}->{{ details.property }}->{{ details.primaryKey[0] }}]) : '' ?>
    {{ field }} === null ? '' : $this->Number->format(${{ singularVar }}->{{ field }}) ?>Number->format(${{ singularVar }}->{{ field }}) ?>
    {{ field }}) ?>
    {{ field }} ? __('Yes') : __('No'); ?>
    +{% if groupedFields.text %} +{% for field in groupedFields.text %} +
    + +
    + Text->autoParagraph(h(${{ singularVar }}->{{ field }})); ?> +
    +
    +{% endfor %} +{% endif %} +{% set relations = associations.BelongsToMany|merge(associations.HasMany) %} +{% for alias, details in relations %} +{% set otherSingularVar = alias|variable %} +{% set otherPluralHumanName = details.controller|underscore|humanize %} + +{% endfor %} +
    +
    +
    diff --git a/tests/bootstrap.php b/tests/bootstrap.php new file mode 100644 index 0000000..ddf23a7 --- /dev/null +++ b/tests/bootstrap.php @@ -0,0 +1,55 @@ +loadSqlFiles('tests/schema.sql', 'test'); diff --git a/tests/schema.sql b/tests/schema.sql new file mode 100644 index 0000000..cb878dd --- /dev/null +++ b/tests/schema.sql @@ -0,0 +1 @@ +-- Test database schema for CakeHtmx diff --git a/webroot/.gitkeep b/webroot/.gitkeep new file mode 100644 index 0000000..e69de29