src/Table/Column/Link.php

Summary

Maintainability
A
0 mins
Test Coverage
<?php

declare(strict_types=1);

namespace Atk4\Ui\Table\Column;

use Atk4\Data\Field;
use Atk4\Data\Model;
use Atk4\Ui\HtmlTemplate;
use Atk4\Ui\Table;

/**
 * Implements Column helper for grid.
 * Use
 *   new Link('https://yahoo.com?id={$id}');
 * or
 *   new Link(['order', 'id' => 'id' ]);
 * or
 *   new Link(['order', 'id' ]);
 * or
 *   new Link([['order', 'x' => $myval], 'id']);.
 */
class Link extends Table\Column
{
    /**
     * If $url is set up, we will use pattern-matching to fill-in any part of this
     * with values of the model.
     *
     * @var string|HtmlTemplate
     */
    public $url;

    /**
     * If string 'example', then will be passed to $app->url('example') along with any defined arguments.
     * If set as array, then non-0 key/values will be also passed to the URL:
     *  $page = ['example', 'type' => '123'];.
     *
     * $url = $app->url(['example', 'type' => '123']);
     *
     * In addition to above "args" refer to values picked up from a current row.
     *
     * @var string|array<0|string, string|int|false>|null
     */
    public $page;

    /**
     * When constructing a URL using 'page', this specifies list of values which will be added
     * to the destination URL. For example if you set $args = ['document_id' => 'id'] then row value
     * of ['id'] will be added to URL's property "document_id".
     *
     * For a full example:
     *  $page = ['example', 'type' => 'client'];
     *  $args = ['contact_id' => 'id'];
     *
     * Link URL will be "example.php?type=client&contact_id=4"
     *
     * You can also pass non-key arguments ['id', 'title'] and they will be added up
     * as ?id=4&title=John%20Smith
     *
     * @var array<int|string, string>
     */
    public $args = [];

    /** @var bool use value as label of the link */
    public $useLabel = true;

    /** @var string|null set element class. */
    public $class;

    /** @var string|null Use icon as label of the link. */
    public $icon;

    /**
     * Set html5 target attribute in tag
     * possible values: _blank | _parent | _self | _top | frame#name.
     *
     * @var string|null
     */
    public $target;

    /** @var bool add download in the tag to force download from the URL. */
    public $forceDownload = false;

    /**
     * @param string|array<0|string, string|int|false> $page
     * @param array<int|string, string>                $args
     * @param array<string, mixed>                     $defaults
     */
    public function __construct($page = [], array $args = [], array $defaults = [])
    {
        if (is_array($page)) {
            $defaults['page'] = $page;
        } else {
            $defaults['url'] = $page;
        }

        $defaults['args'] = $args;

        parent::__construct($defaults);
    }

    #[\Override]
    protected function init(): void
    {
        parent::init();

        if (is_string($this->url)) {
            $this->url = new HtmlTemplate($this->url);
        }
        if (is_string($this->page)) {
            $this->page = [$this->page];
        }
    }

    #[\Override]
    public function getDataCellTemplate(?Field $field = null): string
    {
        $attr = ['href' => '{$c_' . $this->shortName . '}'];

        if ($this->forceDownload) {
            $attr['download'] = 'true';
        }

        if ($this->target) {
            $attr['target'] = $this->target;
        }

        $iconHtml = '';
        if ($this->icon) {
            $iconHtml = $this->getApp()->getTag('i', ['class' => $this->icon . ' icon'], '');
        }

        $labelHtml = '';
        if ($this->useLabel) {
            $labelHtml = $field ? ('{$' . $field->shortName . '}') : '[Link]';
        }

        if ($this->class) {
            $attr['class'] = $this->class;
        }

        return $this->getApp()->getTag('a', $attr, [$iconHtml, $labelHtml]);
    }

    #[\Override]
    public function getHtmlTags(Model $row, ?Field $field): array
    {
        if ($this->url) {
            $this->url->trySet($this->getApp()->uiPersistence->typecastSaveRow($row, $row->get()));

            return ['c_' . $this->shortName => $this->url->renderToHtml()];
        }

        $page = $this->page ?? [];

        foreach ($this->args as $key => $val) {
            if (is_int($key)) {
                $key = $val;
            }

            $page[$key] = $this->getApp()->uiPersistence->typecastAttributeSaveField($row->getField($val), $row->get($val));
        }

        return ['c_' . $this->shortName => $this->table->url($page)];
    }
}