2009-01-14 01:44:09 +09:00
|
|
|
<?php
|
2020-04-03 01:22:24 +09:00
|
|
|
// This file is part of GNU social - https://www.gnu.org/software/social
|
|
|
|
//
|
|
|
|
// GNU social is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// GNU social is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2009-01-14 01:44:09 +09:00
|
|
|
/**
|
|
|
|
* Low-level generator for HTML
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @package GNUsocial
|
2009-01-14 01:44:09 +09:00
|
|
|
* @category Output
|
2020-04-03 01:22:24 +09:00
|
|
|
*
|
2009-08-26 07:19:04 +09:00
|
|
|
* @author Evan Prodromou <evan@status.net>
|
|
|
|
* @author Sarven Capadisli <csarven@status.net>
|
2020-04-03 01:22:24 +09:00
|
|
|
* @copyright 2008-2019 Free Software Foundation, Inc http://www.fsf.org
|
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
2009-01-14 01:44:09 +09:00
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
defined('GNUSOCIAL') || die();
|
2009-01-14 01:44:09 +09:00
|
|
|
|
2009-11-21 09:34:01 +09:00
|
|
|
// Can include XHTML options but these are too fragile in practice.
|
|
|
|
define('PAGE_TYPE_PREFS', 'text/html');
|
2009-01-14 01:44:09 +09:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Low-level generator for HTML
|
|
|
|
*
|
|
|
|
* Abstracts some of the code necessary for HTML generation. Especially
|
|
|
|
* has methods for generating HTML form elements. Note that these have
|
|
|
|
* been created kind of haphazardly, not with an eye to making a general
|
|
|
|
* HTML-creation class.
|
|
|
|
*
|
|
|
|
* @see Action
|
2009-01-14 03:38:58 +09:00
|
|
|
* @see XMLOutputter
|
2020-04-03 01:22:24 +09:00
|
|
|
*
|
|
|
|
* @copyright 2008-2019 Free Software Foundation, Inc http://www.fsf.org
|
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
2009-01-14 01:44:09 +09:00
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
class htmloutputter extends XMLOutputter
|
2009-01-14 01:44:09 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
protected $DTD = ['doctype' => 'html',
|
2020-04-03 01:22:24 +09:00
|
|
|
'spec' => '-//W3C//DTD XHTML 1.0 Strict//EN',
|
|
|
|
'uri' => 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd', ];
|
2019-05-07 07:58:45 +09:00
|
|
|
|
2009-01-14 01:44:09 +09:00
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
* Just wraps the XMLOutputter constructor.
|
|
|
|
*
|
2019-05-07 07:58:45 +09:00
|
|
|
* @param string $output URI to output to, default = stdout
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param bool|null $indent Whether to indent output, if null it defaults to true
|
|
|
|
*
|
|
|
|
* @throws ServerException
|
2009-01-14 01:44:09 +09:00
|
|
|
*/
|
2019-05-07 07:58:45 +09:00
|
|
|
public function __construct($output = 'php://output', $indent = null)
|
2009-01-14 01:44:09 +09:00
|
|
|
{
|
|
|
|
parent::__construct($output, $indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start an HTML document
|
|
|
|
*
|
2009-11-10 04:01:46 +09:00
|
|
|
* If $type isn't specified, will attempt to do content negotiation.
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* Attempts to do content negotiation for language, also.
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param null|string $type MIME type to use; default is to do negotation.
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
2019-05-07 07:58:45 +09:00
|
|
|
* @throws ClientException
|
2020-04-03 01:22:24 +09:00
|
|
|
* @throws ServerException
|
|
|
|
*
|
|
|
|
* @return void
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @todo extract content negotiation code to an HTTP module or class.
|
2009-01-14 01:44:09 +09:00
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function startHTML(?string $type = null): void
|
2009-01-14 01:44:09 +09:00
|
|
|
{
|
|
|
|
if (!$type) {
|
|
|
|
$httpaccept = isset($_SERVER['HTTP_ACCEPT']) ?
|
2019-05-07 07:58:45 +09:00
|
|
|
$_SERVER['HTTP_ACCEPT'] : null;
|
2009-01-14 01:44:09 +09:00
|
|
|
|
|
|
|
// XXX: allow content negotiation for RDF, RSS, or XRDS
|
|
|
|
|
|
|
|
$cp = common_accept_to_prefs($httpaccept);
|
|
|
|
$sp = common_accept_to_prefs(PAGE_TYPE_PREFS);
|
|
|
|
|
|
|
|
$type = common_negotiate_type($cp, $sp);
|
|
|
|
|
|
|
|
if (!$type) {
|
2010-04-11 04:12:14 +09:00
|
|
|
// TRANS: Client exception 406
|
2019-05-07 07:58:45 +09:00
|
|
|
throw new ClientException(_('This page is not available in a ' .
|
|
|
|
'media type you accept'), 406);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
header('Content-Type: ' . $type);
|
2009-02-12 04:45:06 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
// Output anti-framing headers to prevent clickjacking (respected by newer
|
2011-06-06 15:08:17 +09:00
|
|
|
// browsers).
|
2019-05-07 07:58:45 +09:00
|
|
|
if (common_config('javascript', 'bustframes')) {
|
2011-08-05 23:42:18 +09:00
|
|
|
header('X-XSS-Protection: 1; mode=block'); // detect XSS Reflection attacks
|
2011-06-06 15:08:17 +09:00
|
|
|
header('X-Frame-Options: SAMEORIGIN'); // no rendering if origin mismatch
|
|
|
|
}
|
|
|
|
|
2009-01-22 20:13:11 +09:00
|
|
|
$this->extraHeaders();
|
2020-04-03 01:22:24 +09:00
|
|
|
if (preg_match('/.*\\/.*xml/', $type)) {
|
2009-09-30 19:53:04 +09:00
|
|
|
// Required for XML documents
|
2013-09-24 08:17:04 +09:00
|
|
|
$this->startXML();
|
2009-08-06 07:55:47 +09:00
|
|
|
}
|
2015-04-23 02:44:58 +09:00
|
|
|
|
|
|
|
$this->writeDTD();
|
2009-01-14 01:44:09 +09:00
|
|
|
|
2009-02-12 04:45:06 +09:00
|
|
|
$language = $this->getLanguage();
|
2009-01-14 01:44:09 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
$attrs = [
|
2020-04-03 01:22:24 +09:00
|
|
|
'xmlns' => 'http://www.w3.org/1999/xhtml',
|
2010-10-30 08:38:00 +09:00
|
|
|
'xml:lang' => $language,
|
2020-04-03 01:22:24 +09:00
|
|
|
'lang' => $language,
|
2019-05-07 07:58:45 +09:00
|
|
|
];
|
2010-10-30 08:38:00 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
if (Event::handle('StartHtmlElement', [$this, &$attrs])) {
|
2010-10-30 08:38:00 +09:00
|
|
|
$this->elementStart('html', $attrs);
|
2019-05-07 07:58:45 +09:00
|
|
|
Event::handle('EndHtmlElement', [$this, &$attrs]);
|
2010-10-30 08:38:00 +09:00
|
|
|
}
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
/**
|
|
|
|
* To specify additional HTTP headers for the action
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function extraHeaders()
|
2015-04-23 02:44:58 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
// Needs to be overloaded
|
2015-04-23 02:44:58 +09:00
|
|
|
}
|
|
|
|
|
2020-04-03 01:22:24 +09:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
protected function writeDTD(): void
|
2015-04-23 02:44:58 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->xw->writeDTD(
|
|
|
|
$this->DTD['doctype'],
|
|
|
|
$this->DTD['spec'],
|
|
|
|
$this->DTD['uri']
|
|
|
|
);
|
2015-04-23 02:44:58 +09:00
|
|
|
}
|
|
|
|
|
2020-04-03 01:22:24 +09:00
|
|
|
/**
|
|
|
|
* @return mixed (array|bool|string)
|
|
|
|
*/
|
2019-05-07 07:58:45 +09:00
|
|
|
public function getLanguage()
|
2009-02-12 04:45:06 +09:00
|
|
|
{
|
|
|
|
// FIXME: correct language for interface
|
|
|
|
return common_language();
|
|
|
|
}
|
|
|
|
|
2020-04-03 01:22:24 +09:00
|
|
|
/**
|
|
|
|
* @param $doctype
|
|
|
|
* @param $spec
|
|
|
|
* @param $uri
|
|
|
|
*/
|
|
|
|
public function setDTD($doctype, $spec, $uri): void
|
2009-01-15 02:26:23 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->DTD = ['doctype' => $doctype, 'spec' => $spec, 'uri' => $uri];
|
2009-01-15 02:26:23 +09:00
|
|
|
}
|
2009-02-12 04:45:06 +09:00
|
|
|
|
2009-01-22 20:13:11 +09:00
|
|
|
/**
|
2019-05-07 07:58:45 +09:00
|
|
|
* Ends an HTML document
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function endHTML(): void
|
2009-01-22 20:13:11 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->elementEnd('html');
|
|
|
|
$this->endXML();
|
2009-01-22 20:13:11 +09:00
|
|
|
}
|
2009-01-15 02:26:23 +09:00
|
|
|
|
2009-01-14 01:44:09 +09:00
|
|
|
/**
|
|
|
|
* Output an HTML text input element
|
|
|
|
*
|
|
|
|
* Despite the name, it is specifically for outputting a
|
|
|
|
* text input element, not other <input> elements. It outputs
|
|
|
|
* a cluster of elements, including a <label> and an associated
|
|
|
|
* instructions span.
|
|
|
|
*
|
2015-02-05 04:52:10 +09:00
|
|
|
* If $attrs['type'] does not exist it will be set to 'text'.
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of label for the element
|
|
|
|
* @param null|string $value value of the element, default null
|
|
|
|
* @param null|string $instructions instructions for valid input
|
|
|
|
* @param null|string $name name of the element; if null, the id will be used
|
|
|
|
* @param bool $required HTML5 required attribute (exclude when false)
|
|
|
|
* @param array $attrs Initial attributes manually set in an array (overwritten by previous options)
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
2019-05-07 07:58:45 +09:00
|
|
|
* @return void
|
2020-04-03 01:22:24 +09:00
|
|
|
*
|
2009-01-14 01:44:09 +09:00
|
|
|
* @todo add a $maxLength parameter
|
|
|
|
* @todo add a $size parameter
|
|
|
|
*
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function input(
|
|
|
|
string $id,
|
|
|
|
?string $label,
|
|
|
|
?string $value = null,
|
|
|
|
?string $instructions = null,
|
|
|
|
?string $name = null,
|
|
|
|
bool $required = false,
|
|
|
|
array $attrs = []
|
|
|
|
): void {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('label', ['for' => $id], $label);
|
2015-02-05 04:52:10 +09:00
|
|
|
if (!array_key_exists('type', $attrs)) {
|
|
|
|
$attrs['type'] = 'text';
|
|
|
|
}
|
2020-04-03 01:22:24 +09:00
|
|
|
$attrs['id'] = $id;
|
2011-04-18 06:29:38 +09:00
|
|
|
$attrs['name'] = is_null($name) ? $id : $name;
|
2015-03-09 07:29:16 +09:00
|
|
|
if (array_key_exists('placeholder', $attrs) && (is_null($attrs['placeholder']) || $attrs['placeholder'] === '')) {
|
|
|
|
// If placeholder is type-aware equal to '' or null, unset it as we apparently don't want a placeholder value
|
|
|
|
unset($attrs['placeholder']);
|
|
|
|
} else {
|
|
|
|
// If the placeholder is set use it, or use the label as fallback.
|
|
|
|
$attrs['placeholder'] = isset($attrs['placeholder']) ? $attrs['placeholder'] : $label;
|
|
|
|
}
|
|
|
|
|
2010-04-26 15:40:36 +09:00
|
|
|
if (!is_null($value)) { // value can be 0 or ''
|
2009-02-04 14:22:41 +09:00
|
|
|
$attrs['value'] = $value;
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
2014-04-06 18:27:41 +09:00
|
|
|
if (!empty($required)) {
|
|
|
|
$attrs['required'] = 'required';
|
|
|
|
}
|
2009-01-14 01:44:09 +09:00
|
|
|
$this->element('input', $attrs);
|
|
|
|
if ($instructions) {
|
2009-01-17 08:41:46 +09:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML checkbox and associated elements
|
|
|
|
*
|
|
|
|
* Note that the value is default 'true' (the string), which can
|
|
|
|
* be used by Action::boolean()
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of label for the element
|
|
|
|
* @param bool $checked if the box is checked, default false
|
|
|
|
* @param null|string $instructions instructions for valid input
|
|
|
|
* @param string $value value of the checkbox, default 'true'
|
|
|
|
* @param bool $disabled show the checkbox disabled, default false
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
2019-05-07 07:58:45 +09:00
|
|
|
public function checkbox(
|
2020-04-03 01:22:24 +09:00
|
|
|
string $id,
|
|
|
|
?string $label,
|
|
|
|
bool $checked = false,
|
|
|
|
?string $instructions = null,
|
|
|
|
string $value = 'true',
|
|
|
|
bool $disabled = false
|
|
|
|
): void {
|
2019-05-07 07:58:45 +09:00
|
|
|
$attrs = ['name' => $id,
|
2020-04-03 01:22:24 +09:00
|
|
|
'type' => 'checkbox',
|
|
|
|
'class' => 'checkbox',
|
|
|
|
'id' => $id, ];
|
2009-01-14 01:44:09 +09:00
|
|
|
if ($value) {
|
2009-02-04 14:22:41 +09:00
|
|
|
$attrs['value'] = $value;
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
if ($checked) {
|
|
|
|
$attrs['checked'] = 'checked';
|
|
|
|
}
|
|
|
|
if ($disabled) {
|
|
|
|
$attrs['disabled'] = 'true';
|
|
|
|
}
|
|
|
|
$this->element('input', $attrs);
|
|
|
|
$this->text(' ');
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element(
|
|
|
|
'label',
|
2020-04-03 01:22:24 +09:00
|
|
|
['class' => 'checkbox',
|
|
|
|
'for' => $id, ],
|
2019-05-07 07:58:45 +09:00
|
|
|
$label
|
|
|
|
);
|
2009-01-14 01:44:09 +09:00
|
|
|
$this->text(' ');
|
|
|
|
if ($instructions) {
|
2009-01-17 08:41:46 +09:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML combobox/select and associated elements
|
|
|
|
*
|
|
|
|
* $content is an array of key-value pairs for the dropdown, where
|
|
|
|
* the key is the option value attribute and the value is the option
|
|
|
|
* text. (Careful on the overuse of 'value' here.)
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of label for the element
|
|
|
|
* @param array $content options array, value => text
|
|
|
|
* @param null|string $instructions instructions for valid input
|
|
|
|
* @param bool $blank_select whether to have a blank entry, default false
|
|
|
|
* @param null|string $selected selected value, default null
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
2019-05-07 07:58:45 +09:00
|
|
|
public function dropdown(
|
2020-04-03 01:22:24 +09:00
|
|
|
string $id,
|
|
|
|
?string $label,
|
|
|
|
array $content,
|
|
|
|
?string $instructions = null,
|
|
|
|
bool $blank_select = false,
|
|
|
|
?string $selected = null
|
|
|
|
): void {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('label', ['for' => $id], $label);
|
|
|
|
$this->elementStart('select', ['id' => $id, 'name' => $id]);
|
2009-01-14 01:44:09 +09:00
|
|
|
if ($blank_select) {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('option', ['value' => '']);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
foreach ($content as $value => $option) {
|
|
|
|
if ($value == $selected) {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element(
|
|
|
|
'option',
|
2020-04-03 01:22:24 +09:00
|
|
|
['value' => $value,
|
|
|
|
'selected' => 'selected', ],
|
2019-05-07 07:58:45 +09:00
|
|
|
$option
|
|
|
|
);
|
2009-01-14 01:44:09 +09:00
|
|
|
} else {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('option', ['value' => $value], $option);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->elementEnd('select');
|
|
|
|
if ($instructions) {
|
2009-01-17 08:41:46 +09:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML hidden element
|
|
|
|
*
|
|
|
|
* $id is re-used as name
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $value hidden element value, default null
|
|
|
|
* @param null|string $name name, if different than ID
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function hidden(string $id, ?string $value = null, ?string $name = null)
|
2009-01-14 01:44:09 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('input', ['name' => $name ?: $id,
|
2020-04-03 01:22:24 +09:00
|
|
|
'type' => 'hidden',
|
|
|
|
'id' => $id,
|
|
|
|
'value' => $value, ]);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML password input and associated elements
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of label for the element
|
|
|
|
* @param null|string $instructions instructions for valid input
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function password(string $id, ?string $label, ?string $instructions = null): void
|
2009-01-14 01:44:09 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('label', ['for' => $id], $label);
|
|
|
|
$attrs = ['name' => $id,
|
2020-04-03 01:22:24 +09:00
|
|
|
'type' => 'password',
|
|
|
|
'class' => 'password',
|
|
|
|
'id' => $id, ];
|
2009-01-14 01:44:09 +09:00
|
|
|
$this->element('input', $attrs);
|
|
|
|
if ($instructions) {
|
2009-01-17 08:41:46 +09:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML submit input and associated elements
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of the button
|
|
|
|
* @param string $cls class of the button, default 'submit'
|
|
|
|
* @param null|string $name name, if different than ID
|
|
|
|
* @param null|string $title title text for the submit button
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function submit(
|
|
|
|
string $id,
|
|
|
|
?string $label,
|
|
|
|
string $cls = 'submit',
|
|
|
|
?string $name = null,
|
|
|
|
?string $title = null
|
|
|
|
): void {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('input', ['type' => 'submit',
|
2020-04-03 01:22:24 +09:00
|
|
|
'id' => $id,
|
|
|
|
'name' => $name ?: $id,
|
|
|
|
'class' => $cls,
|
|
|
|
'value' => $label,
|
|
|
|
'title' => $title, ]);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
|
2009-08-06 08:35:42 +09:00
|
|
|
/**
|
|
|
|
* output a script (almost always javascript) tag
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $src relative or absolute script path
|
2019-05-07 07:58:45 +09:00
|
|
|
* @param string $type 'type' attribute value of the tag
|
2009-08-06 08:35:42 +09:00
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @throws ServerException
|
|
|
|
*
|
2009-08-06 08:35:42 +09:00
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function script(string $src, string $type = 'text/javascript'): void
|
2009-08-06 08:35:42 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
if (Event::handle('StartScriptElement', [$this, &$src, &$type])) {
|
2009-12-04 15:57:14 +09:00
|
|
|
$url = parse_url($src);
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2010-10-14 14:00:13 +09:00
|
|
|
if (empty($url['scheme']) && empty($url['host']) && empty($url['query']) && empty($url['fragment'])) {
|
|
|
|
|
|
|
|
// XXX: this seems like a big assumption
|
|
|
|
|
2010-02-18 02:03:14 +09:00
|
|
|
if (strpos($src, 'plugins/') === 0 || strpos($src, 'local/') === 0) {
|
2015-02-27 20:44:15 +09:00
|
|
|
$src = common_path($src, GNUsocial::isHTTPS()) . '?version=' . GNUSOCIAL_VERSION;
|
2010-10-14 14:00:13 +09:00
|
|
|
} else {
|
2015-02-27 20:44:15 +09:00
|
|
|
if (GNUsocial::isHTTPS()) {
|
2019-05-07 07:58:45 +09:00
|
|
|
$server = common_config('javascript', 'sslserver');
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
if (empty($server)) {
|
2020-04-03 01:22:24 +09:00
|
|
|
if (is_string(common_config('site', 'sslserver')) && mb_strlen(common_config('site', 'sslserver')) > 0) {
|
2010-10-14 14:35:11 +09:00
|
|
|
$server = common_config('site', 'sslserver');
|
2019-05-07 07:58:45 +09:00
|
|
|
} elseif (common_config('site', 'server')) {
|
2010-10-14 14:35:11 +09:00
|
|
|
$server = common_config('site', 'server');
|
|
|
|
}
|
2019-05-07 07:58:45 +09:00
|
|
|
$path = common_config('site', 'path') . '/js/';
|
2010-10-14 14:00:13 +09:00
|
|
|
} else {
|
2019-05-07 07:58:45 +09:00
|
|
|
$path = common_config('javascript', 'sslpath');
|
2010-10-14 14:00:13 +09:00
|
|
|
if (empty($path)) {
|
|
|
|
$path = common_config('javascript', 'path');
|
|
|
|
}
|
|
|
|
}
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2010-10-14 14:00:13 +09:00
|
|
|
$protocol = 'https';
|
|
|
|
} else {
|
|
|
|
$path = common_config('javascript', 'path');
|
2010-02-12 06:59:39 +09:00
|
|
|
|
2010-10-14 14:00:13 +09:00
|
|
|
if (empty($path)) {
|
2010-10-15 03:53:20 +09:00
|
|
|
$path = common_config('site', 'path') . '/js/';
|
2010-10-14 14:00:13 +09:00
|
|
|
}
|
2010-02-18 02:03:14 +09:00
|
|
|
|
2010-10-14 14:00:13 +09:00
|
|
|
$server = common_config('javascript', 'server');
|
|
|
|
|
|
|
|
if (empty($server)) {
|
|
|
|
$server = common_config('site', 'server');
|
2010-02-18 02:03:14 +09:00
|
|
|
}
|
2010-10-14 14:00:13 +09:00
|
|
|
|
|
|
|
$protocol = 'http';
|
2010-02-18 02:03:14 +09:00
|
|
|
}
|
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
if ($path[strlen($path) - 1] != '/') {
|
2010-10-14 14:00:13 +09:00
|
|
|
$path .= '/';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($path[0] != '/') {
|
2019-05-07 07:58:45 +09:00
|
|
|
$path = '/' . $path;
|
2010-10-14 14:00:13 +09:00
|
|
|
}
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
$src = $protocol . '://' . $server . $path . $src . '?version=' . GNUSOCIAL_VERSION;
|
2010-02-18 02:03:14 +09:00
|
|
|
}
|
2009-12-04 15:57:14 +09:00
|
|
|
}
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element(
|
|
|
|
'script',
|
2020-04-03 01:22:24 +09:00
|
|
|
['type' => $type,
|
|
|
|
'src' => $src, ],
|
2019-05-07 07:58:45 +09:00
|
|
|
' '
|
|
|
|
);
|
2010-01-28 01:37:22 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
Event::handle('EndScriptElement', [$this, $src, $type]);
|
2009-12-05 09:44:45 +09:00
|
|
|
}
|
2009-12-05 03:39:51 +09:00
|
|
|
}
|
|
|
|
|
2009-08-06 08:35:42 +09:00
|
|
|
/**
|
|
|
|
* output a css link
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $src relative path within the theme directory, or an absolute path
|
|
|
|
* @param string $theme 'theme' that contains the stylesheet
|
|
|
|
* @param null|string $media
|
|
|
|
*
|
|
|
|
* @throws ServerException
|
2009-08-06 08:35:42 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function cssLink(string $src, ?string $theme = null, ?string $media = null): void
|
2009-08-06 08:35:42 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
if (Event::handle('StartCssLinkElement', [$this, &$src, &$theme, &$media])) {
|
2009-12-04 15:57:14 +09:00
|
|
|
$url = parse_url($src);
|
2019-05-07 07:58:45 +09:00
|
|
|
if (empty($url['scheme']) && empty($url['host']) && empty($url['query']) && empty($url['fragment'])) {
|
|
|
|
if (file_exists(Theme::file($src, $theme))) {
|
|
|
|
$src = Theme::path($src, $theme);
|
|
|
|
} else {
|
2015-02-27 20:44:15 +09:00
|
|
|
$src = common_path($src, GNUsocial::isHTTPS());
|
2009-12-04 15:57:14 +09:00
|
|
|
}
|
2019-05-07 07:58:45 +09:00
|
|
|
$src .= '?version=' . GNUSOCIAL_VERSION;
|
2009-08-07 02:05:40 +09:00
|
|
|
}
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('link', ['rel' => 'stylesheet',
|
2020-04-03 01:22:24 +09:00
|
|
|
'type' => 'text/css',
|
|
|
|
'href' => $src,
|
|
|
|
'media' => $media, ]);
|
2019-05-07 07:58:45 +09:00
|
|
|
Event::handle('EndCssLinkElement', [$this, $src, $theme, $media]);
|
2009-08-06 09:28:46 +09:00
|
|
|
}
|
2009-08-06 08:35:42 +09:00
|
|
|
}
|
|
|
|
|
2009-12-05 10:19:55 +09:00
|
|
|
/**
|
|
|
|
* output a style (almost always css) tag with inline
|
|
|
|
* code.
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $code code to put in the style tag
|
|
|
|
* @param string $type 'type' attribute value of the tag
|
|
|
|
* @param null|string $media 'media' attribute value of the tag
|
2009-12-05 10:19:55 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function style(string $code, string $type = 'text/css', ?string $media = null)
|
2009-12-05 10:19:55 +09:00
|
|
|
{
|
2019-05-07 07:58:45 +09:00
|
|
|
if (Event::handle('StartStyleElement', [$this, &$code, &$type, &$media])) {
|
|
|
|
$this->elementStart('style', ['type' => $type, 'media' => $media]);
|
2009-12-05 10:19:55 +09:00
|
|
|
$this->raw($code);
|
|
|
|
$this->elementEnd('style');
|
2019-05-07 07:58:45 +09:00
|
|
|
Event::handle('EndStyleElement', [$this, $code, $type, $media]);
|
2009-12-05 10:19:55 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-14 01:44:09 +09:00
|
|
|
/**
|
|
|
|
* output an HTML textarea and associated elements
|
|
|
|
*
|
2020-04-03 01:22:24 +09:00
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param null|string $label text of label for the element
|
|
|
|
* @param null|string $content content of the textarea, default none
|
|
|
|
* @param null|string $instructions instructions for valid input
|
|
|
|
* @param null|string $name name of textarea; if null, $id will be used
|
|
|
|
* @param null|int $cols number of columns
|
|
|
|
* @param null|int $rows number of rows
|
|
|
|
* @param bool $required HTML5 required attribute (exclude when false)
|
2009-01-14 01:44:09 +09:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-05-07 07:58:45 +09:00
|
|
|
public function textarea(
|
2020-04-03 01:22:24 +09:00
|
|
|
string $id,
|
|
|
|
?string $label,
|
|
|
|
?string $content = null,
|
|
|
|
?string $instructions = null,
|
|
|
|
?string $name = null,
|
|
|
|
?int $cols = null,
|
|
|
|
?int $rows = null,
|
|
|
|
bool $required = false
|
|
|
|
): void {
|
2019-05-07 07:58:45 +09:00
|
|
|
$this->element('label', ['for' => $id], $label);
|
|
|
|
$attrs = [
|
2011-04-18 07:09:00 +09:00
|
|
|
'rows' => 3,
|
|
|
|
'cols' => 40,
|
2020-04-03 01:22:24 +09:00
|
|
|
'id' => $id,
|
2019-05-07 07:58:45 +09:00
|
|
|
];
|
2011-04-18 07:09:00 +09:00
|
|
|
$attrs['name'] = is_null($name) ? $id : $name;
|
|
|
|
|
|
|
|
if ($cols != null) {
|
|
|
|
$attrs['cols'] = $cols;
|
|
|
|
}
|
|
|
|
if ($rows != null) {
|
|
|
|
$attrs['rows'] = $rows;
|
|
|
|
}
|
2019-05-07 07:58:45 +09:00
|
|
|
|
|
|
|
if (!empty($required)) {
|
|
|
|
$attrs['required'] = 'required';
|
|
|
|
}
|
|
|
|
|
2011-04-18 07:09:00 +09:00
|
|
|
$this->element(
|
|
|
|
'textarea',
|
|
|
|
$attrs,
|
2019-05-07 07:58:45 +09:00
|
|
|
$content
|
2011-04-18 07:09:00 +09:00
|
|
|
);
|
2009-01-14 01:44:09 +09:00
|
|
|
if ($instructions) {
|
2009-01-17 08:41:46 +09:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|
|
|
|
}
|
2009-09-04 04:42:50 +09:00
|
|
|
|
2019-05-07 07:58:45 +09:00
|
|
|
/**
|
|
|
|
* Internal script to autofocus the given element on page onload.
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must refer to an existing element
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function autofocus(string $id): void
|
2009-09-04 04:42:50 +09:00
|
|
|
{
|
2009-12-05 10:33:41 +09:00
|
|
|
$this->inlineScript(
|
2019-05-07 07:58:45 +09:00
|
|
|
' $(document).ready(function() {' .
|
|
|
|
' var el = $("#' . $id . '");' .
|
|
|
|
' if (el.length) { el.focus(); }' .
|
|
|
|
' });'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output a script (almost always javascript) tag with inline
|
|
|
|
* code.
|
|
|
|
*
|
|
|
|
* @param string $code code to put in the script tag
|
|
|
|
* @param string $type 'type' attribute value of the tag
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2020-04-03 01:22:24 +09:00
|
|
|
public function inlineScript(string $code, string $type = 'text/javascript'): void
|
2019-05-07 07:58:45 +09:00
|
|
|
{
|
|
|
|
if (Event::handle('StartInlineScriptElement', [$this, &$code, &$type])) {
|
|
|
|
$this->elementStart('script', ['type' => $type]);
|
|
|
|
if ($type == 'text/javascript') {
|
|
|
|
$this->raw('/*<![CDATA[*/ '); // XHTML compat
|
|
|
|
}
|
|
|
|
$this->raw($code);
|
|
|
|
if ($type == 'text/javascript') {
|
|
|
|
$this->raw(' /*]]>*/'); // XHTML compat
|
|
|
|
}
|
|
|
|
$this->elementEnd('script');
|
|
|
|
Event::handle('EndInlineScriptElement', [$this, $code, $type]);
|
|
|
|
}
|
2009-09-04 04:42:50 +09:00
|
|
|
}
|
2009-01-14 01:44:09 +09:00
|
|
|
}
|