2009-01-13 16:44:09 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2009-08-25 23:12:20 +01:00
|
|
|
* StatusNet, the distributed open-source microblogging tool
|
2009-01-13 16:44:09 +00:00
|
|
|
*
|
|
|
|
* Low-level generator for HTML
|
|
|
|
*
|
|
|
|
* PHP version 5
|
|
|
|
*
|
|
|
|
* LICENCE: This program 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.
|
|
|
|
*
|
|
|
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* @category Output
|
2009-08-25 23:12:20 +01:00
|
|
|
* @package StatusNet
|
2009-08-25 23:19:04 +01:00
|
|
|
* @author Evan Prodromou <evan@status.net>
|
|
|
|
* @author Sarven Capadisli <csarven@status.net>
|
2009-08-25 23:12:20 +01:00
|
|
|
* @copyright 2008 StatusNet, Inc.
|
2009-01-13 16:44:09 +00:00
|
|
|
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
|
2009-08-25 23:16:46 +01:00
|
|
|
* @link http://status.net/
|
2009-01-13 16:44:09 +00:00
|
|
|
*/
|
|
|
|
|
2009-08-26 15:41:36 +01:00
|
|
|
if (!defined('STATUSNET') && !defined('LACONICA')) {
|
2009-01-13 16:44:09 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
require_once INSTALLDIR.'/lib/xmloutputter.php';
|
|
|
|
|
|
|
|
define('PAGE_TYPE_PREFS',
|
|
|
|
'text/html,application/xhtml+xml,'.
|
|
|
|
'application/xml;q=0.3,text/xml;q=0.2');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* @category Output
|
2009-08-25 23:12:20 +01:00
|
|
|
* @package StatusNet
|
2009-08-25 23:19:04 +01:00
|
|
|
* @author Evan Prodromou <evan@status.net>
|
|
|
|
* @author Sarven Capadisli <csarven@status.net>
|
2009-01-13 16:44:09 +00:00
|
|
|
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
|
2009-08-25 23:16:46 +01:00
|
|
|
* @link http://status.net/
|
2009-01-13 18:38:58 +00:00
|
|
|
*
|
2009-01-13 16:44:09 +00:00
|
|
|
* @see Action
|
2009-01-13 18:38:58 +00:00
|
|
|
* @see XMLOutputter
|
2009-01-13 16:44:09 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
class HTMLOutputter extends XMLOutputter
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
* Just wraps the XMLOutputter constructor.
|
|
|
|
*
|
|
|
|
* @param string $output URI to output to, default = stdout
|
|
|
|
* @param boolean $indent Whether to indent output, default true
|
|
|
|
*/
|
|
|
|
|
|
|
|
function __construct($output='php://output', $indent=true)
|
|
|
|
{
|
|
|
|
parent::__construct($output, $indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start an HTML document
|
|
|
|
*
|
|
|
|
* If $type isn't specified, will attempt to do content negotiation.
|
|
|
|
*
|
|
|
|
* Attempts to do content negotiation for language, also.
|
|
|
|
*
|
|
|
|
* @param string $type MIME type to use; default is to do negotation.
|
|
|
|
*
|
|
|
|
* @todo extract content negotiation code to an HTTP module or class.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function startHTML($type=null)
|
|
|
|
{
|
|
|
|
if (!$type) {
|
|
|
|
$httpaccept = isset($_SERVER['HTTP_ACCEPT']) ?
|
|
|
|
$_SERVER['HTTP_ACCEPT'] : null;
|
|
|
|
|
|
|
|
// 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) {
|
2009-02-11 20:39:49 +00:00
|
|
|
throw new ClientException(_('This page is not available in a '.
|
|
|
|
'media type you accept'), 406);
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-30 11:51:59 +01:00
|
|
|
header('Content-Type: '.$type.'; charset=UTF-8');
|
2009-02-11 19:45:06 +00:00
|
|
|
|
2009-01-22 11:13:11 +00:00
|
|
|
$this->extraHeaders();
|
2009-08-05 23:55:47 +01:00
|
|
|
if( ! substr($type,0,strlen('text/html'))=='text/html' ){
|
|
|
|
// Browsers don't like it when <?xml it output for non-xhtml documents
|
|
|
|
$this->xw->startDocument('1.0', 'UTF-8');
|
|
|
|
}
|
2009-09-30 11:37:46 +01:00
|
|
|
$this->xw->writeDTD('html',
|
|
|
|
'-//W3C//DTD XHTML 1.0 Strict//EN',
|
|
|
|
'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd');
|
2009-01-13 16:44:09 +00:00
|
|
|
|
2009-02-11 19:45:06 +00:00
|
|
|
$language = $this->getLanguage();
|
2009-01-13 16:44:09 +00:00
|
|
|
|
|
|
|
$this->elementStart('html', array('xmlns' => 'http://www.w3.org/1999/xhtml',
|
|
|
|
'xml:lang' => $language,
|
|
|
|
'lang' => $language));
|
|
|
|
}
|
|
|
|
|
2009-02-11 19:45:06 +00:00
|
|
|
function getLanguage()
|
|
|
|
{
|
|
|
|
// FIXME: correct language for interface
|
|
|
|
return common_language();
|
|
|
|
}
|
|
|
|
|
2009-01-14 17:26:23 +00:00
|
|
|
/**
|
|
|
|
* Ends an HTML document
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function endHTML()
|
|
|
|
{
|
|
|
|
$this->elementEnd('html');
|
|
|
|
$this->endXML();
|
|
|
|
}
|
2009-02-11 19:45:06 +00:00
|
|
|
|
2009-01-22 11:13:11 +00:00
|
|
|
/**
|
|
|
|
* To specify additional HTTP headers for the action
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function extraHeaders()
|
|
|
|
{
|
|
|
|
// Needs to be overloaded
|
|
|
|
}
|
2009-01-14 17:26:23 +00:00
|
|
|
|
2009-01-13 16:44:09 +00: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.
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of label for the element
|
|
|
|
* @param string $value value of the element, default null
|
|
|
|
* @param string $instructions instructions for valid input
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
* @todo add a $maxLength parameter
|
|
|
|
* @todo add a $size parameter
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
|
|
|
function input($id, $label, $value=null, $instructions=null)
|
|
|
|
{
|
|
|
|
$this->element('label', array('for' => $id), $label);
|
|
|
|
$attrs = array('name' => $id,
|
|
|
|
'type' => 'text',
|
|
|
|
'id' => $id);
|
|
|
|
if ($value) {
|
2009-02-04 05:22:41 +00:00
|
|
|
$attrs['value'] = $value;
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
$this->element('input', $attrs);
|
|
|
|
if ($instructions) {
|
2009-01-16 23:41:46 +00:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML checkbox and associated elements
|
|
|
|
*
|
|
|
|
* Note that the value is default 'true' (the string), which can
|
|
|
|
* be used by Action::boolean()
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of label for the element
|
|
|
|
* @param string $checked if the box is checked, default false
|
|
|
|
* @param string $instructions instructions for valid input
|
|
|
|
* @param string $value value of the checkbox, default 'true'
|
|
|
|
* @param string $disabled show the checkbox disabled, default false
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
|
|
|
|
|
|
|
function checkbox($id, $label, $checked=false, $instructions=null,
|
|
|
|
$value='true', $disabled=false)
|
|
|
|
{
|
|
|
|
$attrs = array('name' => $id,
|
|
|
|
'type' => 'checkbox',
|
|
|
|
'class' => 'checkbox',
|
|
|
|
'id' => $id);
|
|
|
|
if ($value) {
|
2009-02-04 05:22:41 +00:00
|
|
|
$attrs['value'] = $value;
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
if ($checked) {
|
|
|
|
$attrs['checked'] = 'checked';
|
|
|
|
}
|
|
|
|
if ($disabled) {
|
|
|
|
$attrs['disabled'] = 'true';
|
|
|
|
}
|
|
|
|
$this->element('input', $attrs);
|
|
|
|
$this->text(' ');
|
2009-01-17 00:01:53 +00:00
|
|
|
$this->element('label', array('class' => 'checkbox',
|
2009-01-13 16:44:09 +00:00
|
|
|
'for' => $id),
|
|
|
|
$label);
|
|
|
|
$this->text(' ');
|
|
|
|
if ($instructions) {
|
2009-01-16 23:41:46 +00:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-13 16:44:09 +00: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.)
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of label for the element
|
|
|
|
* @param array $content options array, value => text
|
|
|
|
* @param string $instructions instructions for valid input
|
|
|
|
* @param string $blank_select whether to have a blank entry, default false
|
|
|
|
* @param string $selected selected value, default null
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
|
|
|
|
|
|
|
function dropdown($id, $label, $content, $instructions=null,
|
|
|
|
$blank_select=false, $selected=null)
|
|
|
|
{
|
|
|
|
$this->element('label', array('for' => $id), $label);
|
|
|
|
$this->elementStart('select', array('id' => $id, 'name' => $id));
|
|
|
|
if ($blank_select) {
|
|
|
|
$this->element('option', array('value' => ''));
|
|
|
|
}
|
|
|
|
foreach ($content as $value => $option) {
|
|
|
|
if ($value == $selected) {
|
|
|
|
$this->element('option', array('value' => $value,
|
2009-02-09 15:35:38 +00:00
|
|
|
'selected' => 'selected'),
|
2009-01-13 16:44:09 +00:00
|
|
|
$option);
|
|
|
|
} else {
|
|
|
|
$this->element('option', array('value' => $value), $option);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->elementEnd('select');
|
|
|
|
if ($instructions) {
|
2009-01-16 23:41:46 +00:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML hidden element
|
|
|
|
*
|
|
|
|
* $id is re-used as name
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $value hidden element value, default null
|
2009-01-14 06:04:09 +00:00
|
|
|
* @param string $name name, if different than ID
|
2009-01-13 16:44:09 +00:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
|
2009-01-14 06:04:09 +00:00
|
|
|
function hidden($id, $value, $name=null)
|
2009-01-13 16:44:09 +00:00
|
|
|
{
|
2009-01-14 06:04:09 +00:00
|
|
|
$this->element('input', array('name' => ($name) ? $name : $id,
|
2009-01-13 16:44:09 +00:00
|
|
|
'type' => 'hidden',
|
|
|
|
'id' => $id,
|
|
|
|
'value' => $value));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML password input and associated elements
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of label for the element
|
|
|
|
* @param string $instructions instructions for valid input
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
|
|
|
|
|
|
|
function password($id, $label, $instructions=null)
|
|
|
|
{
|
|
|
|
$this->element('label', array('for' => $id), $label);
|
|
|
|
$attrs = array('name' => $id,
|
|
|
|
'type' => 'password',
|
|
|
|
'class' => 'password',
|
|
|
|
'id' => $id);
|
|
|
|
$this->element('input', $attrs);
|
|
|
|
if ($instructions) {
|
2009-01-16 23:41:46 +00:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output an HTML submit input and associated elements
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of the button
|
|
|
|
* @param string $cls class of the button, default 'submit'
|
2009-01-14 06:04:09 +00:00
|
|
|
* @param string $name name, if different than ID
|
2009-01-13 16:44:09 +00:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
*/
|
|
|
|
|
2009-01-17 21:09:42 +00:00
|
|
|
function submit($id, $label, $cls='submit', $name=null, $title=null)
|
2009-01-13 16:44:09 +00:00
|
|
|
{
|
|
|
|
$this->element('input', array('type' => 'submit',
|
|
|
|
'id' => $id,
|
2009-01-14 06:04:09 +00:00
|
|
|
'name' => ($name) ? $name : $id,
|
2009-01-13 16:44:09 +00:00
|
|
|
'class' => $cls,
|
2009-01-17 20:57:48 +00:00
|
|
|
'value' => $label,
|
2009-01-17 21:09:42 +00:00
|
|
|
'title' => $title));
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
|
2009-08-06 00:35:42 +01:00
|
|
|
/**
|
|
|
|
* output a script (almost always javascript) tag
|
|
|
|
*
|
|
|
|
* @param string $src relative or absolute script path
|
|
|
|
* @param string $type 'type' attribute value of the tag
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function script($src, $type='text/javascript')
|
|
|
|
{
|
2009-08-06 01:28:46 +01:00
|
|
|
$url = parse_url($src);
|
2009-08-09 03:56:42 +01:00
|
|
|
if( empty($url->scheme) && empty($url->host) && empty($url->query) && empty($url->fragment))
|
2009-08-06 01:28:46 +01:00
|
|
|
{
|
2009-08-25 23:42:34 +01:00
|
|
|
$src = common_path($src) . '?version=' . STATUSNET_VERSION;
|
2009-08-06 01:28:46 +01:00
|
|
|
}
|
2009-08-06 00:35:42 +01:00
|
|
|
$this->element('script', array('type' => $type,
|
2009-08-06 01:28:46 +01:00
|
|
|
'src' => $src),
|
2009-08-06 00:35:42 +01:00
|
|
|
' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* output a css link
|
|
|
|
*
|
2009-08-06 01:28:46 +01:00
|
|
|
* @param string $src relative path within the theme directory, or an absolute path
|
2009-08-06 00:35:42 +01:00
|
|
|
* @param string $theme 'theme' that contains the stylesheet
|
|
|
|
* @param string media 'media' attribute of the tag
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2009-08-06 16:18:57 +01:00
|
|
|
function cssLink($src,$theme=null,$media=null)
|
2009-08-06 00:35:42 +01:00
|
|
|
{
|
2009-08-06 01:28:46 +01:00
|
|
|
$url = parse_url($src);
|
2009-08-09 03:56:42 +01:00
|
|
|
if( empty($url->scheme) && empty($url->host) && empty($url->query) && empty($url->fragment))
|
2009-08-06 01:28:46 +01:00
|
|
|
{
|
2009-08-06 18:05:40 +01:00
|
|
|
if(file_exists(theme_file($src,$theme))){
|
2009-08-25 23:42:34 +01:00
|
|
|
$src = theme_path($src, $theme) . '?version=' . STATUSNET_VERSION;
|
2009-08-06 18:05:40 +01:00
|
|
|
}else{
|
|
|
|
$src = common_path($src);
|
|
|
|
}
|
2009-08-06 01:28:46 +01:00
|
|
|
}
|
2009-08-06 00:35:42 +01:00
|
|
|
$this->element('link', array('rel' => 'stylesheet',
|
|
|
|
'type' => 'text/css',
|
2009-08-06 01:28:46 +01:00
|
|
|
'href' => $src,
|
2009-08-06 00:35:42 +01:00
|
|
|
'media' => $media));
|
|
|
|
}
|
|
|
|
|
2009-01-13 16:44:09 +00:00
|
|
|
/**
|
|
|
|
* output an HTML textarea and associated elements
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must be unique on page
|
|
|
|
* @param string $label text of label for the element
|
|
|
|
* @param string $content content of the textarea, default none
|
|
|
|
* @param string $instructions instructions for valid input
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
* @todo add a $name parameter
|
|
|
|
* @todo add a $cols parameter
|
|
|
|
* @todo add a $rows parameter
|
|
|
|
*/
|
|
|
|
|
|
|
|
function textarea($id, $label, $content=null, $instructions=null)
|
|
|
|
{
|
|
|
|
$this->element('label', array('for' => $id), $label);
|
|
|
|
$this->element('textarea', array('rows' => 3,
|
|
|
|
'cols' => 40,
|
|
|
|
'name' => $id,
|
|
|
|
'id' => $id),
|
|
|
|
($content) ? $content : '');
|
|
|
|
if ($instructions) {
|
2009-01-16 23:41:46 +00:00
|
|
|
$this->element('p', 'form_guide', $instructions);
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-03 20:42:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal script to autofocus the given element on page onload.
|
|
|
|
*
|
|
|
|
* @param string $id element ID, must refer to an existing element
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
function autofocus($id)
|
|
|
|
{
|
|
|
|
$this->elementStart('script', array('type' => 'text/javascript'));
|
|
|
|
$this->raw('
|
|
|
|
<!--
|
|
|
|
$(document).ready(function() {
|
|
|
|
var el = $("#' . $id . '");
|
|
|
|
if (el.length) {
|
|
|
|
el.focus();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
-->
|
|
|
|
');
|
|
|
|
$this->elementEnd('script');
|
|
|
|
}
|
2009-01-13 16:44:09 +00:00
|
|
|
}
|