CodeIgniter4/system/Helpers/html_helper.php

785 lines
18 KiB
PHP
Raw Normal View History

2016-11-07 17:41:46 +08:00
<?php
2020-07-29 01:47:24 +08:00
2016-11-07 17:41:46 +08:00
/**
* CodeIgniter
*
* An open source application development framework for PHP
*
* This content is released under the MIT License (MIT)
*
2018-12-26 21:49:51 -08:00
* Copyright (c) 2014-2019 British Columbia Institute of Technology
2020-02-06 22:58:39 -06:00
* Copyright (c) 2019-2020 CodeIgniter Foundation
2016-11-07 17:41:46 +08:00
*
* 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.
*
* @package CodeIgniter
* @author CodeIgniter Dev Team
2020-02-06 22:58:39 -06:00
* @copyright 2019-2020 CodeIgniter Foundation
* @license https://opensource.org/licenses/MIT MIT License
2017-01-20 00:45:11 -08:00
* @link https://codeigniter.com
2019-05-04 00:58:04 -07:00
* @since Version 4.0.0
2016-11-07 17:41:46 +08:00
* @filesource
*/
2018-11-10 02:57:39 -02:00
2020-10-07 17:48:51 +00:00
use CodeIgniter\Files\Exceptions\FileNotFoundException;
2020-10-04 00:27:56 +07:00
use Config\DocTypes;
2020-10-07 17:48:51 +00:00
use Config\Mimes;
2016-11-07 17:41:46 +08:00
// --------------------------------------------------------------------
/**
* CodeIgniter HTML Helpers
2016-11-07 17:41:46 +08:00
*
2019-05-05 23:53:08 -07:00
* @package CodeIgniter
2016-11-07 17:41:46 +08:00
*/
2018-11-10 02:57:39 -02:00
if (! function_exists('ul'))
2016-11-07 17:41:46 +08:00
{
/**
* Unordered List
*
2017-11-04 05:49:12 -02:00
* Generates an HTML unordered list from an single or
* multi-dimensional array.
*
2020-07-29 01:47:24 +08:00
* @param array $list
* @param mixed $attributes HTML attributes string, array, object
*
2018-11-10 02:57:39 -02:00
* @return string
*/
function ul(array $list, $attributes = ''): string
{
return _list('ul', $list, $attributes);
}
2016-11-07 17:41:46 +08:00
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('ol'))
2016-11-07 17:41:46 +08:00
{
/**
* Ordered List
*
* Generates an HTML ordered list from an single or multi-dimensional array.
*
2020-07-29 01:47:24 +08:00
* @param array $list
* @param mixed $attributes HTML attributes string, array, object
*
2018-11-10 02:57:39 -02:00
* @return string
*/
function ol(array $list, $attributes = ''): string
{
return _list('ol', $list, $attributes);
}
2016-11-07 17:41:46 +08:00
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('_list'))
2016-11-07 17:41:46 +08:00
{
/**
* Generates the list
*
* Generates an HTML ordered list from an single or multi-dimensional array.
*
2020-07-29 01:47:24 +08:00
* @param string $type
* @param mixed $list
* @param mixed $attributes string, array, object
* @param integer $depth
*
2018-11-10 02:57:39 -02:00
* @return string
*/
function _list(string $type = 'ul', $list = [], $attributes = '', int $depth = 0): string
{
// Set the indentation based on the depth
$out = str_repeat(' ', $depth)
// Write the opening list tag
. '<' . $type . stringify_attributes($attributes) . ">\n";
// Cycle through the list elements. If an array is
// encountered we will recursively call _list()
static $_last_list_item = '';
foreach ($list as $key => $val)
{
$_last_list_item = $key;
$out .= str_repeat(' ', $depth + 2) . '<li>';
2018-11-10 02:57:39 -02:00
if (! is_array($val))
{
$out .= $val;
}
else
{
$out .= $_last_list_item
. "\n"
. _list($type, $val, '', $depth + 4)
. str_repeat(' ', $depth + 2);
}
$out .= "</li>\n";
}
// Set the indentation for the closing tag and apply it
return $out . str_repeat(' ', $depth) . '</' . $type . ">\n";
}
2016-11-07 17:41:46 +08:00
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('img'))
2016-11-07 17:41:46 +08:00
{
/**
* Image
*
* Generates an image element
*
2020-10-07 17:48:51 +00:00
* @param string|array $src Image source URI, or array of attributes and values
* @param boolean $indexPage Whether to treat $src as a routed URI string
* @param string|array|object $attributes Additional HTML attributes
2020-07-29 01:47:24 +08:00
*
2018-11-10 02:57:39 -02:00
* @return string
*/
2018-06-29 12:59:46 -07:00
function img($src = '', bool $indexPage = false, $attributes = ''): string
{
2018-11-10 02:57:39 -02:00
if (! is_array($src))
{
$src = ['src' => $src];
}
2020-10-07 17:48:51 +00:00
if (! isset($src['src']))
{
$src['src'] = $attributes['src'] ?? '';
}
2018-11-10 02:57:39 -02:00
if (! isset($src['alt']))
{
2018-11-27 22:44:47 +01:00
$src['alt'] = $attributes['alt'] ?? '';
}
$img = '<img';
2020-10-07 17:48:51 +00:00
// Check for a relative URI
if (! preg_match('#^([a-z]+:)?//#i', $src['src']) && strpos($src['src'], 'data:') !== 0)
{
2020-10-07 17:48:51 +00:00
if ($indexPage === true)
{
2020-10-07 17:48:51 +00:00
$img .= ' src="' . site_url($src['src']) . '"';
}
else
{
2020-10-07 17:48:51 +00:00
$img .= ' src="' . slash_item('baseURL') . $src['src'] . '"';
}
2020-10-07 17:48:51 +00:00
unset($src['src']);
}
2020-10-07 17:48:51 +00:00
// Append any other values
foreach ($src as $key => $value)
{
2020-10-07 17:48:51 +00:00
$img .= ' ' . $key . '="' . $value . '"';
}
// Prevent passing completed values to stringify_attributes
if (is_array($attributes))
{
unset($attributes['alt'], $attributes['src']);
}
return $img . stringify_attributes($attributes) . ' />';
}
2016-11-07 17:41:46 +08:00
}
2020-10-07 17:48:51 +00:00
if (! function_exists('img_data'))
{
/**
* Image (data)
*
* Generates a src-ready string from an image using the "data:" protocol
*
* @param string $path Image source path
* @param string|null $mime MIME type to use, or null to guess
*
* @return string
*/
function img_data(string $path, string $mime = null): string
{
if (! is_file($path) || ! is_readable($path))
{
throw FileNotFoundException::forFileNotFound($path);
}
// Read in file binary data
$handle = fopen($path, 'rb');
$data = fread($handle, filesize($path));
fclose($handle);
// Encode as base64
$data = base64_encode($data);
// Figure out the type (Hail Mary to JPEG)
$mime = $mime ?? Mimes::guessTypeFromExtension(pathinfo($path, PATHINFO_EXTENSION)) ?? 'image/jpg';
return 'data:' . $mime . ';base64,' . $data;
}
}
2016-11-07 17:41:46 +08:00
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('doctype'))
2016-11-07 17:41:46 +08:00
{
/**
* Doctype
*
* Generates a page document type declaration
*
* Examples of valid options: html5, xhtml-11, xhtml-strict, xhtml-trans,
* xhtml-frame, html4-strict, html4-trans, and html4-frame.
* All values are saved in the doctypes config file.
*
2020-07-29 01:47:24 +08:00
* @param string $type The doctype to be generated
*
2018-11-10 02:57:39 -02:00
* @return string
*/
2018-06-29 12:59:46 -07:00
function doctype(string $type = 'html5'): string
{
2020-10-04 00:27:56 +07:00
$config = new DocTypes();
2018-06-29 12:59:46 -07:00
$doctypes = $config->list;
2017-12-13 00:54:31 -02:00
return $doctypes[$type] ?? false;
}
2016-11-07 17:41:46 +08:00
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('script_tag'))
2016-11-07 17:41:46 +08:00
{
/**
* Script
*
* Generates link to a JS file
*
2020-07-29 01:47:24 +08:00
* @param mixed $src Script source or an array
* @param boolean $indexPage Should indexPage be added to the JS path
*
2018-11-10 02:57:39 -02:00
* @return string
*/
2018-06-29 12:59:46 -07:00
function script_tag($src = '', bool $indexPage = false): string
{
$script = '<script ';
2018-11-10 02:57:39 -02:00
if (! is_array($src))
2018-06-29 12:59:46 -07:00
{
$src = ['src' => $src];
}
2018-06-29 12:59:46 -07:00
foreach ($src as $k => $v)
{
2018-06-29 12:59:46 -07:00
if ($k === 'src' && ! preg_match('#^([a-z]+:)?//#i', $v))
{
2018-06-29 12:59:46 -07:00
if ($indexPage === true)
{
2018-06-29 12:59:46 -07:00
$script .= 'src="' . site_url($v) . '" ';
}
else
{
2018-06-29 12:59:46 -07:00
$script .= 'src="' . slash_item('baseURL') . $v . '" ';
}
}
else
{
2018-06-29 12:59:46 -07:00
$script .= $k . '="' . $v . '" ';
}
}
2018-11-10 02:57:39 -02:00
return $script . 'type="text/javascript"' . '></script>';
}
2016-11-07 17:41:46 +08:00
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('link_tag'))
2016-11-07 17:41:46 +08:00
{
/**
* Link
*
* Generates link to a CSS file
*
2020-07-29 01:47:24 +08:00
* @param mixed $href Stylesheet href or an array
* @param string $rel
* @param string $type
* @param string $title
* @param string $media
* @param boolean $indexPage should indexPage be added to the CSS path.
* @param string $hreflang
*
2018-11-10 02:57:39 -02:00
* @return string
*/
function link_tag($href = '', string $rel = 'stylesheet', string $type = 'text/css', string $title = '', string $media = '', bool $indexPage = false, string $hreflang = ''): string
{
$link = '<link ';
2018-06-29 12:59:46 -07:00
// extract fields if needed
if (is_array($href))
{
2018-11-10 02:57:39 -02:00
$rel = $href['rel'] ?? $rel;
$type = $href['type'] ?? $type;
$title = $href['title'] ?? $title;
$media = $href['media'] ?? $media;
2020-07-29 01:47:24 +08:00
$hreflang = $href['hreflang'] ?? '';
2018-06-29 12:59:46 -07:00
$indexPage = $href['indexPage'] ?? $indexPage;
2018-11-10 02:57:39 -02:00
$href = $href['href'] ?? '';
}
2018-06-29 12:59:46 -07:00
2018-11-10 02:57:39 -02:00
if (! preg_match('#^([a-z]+:)?//#i', $href))
{
2018-06-29 12:59:46 -07:00
if ($indexPage === true)
{
$link .= 'href="' . site_url($href) . '" ';
}
else
{
$link .= 'href="' . slash_item('baseURL') . $href . '" ';
}
2018-06-29 12:59:46 -07:00
}
else
2018-11-10 02:57:39 -02:00
{
2018-06-29 12:59:46 -07:00
$link .= 'href="' . $href . '" ';
2018-11-10 02:57:39 -02:00
}
if ($hreflang !== '')
{
2020-07-29 01:47:24 +08:00
$link .= 'hreflang="' . $hreflang . '" ';
}
$link .= 'rel="' . $rel . '" ';
if (! in_array($rel, ['alternate', 'canonical'], true))
{
2020-07-29 01:47:24 +08:00
$link .= 'type="' . $type . '" ';
}
2018-06-29 12:59:46 -07:00
if ($media !== '')
{
$link .= 'media="' . $media . '" ';
}
2018-06-29 12:59:46 -07:00
if ($title !== '')
{
$link .= 'title="' . $title . '" ';
}
2018-11-10 02:57:39 -02:00
return $link . '/>';
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('video'))
{
/**
* Video
*
2019-03-10 17:48:18 +05:30
* Generates a video element to embed videos. The video element can
* contain one or more video sources
*
2020-07-29 01:47:24 +08:00
* @param mixed $src Either a source string or an array of sources
* @param string $unsupportedMessage The message to display if the media tag is not supported by the browser
* @param string $attributes HTML attributes
* @param array $tracks
* @param boolean $indexPage
*
* @return string
*/
function video($src, string $unsupportedMessage = '', string $attributes = '', array $tracks = [], bool $indexPage = false): string
{
if (is_array($src))
{
return _media('video', $src, $unsupportedMessage, $attributes, $tracks);
}
2017-12-31 06:28:29 -02:00
$video = '<video';
if (_has_protocol($src))
{
$video .= ' src="' . $src . '"';
}
elseif ($indexPage === true)
{
$video .= ' src="' . site_url($src) . '"';
}
else
{
$video .= ' src="' . slash_item('baseURL') . $src . '"';
}
if ($attributes !== '')
{
$video .= ' ' . $attributes;
}
$video .= ">\n";
2018-11-10 02:57:39 -02:00
if (! empty($tracks))
{
foreach ($tracks as $track)
2017-12-31 06:28:29 -02:00
{
$video .= _space_indent() . $track . "\n";
2017-12-31 06:28:29 -02:00
}
}
2018-11-10 02:57:39 -02:00
if (! empty($unsupportedMessage))
{
$video .= _space_indent()
. $unsupportedMessage
. "\n";
}
return $video . "</video>\n";
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('audio'))
{
/**
* Audio
*
* Generates an audio element to embed sounds
*
2018-11-10 02:57:39 -02:00
* @param mixed $src Either a source string or an array of sources
* @param string $unsupportedMessage The message to display if the media tag is not supported by the browser.
* @param string $attributes HTML attributes
* @param array $tracks
2018-11-10 02:57:39 -02:00
* @param boolean $indexPage
*
* @return string
*/
function audio($src, string $unsupportedMessage = '', string $attributes = '', array $tracks = [], bool $indexPage = false): string
{
if (is_array($src))
{
return _media('audio', $src, $unsupportedMessage, $attributes, $tracks);
}
2017-12-31 06:28:29 -02:00
$audio = '<audio';
if (_has_protocol($src))
{
$audio .= ' src="' . $src . '"';
}
elseif ($indexPage === true)
{
$audio .= ' src="' . site_url($src) . '"';
}
else
{
$audio .= ' src="' . slash_item('baseURL') . $src . '"';
}
if ($attributes !== '')
{
$audio .= ' ' . $attributes;
}
$audio .= '>';
2018-11-10 02:57:39 -02:00
if (! empty($tracks))
{
foreach ($tracks as $track)
2017-12-31 06:28:29 -02:00
{
$audio .= "\n" . _space_indent() . $track;
2017-12-31 06:28:29 -02:00
}
}
2018-11-10 02:57:39 -02:00
if (! empty($unsupportedMessage))
{
$audio .= "\n" . _space_indent() . $unsupportedMessage . "\n";
}
return $audio . "</audio>\n";
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('_media'))
{
2019-03-10 17:48:18 +05:30
/**
* Generate media based tag
*
* @param string $name
* @param array $types
2019-05-05 07:46:30 -07:00
* @param string $unsupportedMessage The message to display if the media tag is not supported by the browser.
2019-03-10 17:48:18 +05:30
* @param string $attributes
* @param array $tracks
*
* @return string
*/
function _media(string $name, array $types = [], string $unsupportedMessage = '', string $attributes = '', array $tracks = []): string
{
$media = '<' . $name;
if (empty($attributes))
{
$media .= '>';
}
else
{
$media .= ' ' . $attributes . '>';
}
$media .= "\n";
foreach ($types as $option)
{
$media .= _space_indent() . $option . "\n";
}
2018-11-10 02:57:39 -02:00
if (! empty($tracks))
{
foreach ($tracks as $track)
{
$media .= _space_indent() . $track . "\n";
}
}
2018-11-10 02:57:39 -02:00
if (! empty($unsupportedMessage))
{
$media .= _space_indent() . $unsupportedMessage . "\n";
}
return $media . ('</' . $name . ">\n");
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('source'))
{
/**
* Source
*
* Generates a source element that specifies multiple media resources
* for either audio or video element
*
2020-07-29 01:47:24 +08:00
* @param string $src The path of the media resource
* @param string $type The MIME-type of the resource with optional codecs parameters
* @param string $attributes HTML attributes
* @param boolean $indexPage
*
* @return string
*/
function source(string $src, string $type = 'unknown', string $attributes = '', bool $indexPage = false): string
{
2018-11-10 02:57:39 -02:00
if (! _has_protocol($src))
{
if ($indexPage === true)
{
$src = site_url($src);
}
else
{
$src = slash_item('baseURL') . $src;
}
2018-11-10 02:57:39 -02:00
}
$source = '<source src="' . $src
. '" type="' . $type . '"';
2018-11-10 02:57:39 -02:00
if (! empty($attributes))
{
$source .= ' ' . $attributes;
}
return $source . ' />';
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('track'))
{
/**
* Track
*
* Generates a track element to specify timed tracks. The tracks are
* formatted in WebVTT format.
*
2020-07-29 01:47:24 +08:00
* @param string $src The path of the .VTT file
* @param string $kind
* @param string $srcLanguage
* @param string $label
*
* @return string
*/
function track(string $src, string $kind, string $srcLanguage, string $label): string
{
return '<track src="' . $src
. '" kind="' . $kind
. '" srclang="' . $srcLanguage
. '" label="' . $label
. '" />';
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('object'))
{
/**
* Object
*
* Generates an object element that represents the media
* as either image or a resource plugin such as audio, video,
* Java applets, ActiveX, PDF and Flash
*
2018-11-10 02:57:39 -02:00
* @param string $data A resource URL
* @param string $type Content-type of the resource
* @param string $attributes HTML attributes
* @param array $params
* @param boolean $indexPage
*
* @return string
*/
function object(string $data, string $type = 'unknown', string $attributes = '', array $params = [], bool $indexPage = false): string
{
2018-11-10 02:57:39 -02:00
if (! _has_protocol($data))
{
if ($indexPage === true)
{
$data = site_url($data);
}
else
{
$data = slash_item('baseURL') . $data;
}
2018-11-10 02:57:39 -02:00
}
$object = '<object data="' . $data . '" '
. $attributes . '>';
2018-11-10 02:57:39 -02:00
if (! empty($params))
{
$object .= "\n";
}
foreach ($params as $param)
{
$object .= _space_indent() . $param . "\n";
}
return $object . "</object>\n";
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('param'))
{
/**
* Param
*
* Generates a param element that defines parameters
* for the object element.
*
2020-07-29 01:47:24 +08:00
* @param string $name The name of the parameter
* @param string $value The value of the parameter
* @param string $type The MIME-type
* @param string $attributes HTML attributes
*
* @return string
*/
function param(string $name, string $value, string $type = 'ref', string $attributes = ''): string
{
return '<param name="' . $name
. '" type="' . $type
. '" value="' . $value
. '" ' . $attributes . ' />';
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('embed'))
{
/**
* Embed
*
* Generates an embed element
*
2020-07-29 01:47:24 +08:00
* @param string $src The path of the resource to embed
* @param string $type MIME-type
* @param string $attributes HTML attributes
* @param boolean $indexPage
*
* @return string
*/
2018-11-10 02:57:39 -02:00
function embed(string $src, string $type = 'unknown', string $attributes = '', bool $indexPage = false): string
{
2018-11-10 02:57:39 -02:00
if (! _has_protocol($src))
{
if ($indexPage === true)
{
$src = site_url($src);
}
else
{
$src = slash_item('baseURL') . $src;
}
2018-11-10 02:57:39 -02:00
}
return '<embed src="' . $src
. '" type="' . $type . '" '
. $attributes . " />\n";
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('_has_protocol'))
{
2019-03-10 17:48:18 +05:30
/**
2019-05-05 07:46:30 -07:00
* Test the protocol of a URI.
*
2019-03-10 17:48:18 +05:30
* @param string $url
*
* @return false|integer
2019-03-10 17:48:18 +05:30
*/
function _has_protocol(string $url)
{
return preg_match('#^([a-z]+:)?//#i', $url);
}
}
// ------------------------------------------------------------------------
2018-11-10 02:57:39 -02:00
if (! function_exists('_space_indent'))
{
2019-03-10 17:48:18 +05:30
/**
2019-05-05 07:46:30 -07:00
* Provide space indenting.
*
* @param integer $depth
2019-03-10 17:48:18 +05:30
*
* @return string
*/
2019-04-17 23:36:14 +05:30
function _space_indent(int $depth = 2): string
{
return str_repeat(' ', $depth);
}
2016-11-07 17:41:46 +08:00
}
2017-11-04 05:49:12 -02:00
// ------------------------------------------------------------------------