2001-12-01 15:20:48 +00:00
< ? php
2005-08-11 12:57:41 +00:00
// $Id$
2001-12-01 15:20:48 +00:00
2004-07-13 07:21:14 +00:00
/**
* @ file
* Common functions that many Drupal modules will need to reference .
*
* The functions that are critical and need to be available even when serving
* a cached page are instead located in bootstrap . inc .
*/
2005-05-07 01:48:06 +00:00
/**
* Return status for saving which involved creating a new item .
*/
define ( 'SAVED_NEW' , 1 );
/**
* Return status for saving which involved an update to an existing item .
*/
define ( 'SAVED_UPDATED' , 2 );
/**
* Return status for saving which deleted an existing item .
*/
define ( 'SAVED_DELETED' , 3 );
2005-08-16 18:06:18 +00:00
/**
* Set content for a specified region .
*
* @ param $region
* Page region the content is assigned to .
*
* @ param $data
* Content to be set .
*/
function drupal_set_content ( $region = null , $data = null ) {
static $content = array ();
if ( ! is_null ( $region ) && ! is_null ( $data )) {
$content [ $region ][] = $data ;
}
return $content ;
}
/**
* Get assigned content .
*
* @ param $region
* A specified region to fetch content for . If null , all regions will be returned .
*
* @ param $delimiter
* Content to be inserted between exploded array elements .
*/
function drupal_get_content ( $region = null , $delimiter = ' ' ) {
$content = drupal_set_content ();
if ( isset ( $region )) {
if ( is_array ( $content [ $region ])) {
return implode ( $delimiter , $content [ $region ]);
}
}
else {
foreach ( array_keys ( $content ) as $region ) {
if ( is_array ( $content [ $region ])) {
$content [ $region ] = implode ( $delimiter , $content [ $region ]);
}
}
return $content ;
}
}
2003-11-23 10:41:04 +00:00
/**
2004-09-09 05:51:08 +00:00
* Set the breadcrumb trail for the current page .
2003-12-08 06:32:19 +00:00
*
2004-09-09 05:51:08 +00:00
* @ param $breadcrumb
* Array of links , starting with " home " and proceeding up to but not including
* the current page .
2004-01-06 19:52:14 +00:00
*/
2003-11-23 10:41:04 +00:00
function drupal_set_breadcrumb ( $breadcrumb = NULL ) {
static $stored_breadcrumb ;
if ( isset ( $breadcrumb )) {
$stored_breadcrumb = $breadcrumb ;
}
return $stored_breadcrumb ;
}
2004-09-09 05:51:08 +00:00
/**
* Get the breadcrumb trail for the current page .
*/
2003-11-23 10:41:04 +00:00
function drupal_get_breadcrumb () {
$breadcrumb = drupal_set_breadcrumb ();
if ( ! isset ( $breadcrumb )) {
$breadcrumb = menu_get_active_breadcrumb ();
}
return $breadcrumb ;
}
2004-01-14 22:30:09 +00:00
/**
2004-09-09 05:51:08 +00:00
* Add output to the head tag of the HTML page .
2004-11-15 10:47:18 +00:00
* This function can be called as long the headers aren ' t sent .
2004-01-14 22:30:09 +00:00
*/
function drupal_set_html_head ( $data = NULL ) {
2004-04-12 08:27:57 +00:00
static $stored_head = '' ;
2004-01-14 22:30:09 +00:00
if ( ! is_null ( $data )) {
2004-07-13 07:21:14 +00:00
$stored_head .= $data . " \n " ;
2004-01-14 22:30:09 +00:00
}
return $stored_head ;
}
2004-09-09 05:51:08 +00:00
/**
* Retrieve output to be displayed in the head tag of the HTML page .
*/
2004-01-14 22:30:09 +00:00
function drupal_get_html_head () {
global $base_url ;
2004-01-29 06:47:19 +00:00
$output = " <meta http-equiv= \" Content-Type \" content= \" text/html; charset=utf-8 \" /> \n " ;
2004-01-14 22:30:09 +00:00
$output .= " <base href= \" $base_url / \" /> \n " ;
2004-08-20 07:51:27 +00:00
$output .= theme ( 'stylesheet_import' , 'misc/drupal.css' );
2004-01-14 22:30:09 +00:00
return $output . drupal_set_html_head ();
}
2004-09-09 05:51:08 +00:00
/**
2005-05-14 09:23:47 +00:00
* Reset the static variable which holds the aliases mapped for this request .
2004-09-09 05:51:08 +00:00
*/
2005-05-14 09:23:47 +00:00
function drupal_clear_path_cache () {
drupal_lookup_path ( 'wipe' );
2003-10-03 14:10:05 +00:00
}
2004-01-06 19:52:14 +00:00
2004-02-08 17:12:44 +00:00
/**
2004-07-13 07:21:14 +00:00
* Given a path alias , return the internal path it represents .
2004-02-08 17:12:44 +00:00
*/
function drupal_get_normal_path ( $path ) {
2005-05-14 09:23:47 +00:00
//drupal_get_path_alias($path);
if ( $src = drupal_lookup_path ( 'alias' , $path )) {
return $src ;
2004-02-08 17:12:44 +00:00
}
2004-07-13 07:21:14 +00:00
elseif ( function_exists ( 'conf_url_rewrite' )) {
2004-02-08 17:12:44 +00:00
return conf_url_rewrite ( $path , 'incoming' );
}
else {
return $path ;
}
}
2004-01-06 19:52:14 +00:00
2004-01-14 22:30:09 +00:00
/**
2004-09-09 05:51:08 +00:00
* Set an HTTP response header for the current page .
2004-01-14 22:30:09 +00:00
*/
function drupal_set_header ( $header = NULL ) {
2004-07-29 01:41:33 +00:00
// We use an array to guarantee there are no leading or trailing delimiters.
2004-09-09 05:51:08 +00:00
// Otherwise, header('') could get called when serving the page later, which
2004-07-29 01:41:33 +00:00
// ends HTTP headers prematurely on some PHP versions.
static $stored_headers = array ();
2004-01-14 22:30:09 +00:00
2004-07-29 01:41:33 +00:00
if ( strlen ( $header )) {
2004-01-14 22:30:09 +00:00
header ( $header );
2004-07-29 01:41:33 +00:00
$stored_headers [] = $header ;
2004-01-14 22:30:09 +00:00
}
2004-07-29 01:41:33 +00:00
return implode ( " \n " , $stored_headers );
2004-01-14 22:30:09 +00:00
}
2004-09-09 05:51:08 +00:00
/**
* Get the HTTP response headers for the current page .
*/
2004-01-14 22:30:09 +00:00
function drupal_get_headers () {
return drupal_set_header ();
}
2004-02-08 17:12:44 +00:00
/**
* @ name HTTP handling
* @ {
2004-09-09 05:51:08 +00:00
* Functions to properly handle HTTP responses .
2004-02-08 17:12:44 +00:00
*/
2005-02-01 19:45:58 +00:00
/**
* Prepare a destination query string for use in combination with
2005-07-20 10:48:20 +00:00
* drupal_goto () . Used to direct the user back to the referring page
* after completing a form . By default the current URL is returned .
* If a destination exists in the previous request , that destination
* is returned . As such , a destination can persist across multiple
* pages .
2005-02-01 19:45:58 +00:00
*
* @ see drupal_goto ()
*/
function drupal_get_destination () {
2005-07-20 10:48:20 +00:00
if ( $_REQUEST [ 'destination' ]) {
return 'destination=' . urlencode ( $_REQUEST [ 'destination' ]);
}
else {
$destination [] = $_GET [ 'q' ];
$params = array ( 'page' , 'sort' , 'order' );
foreach ( $params as $param ) {
if ( isset ( $_GET [ $param ])) {
$destination [] = " $param = " . $_GET [ $param ];
}
2005-02-01 19:45:58 +00:00
}
2005-07-20 10:48:20 +00:00
return 'destination=' . urlencode ( implode ( '&' , $destination ));
2005-02-01 19:45:58 +00:00
}
}
2004-01-06 19:52:14 +00:00
/**
2004-07-11 07:31:11 +00:00
* Send the user to a different Drupal page .
2004-01-06 19:52:14 +00:00
*
2004-07-11 07:31:11 +00:00
* This issues an on - site HTTP redirect . The function makes sure the redirected
* URL is formatted correctly .
2004-01-06 19:52:14 +00:00
*
2005-02-01 19:45:58 +00:00
* Usually the redirected URL is constructed from this function ' s input
* parameters . However you may override that behavior by setting a
* < em > destination </ em > in either the $_REQUEST - array ( i . e . by using
* the query string of an URI ) or the $_REQUEST [ 'edit' ] - array ( i . e . by
* using a hidden form field ) . This is used to direct the user back to
* the proper page after completing a form . For example , after editing
* a post on the 'admin/node' - page or after having logged on using the
* 'user login' - block in a sidebar . The function drupal_get_destination ()
* can be used to help set the destination URL .
*
2004-07-11 07:31:11 +00:00
* It is advised to use drupal_goto () instead of PHP ' s header (), because
* drupal_goto () will append the user ' s session ID to the URI when PHP is
* compiled with " --enable-trans-sid " .
*
* This function ends the request ; use it rather than a print theme ( 'page' )
* statement in your menu callback .
*
* @ param $path
* A Drupal path .
* @ param $query
* The query string component , if any .
* @ param $fragment
* The destination fragment identifier ( named anchor ) .
2005-02-01 19:45:58 +00:00
*
* @ see drupal_get_destination ()
2004-01-06 19:52:14 +00:00
*/
2004-07-11 07:31:11 +00:00
function drupal_goto ( $path = '' , $query = NULL , $fragment = NULL ) {
2005-02-01 19:45:58 +00:00
if ( $_REQUEST [ 'destination' ]) {
extract ( parse_url ( $_REQUEST [ 'destination' ]));
}
else if ( $_REQUEST [ 'edit' ][ 'destination' ]) {
extract ( parse_url ( $_REQUEST [ 'edit' ][ 'destination' ]));
}
2005-03-31 09:25:33 +00:00
$url = url ( $path , $query , $fragment , TRUE );
2004-01-06 19:52:14 +00:00
2004-07-11 07:31:11 +00:00
if ( ini_get ( 'session.use_trans_sid' ) && session_id () && ! strstr ( $url , session_id ())) {
$sid = session_name () . '=' . session_id ();
2003-10-03 14:10:05 +00:00
2004-07-11 07:31:11 +00:00
if ( strstr ( $url , '?' ) && ! strstr ( $url , $sid )) {
$url = $url . '&' . $sid ;
2004-01-06 19:52:14 +00:00
}
else {
2004-07-11 07:31:11 +00:00
$url = $url . '?' . $sid ;
2004-01-06 19:52:14 +00:00
}
}
2004-07-11 07:31:11 +00:00
// Before the redirect, allow modules to react to the end of the page request.
module_invoke_all ( 'exit' , $url );
header ( 'Location: ' . $url );
2004-01-06 19:52:14 +00:00
2004-07-11 07:31:11 +00:00
// The "Location" header sends a REDIRECT status code to the http
// daemon. In some cases this can go wrong, so we make sure none
// of the code below the drupal_goto() call gets executed when we redirect.
2004-01-06 19:52:14 +00:00
exit ();
}
/**
* Generates a 404 error if the request can not be handled .
*/
2003-12-16 21:06:34 +00:00
function drupal_not_found () {
2004-04-21 13:56:38 +00:00
header ( 'HTTP/1.0 404 Not Found' );
2005-03-31 09:25:33 +00:00
watchdog ( 'page not found' , t ( '%page not found.' , array ( '%page' => theme ( 'placeholder' , $_GET [ 'q' ]))), WATCHDOG_WARNING );
2003-12-16 21:06:34 +00:00
$path = drupal_get_normal_path ( variable_get ( 'site_404' , '' ));
2004-07-02 18:46:42 +00:00
$status = MENU_NOT_FOUND ;
2003-12-16 21:06:34 +00:00
if ( $path ) {
menu_set_active_item ( $path );
2005-04-28 19:23:19 +00:00
$return = menu_execute_active_handler ();
2003-12-16 21:06:34 +00:00
}
2005-04-28 19:23:19 +00:00
if ( empty ( $return )) {
2004-12-15 21:19:42 +00:00
drupal_set_title ( t ( 'Page not found' ));
2003-12-16 21:06:34 +00:00
}
2005-04-28 19:23:19 +00:00
print theme ( 'page' , $return );
2003-12-16 21:06:34 +00:00
}
2004-01-07 19:52:10 +00:00
2004-04-21 13:56:38 +00:00
/**
* Generates a 403 error if the request is not allowed .
*/
function drupal_access_denied () {
header ( 'HTTP/1.0 403 Forbidden' );
2005-03-31 09:25:33 +00:00
watchdog ( 'access denied' , t ( '%page denied access.' , array ( '%page' => theme ( 'placeholder' , $_GET [ 'q' ]))), WATCHDOG_WARNING , l ( t ( 'view' ), $_GET [ 'q' ]));
2004-04-21 13:56:38 +00:00
$path = drupal_get_normal_path ( variable_get ( 'site_403' , '' ));
2004-07-02 18:46:42 +00:00
$status = MENU_NOT_FOUND ;
2004-04-21 13:56:38 +00:00
if ( $path ) {
menu_set_active_item ( $path );
2005-04-28 19:23:19 +00:00
$return = menu_execute_active_handler ();
2004-04-21 13:56:38 +00:00
}
2005-04-28 19:23:19 +00:00
if ( empty ( $return )) {
2004-12-15 21:19:42 +00:00
drupal_set_title ( t ( 'Access denied' ));
2005-04-28 19:23:19 +00:00
$return = message_access ();
2004-04-21 13:56:38 +00:00
}
2005-04-28 19:23:19 +00:00
print theme ( 'page' , $return );
2004-04-21 13:56:38 +00:00
}
2004-01-07 19:52:10 +00:00
/**
2004-07-13 07:21:14 +00:00
* Perform an HTTP request .
2004-01-07 19:52:10 +00:00
*
2004-07-13 07:21:14 +00:00
* This is a flexible and powerful HTTP client implementation . Correctly handles
* GET , POST , PUT or any other HTTP requests . Handles redirects .
*
* @ param $url
* A string containing a fully qualified URI .
* @ param $headers
* An array containing an HTTP header => value pair .
* @ param $method
* A string defining the HTTP request to use .
* @ param $data
* A string containing data to include in the request .
* @ param $retry
* An integer representing how many times to retry the request in case of a
* redirect .
* @ return
* An object containing the HTTP request headers , response code , headers ,
* data , and redirect status .
2004-01-07 19:52:10 +00:00
*/
function drupal_http_request ( $url , $headers = array (), $method = 'GET' , $data = NULL , $retry = 3 ) {
2005-01-22 11:15:24 +00:00
$result = new StdClass ();
2004-07-13 07:21:14 +00:00
// Parse the URL, and make sure we can handle the schema.
2004-01-07 19:52:10 +00:00
$uri = parse_url ( $url );
switch ( $uri [ 'scheme' ]) {
case 'http' :
$fp = @ fsockopen ( $uri [ 'host' ], ( $uri [ 'port' ] ? $uri [ 'port' ] : 80 ), $errno , $errstr , 15 );
break ;
case 'https' :
2004-07-13 07:21:14 +00:00
// Note: Only works for PHP 4.3 compiled with OpenSSL.
$fp = @ fsockopen ( 'ssl://' . $uri [ 'host' ], ( $uri [ 'port' ] ? $uri [ 'port' ] : 443 ), $errno , $errstr , 20 );
2004-01-07 19:52:10 +00:00
break ;
default :
2004-07-13 07:21:14 +00:00
$result -> error = 'invalid schema ' . $uri [ 'scheme' ];
2004-01-07 19:52:10 +00:00
return $result ;
}
2004-07-13 07:21:14 +00:00
// Make sure the socket opened properly.
2004-01-07 19:52:10 +00:00
if ( ! $fp ) {
2004-07-13 07:21:14 +00:00
$result -> error = trim ( $errno . ' ' . $errstr );
2004-01-07 19:52:10 +00:00
return $result ;
}
2004-07-13 07:21:14 +00:00
// Construct the path to act on.
2004-01-07 19:52:10 +00:00
$path = $uri [ 'path' ] ? $uri [ 'path' ] : '/' ;
if ( $uri [ 'query' ]) {
2004-07-13 07:21:14 +00:00
$path .= '?' . $uri [ 'query' ];
2004-01-07 19:52:10 +00:00
}
2004-09-09 05:51:08 +00:00
// Create HTTP request.
2004-01-07 19:52:10 +00:00
$defaults = array (
2004-07-13 07:21:14 +00:00
'Host' => 'Host: ' . $uri [ 'host' ],
2004-01-07 20:43:26 +00:00
'User-Agent' => 'User-Agent: Drupal (+http://www.drupal.org/)' ,
'Content-Length' => 'Content-Length: ' . strlen ( $data )
2004-01-07 19:52:10 +00:00
);
foreach ( $headers as $header => $value ) {
2004-07-13 07:21:14 +00:00
$defaults [ $header ] = $header . ': ' . $value ;
2004-01-07 19:52:10 +00:00
}
2004-07-13 07:21:14 +00:00
$request = $method . ' ' . $path . " HTTP/1.0 \r \n " ;
2004-01-07 19:52:10 +00:00
$request .= implode ( " \r \n " , $defaults );
$request .= " \r \n \r \n " ;
if ( $data ) {
2004-07-13 07:21:14 +00:00
$request .= $data . " \r \n " ;
2004-01-07 19:52:10 +00:00
}
$result -> request = $request ;
fwrite ( $fp , $request );
// Fetch response.
2004-04-27 18:17:17 +00:00
$response = '' ;
2004-04-21 19:43:23 +00:00
while ( ! feof ( $fp ) && $data = fread ( $fp , 1024 )) {
2004-04-27 18:17:17 +00:00
$response .= $data ;
2004-01-07 19:52:10 +00:00
}
fclose ( $fp );
// Parse response.
2004-06-21 20:14:41 +00:00
list ( $headers , $result -> data ) = explode ( " \r \n \r \n " , $response , 2 );
$headers = preg_split ( " / \r \n | \n | \r / " , $headers );
list ( $protocol , $code , $text ) = explode ( ' ' , trim ( array_shift ( $headers )), 3 );
2004-01-07 19:52:10 +00:00
$result -> headers = array ();
// Parse headers.
2004-06-21 20:14:41 +00:00
while ( $line = trim ( array_shift ( $headers ))) {
2004-01-07 19:52:10 +00:00
list ( $header , $value ) = explode ( ':' , $line , 2 );
2005-08-22 20:24:53 +00:00
if ( isset ( $result -> headers [ $header ]) && $header == 'Set-Cookie' ) {
// RFC 2109: the Set-Cookie response header comprises the token Set-
// Cookie:, followed by a comma-separated list of one or more cookies.
$result -> headers [ $header ] .= ',' . trim ( $value );
}
else {
$result -> headers [ $header ] = trim ( $value );
}
2004-01-07 19:52:10 +00:00
}
$responses = array (
100 => 'Continue' , 101 => 'Switching Protocols' ,
200 => 'OK' , 201 => 'Created' , 202 => 'Accepted' , 203 => 'Non-Authoritative Information' , 204 => 'No Content' , 205 => 'Reset Content' , 206 => 'Partial Content' ,
300 => 'Multiple Choices' , 301 => 'Moved Permanently' , 302 => 'Found' , 303 => 'See Other' , 304 => 'Not Modified' , 305 => 'Use Proxy' , 307 => 'Temporary Redirect' ,
400 => 'Bad Request' , 401 => 'Unauthorized' , 402 => 'Payment Required' , 403 => 'Forbidden' , 404 => 'Not Found' , 405 => 'Method Not Allowed' , 406 => 'Not Acceptable' , 407 => 'Proxy Authentication Required' , 408 => 'Request Time-out' , 409 => 'Conflict' , 410 => 'Gone' , 411 => 'Length Required' , 412 => 'Precondition Failed' , 413 => 'Request Entity Too Large' , 414 => 'Request-URI Too Large' , 415 => 'Unsupported Media Type' , 416 => 'Requested range not satisfiable' , 417 => 'Expectation Failed' ,
500 => 'Internal Server Error' , 501 => 'Not Implemented' , 502 => 'Bad Gateway' , 503 => 'Service Unavailable' , 504 => 'Gateway Time-out' , 505 => 'HTTP Version not supported'
);
// RFC 2616 states that all unknown HTTP codes must be treated the same as
2004-09-09 05:51:08 +00:00
// the base code in their class.
2004-01-07 19:52:10 +00:00
if ( ! isset ( $responses [ $code ])) {
$code = floor ( $code / 100 ) * 100 ;
}
switch ( $code ) {
case 200 : // OK
case 304 : // Not modified
break ;
case 301 : // Moved permanently
case 302 : // Moved temporarily
case 307 : // Moved temporarily
$location = $result -> headers [ 'Location' ];
if ( $retry ) {
$result = drupal_http_request ( $result -> headers [ 'Location' ], $headers , $method , $data , -- $retry );
$result -> redirect_code = $result -> code ;
}
$result -> redirect_url = $location ;
break ;
default :
$result -> error = $text ;
}
$result -> code = $code ;
return $result ;
}
2004-09-09 05:51:08 +00:00
/**
* @ } End of " HTTP handling " .
*/
2003-12-16 21:06:34 +00:00
2004-07-13 07:21:14 +00:00
/**
2004-10-04 22:04:07 +00:00
* Log errors as defined by administrator
* Error levels :
* 1 = Log errors to database .
* 2 = Log errors to database and to screen .
2004-07-13 07:21:14 +00:00
*/
2005-03-21 19:26:47 +00:00
function error_handler ( $errno , $message , $filename , $line ) {
2005-03-03 20:13:20 +00:00
if ( $errno & ( E_ALL ^ E_NOTICE )) {
2004-10-04 22:04:07 +00:00
$types = array ( 1 => 'error' , 2 => 'warning' , 4 => 'parse error' , 8 => 'notice' , 16 => 'core error' , 32 => 'core warning' , 64 => 'compile error' , 128 => 'compile warning' , 256 => 'user error' , 512 => 'user warning' , 1024 => 'user notice' , 2048 => 'strict warning' );
$entry = $types [ $errno ] . ': ' . $message . ' in ' . $filename . ' on line ' . $line . '.' ;
2003-04-21 12:36:09 +00:00
2004-10-04 22:04:07 +00:00
if ( variable_get ( 'error_level' , 1 ) == 1 ) {
2004-07-13 07:21:14 +00:00
print '<pre>' . $entry . '</pre>' ;
2003-12-13 14:59:55 +00:00
}
2005-01-09 09:22:40 +00:00
watchdog ( 'php' , t ( '%message in %file on line %line.' , array ( '%error' => $types [ $errno ], '%message' => $message , '%file' => $filename , '%line' => $line )), WATCHDOG_ERROR );
2001-12-01 15:20:48 +00:00
}
}
2005-03-21 19:26:47 +00:00
function _fix_gpc_magic ( & $item ) {
2003-12-13 14:59:55 +00:00
if ( is_array ( $item )) {
2003-12-19 10:52:37 +00:00
array_walk ( $item , '_fix_gpc_magic' );
}
else {
2003-12-19 13:44:08 +00:00
$item = stripslashes ( $item );
2002-12-26 12:16:09 +00:00
}
}
2004-07-13 07:21:14 +00:00
/**
* Correct double - escaping problems caused by " magic quotes " in some PHP
* installations .
*/
2003-10-31 19:34:03 +00:00
function fix_gpc_magic () {
static $fixed = false ;
2004-07-13 07:21:14 +00:00
if ( ! $fixed && ini_get ( 'magic_quotes_gpc' )) {
2003-12-13 14:59:55 +00:00
array_walk ( $_GET , '_fix_gpc_magic' );
array_walk ( $_POST , '_fix_gpc_magic' );
array_walk ( $_COOKIE , '_fix_gpc_magic' );
array_walk ( $_REQUEST , '_fix_gpc_magic' );
$fixed = true ;
}
2003-10-31 19:34:03 +00:00
}
2005-05-21 18:33:59 +00:00
/**
* An unchecked checkbox is not present in $_POST so we fix it here by
* proving a default value of 0. Also , with form_checkboxes () we expect
* an array , but HTML does not send the empty array . This is also taken
* care off .
*/
function fix_checkboxes () {
if ( isset ( $_POST [ 'form_array' ])) {
$_POST [ 'edit' ] = _fix_checkboxes ( $_POST [ 'edit' ], $_POST [ 'form_array' ], array ());
}
if ( isset ( $_POST [ 'form_zero' ])) {
$_POST [ 'edit' ] = _fix_checkboxes ( $_POST [ 'edit' ], $_POST [ 'form_zero' ], 0 );
}
}
function _fix_checkboxes ( $array1 , $array2 , $value ) {
if ( is_array ( $array2 ) && count ( $array2 )) {
foreach ( $array2 as $k => $v ) {
if ( is_array ( $v ) && count ( $v )) {
$array1 [ $k ] = _fix_checkboxes ( $array1 [ $k ], $v , $value );
}
else if ( ! isset ( $array1 [ $k ])) {
$array1 [ $k ] = $value ;
}
}
}
else {
$array1 = $value ;
}
return $array1 ;
}
2004-01-06 19:52:14 +00:00
/**
* @ name Conversion
* @ {
2004-09-09 05:51:08 +00:00
* Converts data structures to different types .
2004-01-06 19:52:14 +00:00
*/
2004-07-13 07:21:14 +00:00
/**
* Convert an associative array to an anonymous object .
*/
2003-12-13 14:59:55 +00:00
function array2object ( $array ) {
if ( is_array ( $array )) {
2005-01-22 11:15:24 +00:00
$object = new StdClass ();
2003-12-13 14:59:55 +00:00
foreach ( $array as $key => $value ) {
2001-12-30 16:16:38 +00:00
$object -> $key = $value ;
}
}
else {
2003-12-13 14:59:55 +00:00
$object = $array ;
2001-12-30 16:16:38 +00:00
}
return $object ;
}
2004-07-13 07:21:14 +00:00
/**
* Convert an object to an associative array .
*/
2003-12-13 14:59:55 +00:00
function object2array ( $object ) {
if ( is_object ( $object )) {
foreach ( $object as $key => $value ) {
2001-12-30 16:16:38 +00:00
$array [ $key ] = $value ;
}
}
else {
2003-12-13 14:59:55 +00:00
$array = $object ;
2001-12-30 16:16:38 +00:00
}
return $array ;
}
2004-09-09 05:51:08 +00:00
/**
* @ } End of " Conversion " .
*/
2001-12-30 16:16:38 +00:00
2004-01-06 19:52:14 +00:00
/**
* @ name Messages
* @ {
2004-09-09 05:51:08 +00:00
* Frequently used messages .
2004-01-06 19:52:14 +00:00
*/
2004-07-13 07:21:14 +00:00
/**
* Return a string with an " access denied " message .
*
* Always consider whether to use drupal_access_denied () instead to return a
* proper ( and customizable ) 403 error .
*/
2001-12-01 15:20:48 +00:00
function message_access () {
2004-07-13 07:21:14 +00:00
return t ( 'You are not authorized to access this page.' );
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Return a string with a " not applicable " message .
*/
2001-12-01 15:20:48 +00:00
function message_na () {
2004-07-13 07:21:14 +00:00
return t ( 'n/a' );
2001-12-01 15:20:48 +00:00
}
2004-09-09 05:51:08 +00:00
/**
* @ } End of " Messages " .
*/
2001-12-01 15:20:48 +00:00
2004-07-13 07:21:14 +00:00
/**
* Initialize the localization system .
*/
2004-08-11 11:26:20 +00:00
function locale_initialize () {
global $user ;
2004-09-15 09:54:32 +00:00
if ( function_exists ( 'i18n_get_lang' )) {
return i18n_get_lang ();
}
2004-08-11 11:26:20 +00:00
if ( function_exists ( 'locale' )) {
$languages = locale_supported_languages ();
$languages = $languages [ 'name' ];
}
else {
2004-09-09 13:36:01 +00:00
// Ensure the locale/language is correctly returned, even without locale.module.
// Useful for e.g. XML/HTML 'lang' attributes.
$languages = array ( 'en' => 'English' );
2004-08-11 11:26:20 +00:00
}
2003-03-04 15:10:37 +00:00
if ( $user -> uid && $languages [ $user -> language ]) {
return $user -> language ;
}
else {
return key ( $languages );
}
2001-12-27 15:27:44 +00:00
}
2004-01-06 19:52:14 +00:00
/**
2004-07-13 07:21:14 +00:00
* Translate strings to the current locale .
2004-01-06 19:52:14 +00:00
*
2004-01-11 20:31:26 +00:00
* When using t (), try to put entire sentences and strings in one t () call .
2004-09-09 05:51:08 +00:00
* This makes it easier for translators . HTML markup within translation strings
* is acceptable , if necessary . The suggested syntax for a link embedded
* within a translation string is :
* @ code
2004-07-13 07:21:14 +00:00
* $msg = t ( ' You must log in below or < a href = " %url " > create a new
* account </ a > before viewing the next page . ', array(' % url '
* => url ( 'user/register' )));
2004-09-09 05:51:08 +00:00
* @ endcode
2004-01-11 20:31:26 +00:00
* We suggest the same syntax for links to other sites . This makes it easy to
* change link URLs if needed ( which happens often ) without requiring updates
* to translations .
2004-01-06 19:52:14 +00:00
*
2004-07-13 07:21:14 +00:00
* @ param $string
2004-09-09 05:51:08 +00:00
* A string containing the English string to translate .
2004-07-13 07:21:14 +00:00
* @ param $args
* An associative array of replacements to make after translation . Incidences
2004-09-09 05:51:08 +00:00
* of any key in this array are replaced with the corresponding value .
2004-07-13 07:21:14 +00:00
* @ return
* The translated string .
2004-01-06 19:52:14 +00:00
*/
2002-04-20 11:52:50 +00:00
function t ( $string , $args = 0 ) {
2004-08-11 11:26:20 +00:00
global $locale ;
if ( function_exists ( 'locale' ) && $locale != 'en' ) {
$string = locale ( $string );
}
2002-04-24 20:55:20 +00:00
2002-04-20 11:52:50 +00:00
if ( ! $args ) {
return $string ;
2002-04-22 09:05:36 +00:00
}
else {
2002-04-20 11:52:50 +00:00
return strtr ( $string , $args );
}
2001-12-27 15:27:44 +00:00
}
2004-07-13 07:21:14 +00:00
/**
2005-03-31 09:25:33 +00:00
* Encode special characters in a plain - text string for display as HTML .
2004-07-13 07:21:14 +00:00
*/
2005-03-31 09:25:33 +00:00
function check_plain ( $text ) {
2005-08-19 11:10:35 +00:00
return htmlspecialchars ( $text );
2003-06-05 18:09:39 +00:00
}
2004-01-06 19:52:14 +00:00
/**
2004-09-09 05:51:08 +00:00
* @ defgroup validation Input validation
2004-02-08 17:12:44 +00:00
* @ {
2004-09-09 05:51:08 +00:00
* Functions to validate user input .
2004-01-06 19:52:14 +00:00
*/
2003-03-28 10:55:27 +00:00
/**
2004-07-13 07:21:14 +00:00
* Verify the syntax of the given e - mail address .
*
* Empty e - mail addresses are allowed . See RFC 2822 for details .
2003-03-28 10:55:27 +00:00
*
2004-07-13 07:21:14 +00:00
* @ param $mail
* A string containing an email address .
2004-01-07 19:52:10 +00:00
* @ return
2004-07-13 07:21:14 +00:00
* TRUE if the address is in a valid format .
2003-03-28 10:55:27 +00:00
*/
2003-04-13 13:42:51 +00:00
function valid_email_address ( $mail ) {
2003-03-28 10:55:27 +00:00
$user = '[a-zA-Z0-9_\-\.\+\^!#\$%&*+\/\=\?\`\|\{\}~\']+' ;
2004-05-24 18:09:28 +00:00
$domain = '(?:(?:[a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.?)+' ;
2003-03-28 10:55:27 +00:00
$ipv4 = '[0-9]{1,3}(\.[0-9]{1,3}){3}' ;
$ipv6 = '[0-9a-fA-F]{1,4}(\:[0-9a-fA-F]{1,4}){7}' ;
2003-12-13 14:59:55 +00:00
return preg_match ( " /^ $user @( $domain |( \ [( $ipv4 | $ipv6 ) \ ])) $ / " , $mail );
2003-03-28 10:55:27 +00:00
}
2003-07-16 20:14:26 +00:00
/**
* Verify the syntax of the given URL .
*
2004-04-12 08:27:57 +00:00
* @ param $url
2004-07-13 07:21:14 +00:00
* The URL to verify .
2004-04-12 08:27:57 +00:00
* @ param $absolute
2004-09-09 05:51:08 +00:00
* Whether the URL is absolute ( beginning with a scheme such as " http: " ) .
2004-04-12 08:27:57 +00:00
* @ return
2004-07-13 07:21:14 +00:00
* TRUE if the URL is in a valid format .
2003-07-16 20:14:26 +00:00
*/
2004-04-12 08:27:57 +00:00
function valid_url ( $url , $absolute = FALSE ) {
2005-03-06 06:57:58 +00:00
$allowed_characters = '[a-z0-9\/:_\-_\.\?\$,~=#&%\+]' ;
2004-03-21 10:28:10 +00:00
if ( $absolute ) {
2005-03-06 06:57:58 +00:00
return preg_match ( " /^(http|https|ftp): \ / \ / " . $allowed_characters . " + $ /i " , $url );
2004-03-21 10:28:10 +00:00
}
else {
2005-03-06 06:57:58 +00:00
return preg_match ( " /^ " . $allowed_characters . " + $ /i " , $url );
2004-03-21 10:28:10 +00:00
}
2003-07-16 20:14:26 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Validate data input by a user .
*
* Ensures that user data cannot be used to perform attacks on the site .
*
* @ param $data
* The input to check .
* @ return
* TRUE if the input data is acceptable .
*/
2004-01-06 19:52:14 +00:00
function valid_input_data ( $data ) {
if ( is_array ( $data ) || is_object ( $data )) {
2004-07-13 07:21:14 +00:00
// Form data can contain a number of nested arrays.
2004-01-06 19:52:14 +00:00
foreach ( $data as $key => $value ) {
2004-01-17 23:19:02 +00:00
if ( ! valid_input_data ( $key ) || ! valid_input_data ( $value )) {
2004-07-13 07:21:14 +00:00
return FALSE ;
2004-01-06 19:52:14 +00:00
}
}
}
2004-12-28 11:04:24 +00:00
else if ( isset ( $data )) {
2004-07-13 07:21:14 +00:00
// Detect dangerous input data.
2004-01-06 19:52:14 +00:00
2004-12-25 14:57:39 +00:00
// Decode all normal character entities.
$data = decode_entities ( $data , array ( '<' , '&' , '"' ));
2004-07-13 07:21:14 +00:00
// Check strings:
$match = preg_match ( '/\Wjavascript\s*:/i' , $data );
$match += preg_match ( '/\Wexpression\s*\(/i' , $data );
$match += preg_match ( '/\Walert\s*\(/i' , $data );
2004-01-06 19:52:14 +00:00
2004-07-13 07:21:14 +00:00
// Check attributes:
2004-01-06 19:52:14 +00:00
$match += preg_match ( " / \ W(dynsrc|datasrc|data|lowsrc|on[a-z]+) \ s*=[^>]+?>/i " , $data );
2004-07-13 07:21:14 +00:00
// Check tags:
2004-01-06 19:52:14 +00:00
$match += preg_match ( " /< \ s*(applet|script|object|style|embed|form|blink|meta|html|frame|iframe|layer|ilayer|head|frameset|xml)/i " , $data );
if ( $match ) {
2005-03-31 09:25:33 +00:00
watchdog ( 'security' , t ( 'Terminated request because of suspicious input data: %data.' , array ( '%data' => theme ( 'placeholder' , $data ))));
2004-07-13 07:21:14 +00:00
return FALSE ;
2004-01-06 19:52:14 +00:00
}
}
2004-07-13 07:21:14 +00:00
return TRUE ;
2004-01-06 19:52:14 +00:00
}
2004-09-09 05:51:08 +00:00
/**
* @ } End of " defgroup validation " .
*/
2004-01-06 19:52:14 +00:00
2004-11-15 21:17:25 +00:00
/**
* Register an event for the current visitor ( hostname / IP ) to the flood control mechanism .
*
* @ param $name
* The name of the event .
*/
function flood_register_event ( $name ) {
db_query ( " INSERT INTO { flood} (event, hostname, timestamp) VALUES ('%s', '%s', %d) " , $name , $_SERVER [ 'REMOTE_ADDR' ], time ());
}
/**
* Check if the current visitor ( hostname / IP ) is allowed to proceed with the specified event .
* The user is allowed to proceed if he did not trigger the specified event more than
* $threshold times per hour .
*
* @ param $name
* The name of the event .
* @ param $number
* The maximum number of the specified event per hour ( per visitor ) .
* @ return
* True if the user did not exceed the hourly threshold . False otherwise .
*/
function flood_is_allowed ( $name , $threshold ) {
$number = db_num_rows ( db_query ( " SELECT event FROM { flood} WHERE event = '%s' AND hostname = '%s' AND timestamp > %d " , $name , $_SERVER [ 'REMOTE_ADDR' ], time () - 3600 ));
return ( $number < $threshold ? TRUE : FALSE );
}
2003-10-07 10:22:33 +00:00
function check_file ( $filename ) {
return is_uploaded_file ( $filename );
2001-12-01 15:20:48 +00:00
}
2004-02-08 17:12:44 +00:00
/**
2004-09-09 05:51:08 +00:00
* @ defgroup format Formatting
2004-02-08 17:12:44 +00:00
* @ {
2004-09-09 05:51:08 +00:00
* Functions to format numbers , strings , dates , etc .
2004-02-08 17:12:44 +00:00
*/
2004-07-13 07:21:14 +00:00
/**
* Formats an RSS channel .
*
* Arbitrary elements may be added using the $args associative array .
*/
function format_rss_channel ( $title , $link , $description , $items , $language = 'en' , $args = array ()) {
2002-04-27 13:19:37 +00:00
// arbitrary elements may be added using the $args associative array
2003-12-13 14:59:55 +00:00
$output = " <channel> \n " ;
2005-03-31 09:25:33 +00:00
$output .= ' <title>' . check_plain ( $title ) . " </title> \n " ;
$output .= ' <link>' . check_url ( $link ) . " </link> \n " ;
$output .= ' <description>' . check_plain ( $description ) . " </description> \n " ;
$output .= ' <language>' . check_plain ( $language ) . " </language> \n " ;
2002-04-27 13:19:37 +00:00
foreach ( $args as $key => $value ) {
2005-03-31 09:25:33 +00:00
$output .= ' <' . $key . '>' . check_plain ( $value ) . " </ $key > \n " ;
2002-04-27 13:19:37 +00:00
}
2001-12-01 15:20:48 +00:00
$output .= $items ;
$output .= " </channel> \n " ;
return $output ;
}
2004-07-13 07:21:14 +00:00
/**
* Format a single RSS item .
*
* Arbitrary elements may be added using the $args associative array .
*/
2002-04-27 13:19:37 +00:00
function format_rss_item ( $title , $link , $description , $args = array ()) {
2003-12-13 14:59:55 +00:00
$output = " <item> \n " ;
2005-03-31 09:25:33 +00:00
$output .= ' <title>' . check_plain ( $title ) . " </title> \n " ;
$output .= ' <link>' . check_url ( $link ) . " </link> \n " ;
$output .= ' <description>' . check_plain ( $description ) . " </description> \n " ;
2002-04-27 13:19:37 +00:00
foreach ( $args as $key => $value ) {
2005-02-01 14:09:31 +00:00
if ( is_array ( $value )) {
if ( $value [ 'key' ]) {
$output .= ' <' . $value [ 'key' ];
if ( is_array ( $value [ 'attributes' ])) {
$output .= drupal_attributes ( $value [ 'attributes' ]);
}
if ( $value [ 'value' ]) {
$output .= '>' . $value [ 'value' ] . '</' . $value [ 'key' ] . " > \n " ;
}
else {
$output .= " /> \n " ;
}
}
}
else {
2005-03-31 09:25:33 +00:00
$output .= ' <' . $key . '>' . check_plain ( $value ) . " </ $key > \n " ;
2005-02-01 14:09:31 +00:00
}
2002-04-27 13:19:37 +00:00
}
2001-12-01 15:20:48 +00:00
$output .= " </item> \n " ;
return $output ;
}
2003-01-21 22:44:25 +00:00
/**
2004-07-13 07:21:14 +00:00
* Format a string containing a count of items .
2003-01-21 22:44:25 +00:00
*
2004-07-13 07:21:14 +00:00
* This function ensures that the string is pluralized correctly . Since t () is
* called by this function , make sure not to pass already - localized strings to it .
*
* @ param $count
* The item count to display .
* @ param $singular
* The string for the singular case . Please make sure it is clear this is
* singular , to ease translation ( e . g . use " 1 new comment " instead of " 1 new " ) .
* @ param $plural
* The string for the plural case . Please make sure it is clear this is plural ,
* to ease translation . Use % count in place of the item count , as in " %count
* new comments " .
* @ return
* A translated string .
2003-01-21 22:44:25 +00:00
*/
2001-12-01 15:20:48 +00:00
function format_plural ( $count , $singular , $plural ) {
2004-10-27 18:23:06 +00:00
if ( $count == 1 ) return t ( $singular , array ( " %count " => $count ));
2004-08-11 11:26:20 +00:00
// get the plural index through the gettext formula
2005-06-22 20:19:58 +00:00
$index = ( function_exists ( 'locale_get_plural' )) ? locale_get_plural ( $count ) : - 1 ;
2004-08-11 11:26:20 +00:00
if ( $index < 0 ) { // backward compatibility
return t ( $plural , array ( " %count " => $count ));
}
else {
switch ( $index ) {
case " 0 " :
2004-10-27 18:23:06 +00:00
return t ( $singular , array ( " %count " => $count ));
2004-08-11 11:26:20 +00:00
case " 1 " :
return t ( $plural , array ( " %count " => $count ));
default :
return t ( strtr ( $plural , array ( " %count " => '%count[' . $index . ']' )), array ( '%count[' . $index . ']' => $count ));
}
}
2001-12-01 15:20:48 +00:00
}
2004-02-08 17:12:44 +00:00
/**
2004-07-13 07:21:14 +00:00
* Generate a string representation for the given byte count .
2004-02-08 17:12:44 +00:00
*
2004-07-13 07:21:14 +00:00
* @ param $size
* The size in bytes .
* @ return
* A translated string representation of the size .
2004-02-08 17:12:44 +00:00
*/
2001-12-01 15:20:48 +00:00
function format_size ( $size ) {
2004-07-13 07:21:14 +00:00
$suffix = t ( 'bytes' );
2005-05-20 11:31:16 +00:00
if ( $size >= 1024 ) {
2001-12-01 15:20:48 +00:00
$size = round ( $size / 1024 , 2 );
2004-07-13 07:21:14 +00:00
$suffix = t ( 'KB' );
2001-12-01 15:20:48 +00:00
}
2005-05-20 11:31:16 +00:00
if ( $size >= 1024 ) {
2001-12-01 15:20:48 +00:00
$size = round ( $size / 1024 , 2 );
2004-07-13 07:21:14 +00:00
$suffix = t ( 'MB' );
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
return t ( '%size %suffix' , array ( '%size' => $size , '%suffix' => $suffix ));
2001-12-01 15:20:48 +00:00
}
2004-02-08 17:12:44 +00:00
/**
2004-07-13 07:21:14 +00:00
* Format a time interval with the requested granularity .
2004-02-08 17:12:44 +00:00
*
2004-07-13 07:21:14 +00:00
* @ param $timestamp
* The length of the interval in seconds .
* @ param $granularity
* How many different units to display in the string .
* @ return
* A translated string representation of the interval .
2004-02-08 17:12:44 +00:00
*/
2004-01-21 06:40:57 +00:00
function format_interval ( $timestamp , $granularity = 2 ) {
2004-07-13 07:21:14 +00:00
$units = array ( '1 year|%count years' => 31536000 , '1 week|%count weeks' => 604800 , '1 day|%count days' => 86400 , '1 hour|%count hours' => 3600 , '1 min|%count min' => 60 , '1 sec|%count sec' => 1 );
2004-04-12 08:27:57 +00:00
$output = '' ;
2003-12-24 12:40:28 +00:00
foreach ( $units as $key => $value ) {
2004-07-13 07:21:14 +00:00
$key = explode ( '|' , $key );
2001-12-01 15:20:48 +00:00
if ( $timestamp >= $value ) {
2004-07-13 07:21:14 +00:00
$output .= ( $output ? ' ' : '' ) . format_plural ( floor ( $timestamp / $value ), $key [ 0 ], $key [ 1 ]);
2001-12-01 15:20:48 +00:00
$timestamp %= $value ;
2004-01-21 06:40:57 +00:00
$granularity -- ;
}
if ( $granularity == 0 ) {
break ;
2001-12-01 15:20:48 +00:00
}
}
2004-07-13 07:21:14 +00:00
return $output ? $output : t ( '0 sec' );
2001-12-01 15:20:48 +00:00
}
2004-02-08 17:12:44 +00:00
/**
2004-07-13 07:21:14 +00:00
* Format a date with the given configured format or a custom format string .
*
2004-02-08 17:12:44 +00:00
* Drupal allows administrators to select formatting strings for 'small' ,
* 'medium' and 'large' date formats . This function can handle these formats ,
* as well as any custom format .
*
2004-07-13 07:21:14 +00:00
* @ param $timestamp
* The exact date to format , as a UNIX timestamp .
* @ param $type
* The format to use . Can be " small " , " medium " or " large " for the preconfigured
* date formats . If " custom " is specified , then $format is required as well .
* @ param $format
2004-12-03 20:38:22 +00:00
* A PHP date format string as required by date () . A backslash should be used
* before a character to avoid interpreting the character as part of a date
* format .
2004-07-13 07:21:14 +00:00
* @ param $timezone
* Time zone offset in seconds ; if omitted , the user ' s time zone is used .
* @ return
* A translated date string in the requested format .
2004-02-08 17:12:44 +00:00
*/
2004-02-08 21:42:59 +00:00
function format_date ( $timestamp , $type = 'medium' , $format = '' , $timezone = NULL ) {
if ( $timezone === NULL ) {
global $user ;
2004-08-10 01:30:09 +00:00
if ( variable_get ( 'configurable_timezones' , 1 ) && $user -> uid && strlen ( $user -> timezone )) {
$timezone = $user -> timezone ;
}
else {
$timezone = variable_get ( 'date_default_timezone' , 0 );
}
2004-02-08 21:42:59 +00:00
}
2001-12-01 15:20:48 +00:00
2004-02-08 21:42:59 +00:00
$timestamp += $timezone ;
2001-12-01 15:20:48 +00:00
switch ( $type ) {
2004-02-08 21:42:59 +00:00
case 'small' :
$format = variable_get ( 'date_format_short' , 'm/d/Y - H:i' );
2001-12-01 15:20:48 +00:00
break ;
2004-02-08 21:42:59 +00:00
case 'large' :
$format = variable_get ( 'date_format_long' , 'l, F j, Y - H:i' );
2001-12-01 15:20:48 +00:00
break ;
2004-02-08 21:42:59 +00:00
case 'custom' :
2003-09-29 18:20:38 +00:00
// No change to format
2001-12-01 15:20:48 +00:00
break ;
2004-02-08 21:42:59 +00:00
case 'medium' :
2001-12-01 15:20:48 +00:00
default :
2004-02-08 21:42:59 +00:00
$format = variable_get ( 'date_format_medium' , 'D, m/d/Y - H:i' );
2003-09-29 18:20:38 +00:00
}
2004-02-08 21:42:59 +00:00
$max = strlen ( $format );
2004-04-12 08:27:57 +00:00
$date = '' ;
2004-07-02 18:46:42 +00:00
for ( $i = 0 ; $i < $max ; $i ++ ) {
$c = $format { $i };
2004-07-21 15:30:35 +00:00
if ( strpos ( 'AaDFlM' , $c ) !== false ) {
2004-02-21 14:08:09 +00:00
$date .= t ( gmdate ( $c , $timestamp ));
2004-02-08 21:42:59 +00:00
}
2004-07-21 15:30:35 +00:00
else if ( strpos ( 'BdgGhHiIjLmnsStTUwWYyz' , $c ) !== false ) {
2004-02-08 21:42:59 +00:00
$date .= gmdate ( $c , $timestamp );
}
else if ( $c == 'r' ) {
$date .= format_date ( $timestamp - $timezone , 'custom' , 'D, d M Y H:i:s O' , $timezone );
2003-09-29 18:20:38 +00:00
}
2004-02-08 21:42:59 +00:00
else if ( $c == 'O' ) {
$date .= sprintf ( '%s%02d%02d' , ( $timezone < 0 ? '-' : '+' ), abs ( $timezone / 3600 ), abs ( $timezone % 3600 ) / 60 );
}
else if ( $c == 'Z' ) {
$date .= $timezone ;
2003-09-29 18:20:38 +00:00
}
2004-12-03 20:38:22 +00:00
else if ( $c == '\\' ) {
$date .= $format [ ++ $i ];
}
2003-09-29 18:20:38 +00:00
else {
2004-02-08 21:42:59 +00:00
$date .= $c ;
2003-09-29 18:20:38 +00:00
}
2001-12-01 15:20:48 +00:00
}
2004-02-08 21:42:59 +00:00
2001-12-01 15:20:48 +00:00
return $date ;
}
2004-09-09 05:51:08 +00:00
/**
* @ } End of " defgroup format " .
*/
2001-12-01 15:20:48 +00:00
2004-01-06 19:52:14 +00:00
/**
2004-07-13 07:21:14 +00:00
* @ defgroup form Form generation
2004-01-06 19:52:14 +00:00
* @ {
2004-09-09 05:51:08 +00:00
* Functions to enable output of HTML forms and form elements .
2004-07-22 16:06:54 +00:00
*
2004-09-09 05:51:08 +00:00
* Drupal uses these functions to achieve consistency in its form presentation ,
* while at the same time simplifying code and reducing the amount of HTML that
2004-10-19 18:02:31 +00:00
* must be explicitly generated by modules .
2004-01-06 19:52:14 +00:00
*/
2004-07-13 07:21:14 +00:00
/**
* Generate a form from a set of form elements .
*
* @ param $form
* An HTML string containing one or more form elements .
* @ param $method
* The query method to use ( " post " or " get " ) .
* @ param $action
* The URL to send the form contents to , if not the current page .
* @ param $attributes
* An associative array of attributes to add to the form tag .
* @ result
* An HTML string with the contents of $form wrapped in a form tag .
*/
function form ( $form , $method = 'post' , $action = NULL , $attributes = NULL ) {
2003-05-19 18:26:49 +00:00
if ( ! $action ) {
2003-06-03 18:04:47 +00:00
$action = request_uri ();
2003-05-19 18:26:49 +00:00
}
2005-05-25 04:18:54 +00:00
// Anonymous div to satisfy XHTML compliancy.
2005-05-25 03:50:25 +00:00
return '<form action="' . check_url ( $action ) . '" method="' . $method . '"' . drupal_attributes ( $attributes ) . " > \n <div> " . $form . " \n </div></form> \n " ;
2001-12-01 15:20:48 +00:00
}
2004-05-31 09:40:56 +00:00
/**
2004-07-13 07:21:14 +00:00
* File an error against the form element with the specified name .
2004-05-31 09:40:56 +00:00
*/
function form_set_error ( $name , $message ) {
$GLOBALS [ 'form' ][ $name ] = $message ;
drupal_set_message ( $message , 'error' );
}
/**
2004-07-04 16:50:02 +00:00
* Return an associative array of all errors .
2004-05-31 09:40:56 +00:00
*/
2004-07-04 16:50:02 +00:00
function form_get_errors () {
2004-07-04 06:19:41 +00:00
if ( array_key_exists ( 'form' , $GLOBALS )) {
return $GLOBALS [ 'form' ];
}
2004-05-31 09:40:56 +00:00
}
/**
* Return the error message filed against the form with the specified name .
*/
function _form_get_error ( $name ) {
2004-07-02 18:46:42 +00:00
if ( array_key_exists ( 'form' , $GLOBALS )) {
return $GLOBALS [ 'form' ][ $name ];
}
2004-05-31 09:40:56 +00:00
}
function _form_get_class ( $name , $required , $error ) {
return $name . ( $required ? ' required' : '' ) . ( $error ? ' error' : '' );
}
2004-07-13 07:21:14 +00:00
/**
* Format a general form item .
*
* @ param $title
* The label for the form item .
* @ param $value
* The contents of the form item .
* @ param $description
* Explanatory text to display after the form item .
* @ param $id
* A unique identifier for the form item .
* @ param $required
* Whether the user must fill in this form element before submitting the form .
* @ param $error
* An error message to display alongside the form element .
* @ return
* A themed HTML string representing the form item .
*/
2004-05-31 09:40:56 +00:00
function form_item ( $title , $value , $description = NULL , $id = NULL , $required = FALSE , $error = FALSE ) {
2004-07-08 11:31:12 +00:00
return theme ( 'form_element' , $title , $value , $description , $id , $required , $error );
2001-12-01 15:20:48 +00:00
}
2003-11-17 19:16:55 +00:00
2004-07-13 07:21:14 +00:00
/**
* Format a group of form items .
*
* @ param $legend
* The label for the form item group .
* @ param $group
* The form items within the group , as an HTML string .
* @ param $description
* Explanatory text to display after the form item group .
2005-05-05 09:07:08 +00:00
* @ param $attributes
* An associative array of HTML attributes to add to the fieldset tag .
2004-07-13 07:21:14 +00:00
* @ return
* A themed HTML string representing the form item group .
*/
2005-05-05 09:07:08 +00:00
function form_group ( $legend , $group , $description = NULL , $attributes = NULL ) {
return '<fieldset' . drupal_attributes ( $attributes ) . '>' . ( $legend ? '<legend>' . $legend . '</legend>' : '' ) . $group . ( $description ? '<div class="description">' . $description . '</div>' : '' ) . " </fieldset> \n " ;
2003-11-06 18:27:58 +00:00
}
2001-12-01 15:20:48 +00:00
2005-06-21 09:45:45 +00:00
/**
* Format a group of form items .
*
* @ param $legend
* The label for the form item group .
* @ param $group
* The form items within the group , as an HTML string .
* @ param $collapsed
* A boolean value decided whether the group starts collapsed .
* @ param $description
* Explanatory text to display after the form item group .
* @ param $attributes
* An associative array of HTML attributes to add to the fieldset tag .
* @ return
* A themed HTML string representing the form item group .
*/
function form_group_collapsible ( $legend , $group , $collapsed = FALSE , $description = NULL , $attributes = NULL ) {
drupal_add_js ( 'misc/collapse.js' );
$attributes [ 'class' ] .= ' collapsible' ;
if ( $collapsed ) {
$attributes [ 'class' ] .= ' collapsed' ;
}
return '<fieldset' . drupal_attributes ( $attributes ) . '>' . ( $legend ? '<legend>' . $legend . '</legend>' : '' ) . $group . ( $description ? '<div class="description">' . $description . '</div>' : '' ) . " </fieldset> \n " ;
}
2004-07-13 07:21:14 +00:00
/**
* Format a radio button .
*
* @ param $title
* The label for the radio button .
* @ param $name
* The internal name used to refer to the button .
* @ param $value
* The value that the form element takes on when selected .
* @ param $checked
* Whether the button will be initially selected when the page is rendered .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the button .
* @ param $required
* Whether the user must select this radio button before submitting the form .
* @ return
* A themed HTML string representing the radio button .
*/
function form_radio ( $title , $name , $value = 1 , $checked = FALSE , $description = NULL , $attributes = NULL , $required = FALSE ) {
$element = '<input type="radio" class="' . _form_get_class ( 'form-radio' , $required , _form_get_error ( $name )) . '" name="edit[' . $name . ']" value="' . $value . '"' . ( $checked ? ' checked="checked"' : '' ) . drupal_attributes ( $attributes ) . ' />' ;
2004-02-24 20:22:45 +00:00
if ( ! is_null ( $title )) {
2004-07-13 07:21:14 +00:00
$element = '<label class="option">' . $element . ' ' . $title . '</label>' ;
2004-02-24 20:22:45 +00:00
}
2004-07-08 11:31:12 +00:00
return theme ( 'form_element' , NULL , $element , $description , $name , $required , _form_get_error ( $name ));
2003-12-24 12:40:28 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a set of radio buttons .
*
* @ param $title
* The label for the radio buttons as a group .
* @ param $name
* The internal name used to refer to the buttons .
* @ param $value
* The currently selected radio button ' s key .
* @ param $options
* An associative array of buttons to display . The keys in this array are
* button values , while the values are the labels to display for each button .
* @ param $description
* Explanatory text to display after the form item .
* @ param $required
* Whether the user must select a radio button before submitting the form .
* @ param $attributes
* An associative array of HTML attributes to add to each button .
* @ return
* A themed HTML string representing the radio button set .
*/
2004-06-04 18:10:05 +00:00
function form_radios ( $title , $name , $value , $options , $description = NULL , $required = FALSE , $attributes = NULL ) {
2003-12-24 12:40:28 +00:00
if ( count ( $options ) > 0 ) {
2004-04-12 08:27:57 +00:00
$choices = '' ;
2003-12-24 12:40:28 +00:00
foreach ( $options as $key => $choice ) {
2005-03-18 07:34:07 +00:00
$choices .= '<label class="option"><input type="radio" class="form-radio" name="edit[' . $name . ']" value="' . $key . '"' . ( $key == $value ? ' checked="checked"' : '' ) . drupal_attributes ( $attributes ) . ' /> ' . $choice . '</label><br />' ;
2003-12-24 12:40:28 +00:00
}
2004-10-12 19:55:53 +00:00
return theme ( 'form_element' , $title , $choices , $description , NULL , $required , _form_get_error ( $name ));
2003-12-24 12:40:28 +00:00
}
2003-04-13 13:42:51 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a checkbox .
*
* @ param $title
* The label for the checkbox .
* @ param $name
* The internal name used to refer to the button .
* @ param $value
* The value that the form element takes on when selected .
* @ param $checked
* Whether the button will be initially selected when the page is rendered .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the button .
* @ param $required
* Whether the user must check this box before submitting the form .
* @ return
* A themed HTML string representing the checkbox .
*/
function form_checkbox ( $title , $name , $value = 1 , $checked = FALSE , $description = NULL , $attributes = NULL , $required = FALSE ) {
2005-05-25 03:50:25 +00:00
$element = '<input type="checkbox" class="' . _form_get_class ( 'form-checkbox' , $required , _form_get_error ( $name )) . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '" value="' . $value . '"' . ( $checked ? ' checked="checked"' : '' ) . drupal_attributes ( $attributes ) . ' />' ;
2004-02-24 20:22:45 +00:00
if ( ! is_null ( $title )) {
2004-07-13 07:21:14 +00:00
$element = '<label class="option">' . $element . ' ' . $title . '</label>' ;
2004-02-24 20:22:45 +00:00
}
2005-05-21 18:33:59 +00:00
return form_hidden ( $name , 1 , 'form_zero' ) . theme ( 'form_element' , NULL , $element , $description , $name , $required , _form_get_error ( $name ));
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a set of checkboxes .
*
* @ param $title
* The label for the checkboxes as a group .
* @ param $name
* The internal name used to refer to the buttons .
* @ param $values
2004-10-19 18:02:31 +00:00
* A linear array of keys of the initially checked boxes .
2004-07-13 07:21:14 +00:00
* @ param $options
* An associative array of buttons to display . The keys in this array are
* button values , while the values are the labels to display for each button .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to each button .
* @ param $required
* Whether the user must check a box before submitting the form .
* @ return
2005-07-29 19:07:17 +00:00
* A themed HTML string representing the checkbox set .
2004-07-13 07:21:14 +00:00
*/
2004-07-08 11:31:12 +00:00
function form_checkboxes ( $title , $name , $values , $options , $description = NULL , $attributes = NULL , $required = FALSE ) {
2004-05-15 15:42:47 +00:00
if ( count ( $options ) > 0 ) {
2004-10-13 17:08:50 +00:00
if ( ! isset ( $values ) || $values == 0 ) {
2004-05-15 15:42:47 +00:00
$values = array ();
}
$choices = '' ;
foreach ( $options as $key => $choice ) {
2005-03-18 07:34:07 +00:00
$choices .= '<label class="option"><input type="checkbox" class="form-checkbox" name="edit[' . $name . '][]" value="' . $key . '"' . ( in_array ( $key , $values ) ? ' checked="checked"' : '' ) . drupal_attributes ( $attributes ) . ' /> ' . $choice . '</label><br />' ;
2004-05-15 15:42:47 +00:00
}
2005-05-21 18:33:59 +00:00
return form_hidden ( $name , 1 , 'form_array' ) . theme ( 'form_element' , $title , $choices , $description , NULL , $required , _form_get_error ( $name ));
2004-05-15 15:42:47 +00:00
}
}
2004-07-13 07:21:14 +00:00
/**
* Format a single - line text field .
*
* @ param $title
* The label for the text field .
* @ param $name
* The internal name used to refer to the field .
* @ param $value
* The initial value for the field at page load time .
* @ param $size
* A measure of the visible size of the field ( passed directly to HTML ) .
* @ param $maxlength
* The maximum number of characters that may be entered in the field .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ param $required
* Whether the user must enter some text in the field .
* @ return
* A themed HTML string representing the field .
*/
2004-04-24 15:39:31 +00:00
function form_textfield ( $title , $name , $value , $size , $maxlength , $description = NULL , $attributes = NULL , $required = FALSE ) {
2004-07-13 07:21:14 +00:00
$size = $size ? ' size="' . $size . '"' : '' ;
2005-05-25 03:50:25 +00:00
return theme ( 'form_element' , $title , '<input type="text" maxlength="' . $maxlength . '" class="' . _form_get_class ( 'form-text' , $required , _form_get_error ( $name )) . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '"' . $size . ' value="' . check_plain ( $value ) . '"' . drupal_attributes ( $attributes ) . ' />' , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
2001-12-01 15:20:48 +00:00
}
2005-05-24 06:00:22 +00:00
/**
* Format a single - line text field that uses Ajax for autocomplete .
*
* @ param $title
* The label for the text field .
* @ param $name
* The internal name used to refer to the field .
* @ param $value
* The initial value for the field at page load time .
* @ param $size
* A measure of the visible size of the field ( passed directly to HTML ) .
* @ param $maxlength
* The maximum number of characters that may be entered in the field .
* @ param $callback_path
* A drupal path for the Ajax autocomplete callback .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ param $required
* Whether the user must enter some text in the field .
* @ return
* A themed HTML string representing the field .
*/
function form_autocomplete ( $title , $name , $value , $size , $maxlength , $callback_path , $description = NULL , $attributes = NULL , $required = FALSE ) {
drupal_add_js ( 'misc/autocomplete.js' );
$size = $size ? ' size="' . $size . '"' : '' ;
2005-05-25 03:50:25 +00:00
$output = theme ( 'form_element' , $title , '<input type="text" maxlength="' . $maxlength . '" class="' . _form_get_class ( 'form-text form-autocomplete' , $required , _form_get_error ( $name )) . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '"' . $size . ' value="' . check_plain ( $value ) . '"' . drupal_attributes ( $attributes ) . ' />' , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
$output .= '<input class="autocomplete" type="hidden" id="edit-' . form_clean_id ( $name ) . '-autocomplete" value="' . check_url ( url ( $callback_path , NULL , NULL , TRUE )) . '" disabled="disabled" />' ;
2005-05-24 06:00:22 +00:00
return $output ;
}
2004-07-13 07:21:14 +00:00
/**
* Format a single - line text field that does not display its contents visibly .
*
* @ param $title
* The label for the text field .
* @ param $name
* The internal name used to refer to the field .
* @ param $value
* The initial value for the field at page load time .
* @ param $size
* A measure of the visible size of the field ( passed directly to HTML ) .
* @ param $maxlength
* The maximum number of characters that may be entered in the field .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ param $required
* Whether the user must enter some text in the field .
* @ return
* A themed HTML string representing the field .
*/
2004-04-24 15:39:31 +00:00
function form_password ( $title , $name , $value , $size , $maxlength , $description = NULL , $attributes = NULL , $required = FALSE ) {
2004-07-13 07:21:14 +00:00
$size = $size ? ' size="' . $size . '"' : '' ;
2005-05-25 03:50:25 +00:00
return theme ( 'form_element' , $title , '<input type="password" class="' . _form_get_class ( 'form-password' , $required , _form_get_error ( $name )) . '" maxlength="' . $maxlength . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '"' . $size . ' value="' . check_plain ( $value ) . '"' . drupal_attributes ( $attributes ) . ' />' , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a multiple - line text field .
*
* @ param $title
* The label for the text field .
* @ param $name
* The internal name used to refer to the field .
* @ param $value
* The initial value for the field at page load time .
* @ param $cols
* The width of the field , in columns of text .
* @ param $rows
* The height of the field , in rows of text .
* @ param $description
* Explanatory text to display after the form item .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ param $required
* Whether the user must enter some text in the field .
* @ return
* A themed HTML string representing the field .
*/
2004-04-24 15:39:31 +00:00
function form_textarea ( $title , $name , $value , $cols , $rows , $description = NULL , $attributes = NULL , $required = FALSE ) {
2004-07-13 07:21:14 +00:00
$cols = $cols ? ' cols="' . $cols . '"' : '' ;
2005-04-06 07:43:10 +00:00
$pre = '' ;
$post = '' ;
2004-08-21 17:06:51 +00:00
2005-04-06 07:43:10 +00:00
// optionally plug in a WYSIWYG editor
2004-08-21 17:06:51 +00:00
foreach ( module_list () as $module_name ) {
if ( module_hook ( $module_name , 'textarea' )) {
2005-04-06 07:43:10 +00:00
$pre .= module_invoke ( $module_name , 'textarea' , 'pre' , $name );
$post .= module_invoke ( $module_name , 'textarea' , 'post' , $name );
2004-08-21 17:06:51 +00:00
}
}
2005-05-25 03:50:25 +00:00
return theme ( 'form_element' , $title , $pre . '<textarea' . $cols . ' rows="' . $rows . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '" class="' . _form_get_class ( 'textarea' , $required , _form_get_error ( $name )) . '"' . drupal_attributes ( $attributes ) . '>' . check_plain ( $value ) . '</textarea>' . $post , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
2001-12-01 15:20:48 +00:00
}
2004-05-05 15:58:26 +00:00
/**
2004-07-13 07:21:14 +00:00
* Format a dropdown menu or scrolling selection box .
2004-05-05 15:58:26 +00:00
*
2004-07-13 07:21:14 +00:00
* @ param $title
* The label for the form element .
* @ param $name
* The internal name used to refer to the form element .
* @ param $value
* The key of the currently selected item , or a linear array of keys of all the
* currently selected items if multiple selections are allowed .
* @ param $options
* An associative array of buttons to display . The keys in this array are
* button values , while the values are the labels to display for each button .
* @ param $description
* Explanatory text to display after the form item .
* @ param $extra
* Additional HTML to inject into the select element tag .
* @ param $multiple
* Whether the user may select more than one item .
* @ param $required
* Whether the user must select a value before submitting the form .
* @ return
* A themed HTML string representing the form element .
2004-05-05 15:58:26 +00:00
*
2004-07-13 07:21:14 +00:00
* It is possible to group options together ; to do this , change the format of
* $options to an associative array in which the keys are group labels , and the
* values are associative arrays in the normal $options format .
2004-05-05 15:58:26 +00:00
*/
2004-07-13 07:21:14 +00:00
function form_select ( $title , $name , $value , $options , $description = NULL , $extra = 0 , $multiple = FALSE , $required = FALSE ) {
2004-04-12 08:27:57 +00:00
$select = '' ;
2003-12-24 12:40:28 +00:00
foreach ( $options as $key => $choice ) {
2004-05-05 15:58:26 +00:00
if ( is_array ( $choice )) {
2004-07-13 07:21:14 +00:00
$select .= '<optgroup label="' . $key . '">' ;
2004-05-05 15:58:26 +00:00
foreach ( $choice as $key => $choice ) {
2005-03-31 09:25:33 +00:00
$select .= '<option value="' . $key . '"' . ( is_array ( $value ) ? ( in_array ( $key , $value ) ? ' selected="selected"' : '' ) : ( $value == $key ? ' selected="selected"' : '' )) . '>' . check_plain ( $choice ) . '</option>' ;
2004-05-05 15:58:26 +00:00
}
2004-07-13 07:21:14 +00:00
$select .= '</optgroup>' ;
2004-05-05 15:58:26 +00:00
}
else {
2005-03-31 09:25:33 +00:00
$select .= '<option value="' . $key . '"' . ( is_array ( $value ) ? ( in_array ( $key , $value ) ? ' selected="selected"' : '' ) : ( $value == $key ? ' selected="selected"' : '' )) . '>' . check_plain ( $choice ) . '</option>' ;
2004-05-05 15:58:26 +00:00
}
2003-06-05 18:09:39 +00:00
}
2005-05-25 03:50:25 +00:00
return theme ( 'form_element' , $title , '<select name="edit[' . $name . ']' . ( $multiple ? '[]' : '' ) . '"' . ( $multiple ? ' multiple="multiple" ' : '' ) . ( $extra ? ' ' . $extra : '' ) . ' id="edit-' . form_clean_id ( $name ) . '">' . $select . '</select>' , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
2003-06-05 18:09:39 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a file upload field .
*
* @ param $title
* The label for the file upload field .
* @ param $name
* The internal name used to refer to the field .
* @ param $size
* A measure of the visible size of the field ( passed directly to HTML ) .
* @ param $description
* Explanatory text to display after the form item .
* @ param $required
* Whether the user must upload a file to the field .
* @ return
* A themed HTML string representing the field .
*
* For assistance with handling the uploaded file correctly , see the API
* provided by file . inc .
*/
2004-04-24 15:39:31 +00:00
function form_file ( $title , $name , $size , $description = NULL , $required = FALSE ) {
2005-05-25 03:50:25 +00:00
return theme ( 'form_element' , $title , '<input type="file" class="' . _form_get_class ( 'form-file' , $required , _form_get_error ( $name )) . '" name="edit[' . $name . ']" id="edit-' . form_clean_id ( $name ) . '" size="' . $size . " \" /> \n " , $description , 'edit-' . $name , $required , _form_get_error ( $name ));
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Store data in a hidden form field .
*
* @ param $name
* The internal name used to refer to the field .
* @ param $value
* The stored data .
* @ return
* A themed HTML string representing the hidden field .
*
* This function can be useful in retaining information between page requests ,
* but be sure to validate the data on the receiving page as it is possible for
* an attacker to change the value before it is submitted .
*/
2005-05-21 18:33:59 +00:00
function form_hidden ( $name , $value , $edit = 'edit' ) {
return '<input type="hidden" name="' . $edit . '[' . $name . ']" value="' . check_plain ( $value ) . " \" /> \n " ;
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format an action button .
*
* @ param $value
* Both the label for the button , and the value passed to the target page
* when this button is clicked .
* @ param $name
* The internal name used to refer to the button .
* @ param $type
* What type to pass to the HTML input tag .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ return
* A themed HTML string representing the button .
*/
function form_button ( $value , $name = 'op' , $type = 'submit' , $attributes = NULL ) {
2005-03-31 09:25:33 +00:00
return '<input type="' . $type . '" class="form-' . $type . '" name="' . $name . '" value="' . check_plain ( $value ) . '" ' . drupal_attributes ( $attributes ) . " /> \n " ;
2003-12-15 19:58:53 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a form submit button .
*
* @ param $value
* Both the label for the button , and the value passed to the target page
* when this button is clicked .
* @ param $name
* The internal name used to refer to the button .
* @ param $attributes
* An associative array of HTML attributes to add to the form item .
* @ return
* A themed HTML string representing the button .
*/
function form_submit ( $value , $name = 'op' , $attributes = NULL ) {
return form_button ( $value , $name , 'submit' , $attributes );
2001-12-01 15:20:48 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format a weight selection menu .
*
* @ param $title
* The label for the form element .
* @ param $name
* The internal name used to refer to the form element .
* @ param $value
* The selected weight value at page load time .
* @ param $delta
* The largest in absolute value the weight can be . For example , if set to 10 ,
* weights could range from - 10 to 10 inclusive .
* @ param $description
* Explanatory text to display after the form item .
* @ param $extra
* Additional HTML to inject into the select element tag .
* @ return
* A themed HTML string representing the form element .
*/
function form_weight ( $title = NULL , $name = 'weight' , $value = 0 , $delta = 10 , $description = NULL , $extra = 0 ) {
2003-04-03 22:18:27 +00:00
for ( $n = ( - 1 * $delta ); $n <= $delta ; $n ++ ) {
2002-10-26 15:17:26 +00:00
$weights [ $n ] = $n ;
}
return form_select ( $title , $name , $value , $weights , $description , $extra );
}
2004-07-22 16:06:54 +00:00
2005-05-25 03:50:25 +00:00
/**
* Remove invalid characters from an HTML ID attribute string
*
* @ param $id
* The ID to clean
* @ return
* The cleaned ID
*/
function form_clean_id ( $id = NULL ) {
$id = str_replace ( '][' , '-' , $id );
return $id ;
}
2004-07-22 16:06:54 +00:00
/**
2004-09-09 05:51:08 +00:00
* @ } End of " defgroup form " .
2004-07-22 16:06:54 +00:00
*/
2003-06-19 17:26:27 +00:00
2004-07-13 07:21:14 +00:00
/**
* Generate an internal Drupal URL .
*
* @ param $path
* The Drupal path being linked to , such as " admin/node " .
* @ param $query
* A query string to append to the link .
* @ param $fragment
* A fragment identifier ( named anchor ) to append to the link .
* @ param $absolute
* Whether to force the output to be an absolute link ( beginning with http : ) .
* Useful for links that will be displayed outside the site , such as in an RSS feed .
* @ return
* an HTML string containing a link to the given path .
*
* When creating links in modules , consider whether l () could be a better
* alternative than url () .
*/
function url ( $path = NULL , $query = NULL , $fragment = NULL , $absolute = FALSE ) {
2004-01-14 22:41:15 +00:00
global $base_url ;
2004-01-17 23:19:02 +00:00
2003-05-31 13:05:06 +00:00
static $script ;
if ( empty ( $script )) {
2004-09-09 05:51:08 +00:00
// On some web servers, such as IIS, we can't omit "index.php". So, we
// generate "index.php?q=foo" instead of "?q=foo" on anything that is not
// Apache.
2004-07-13 07:21:14 +00:00
$script = ( strpos ( $_SERVER [ 'SERVER_SOFTWARE' ], 'Apache' ) === false ) ? 'index.php' : '' ;
2003-05-31 13:05:06 +00:00
}
2004-07-13 07:21:14 +00:00
$path = drupal_get_path_alias ( $path );
2003-09-30 17:01:34 +00:00
2003-12-29 19:10:26 +00:00
if ( isset ( $fragment )) {
2004-07-13 07:21:14 +00:00
$fragment = '#' . $fragment ;
2003-12-29 19:10:26 +00:00
}
2005-03-31 09:25:33 +00:00
$base = ( $absolute ? $base_url . '/' : '' );
2004-02-11 19:21:14 +00:00
2004-07-13 07:21:14 +00:00
if ( variable_get ( 'clean_url' , '0' ) == '0' ) {
if ( isset ( $path )) {
2003-02-14 19:52:45 +00:00
if ( isset ( $query )) {
2005-03-31 09:25:33 +00:00
return $base . $script . '?q=' . $path . '&' . $query . $fragment ;
2003-02-14 19:52:45 +00:00
}
else {
2004-07-13 07:21:14 +00:00
return $base . $script . '?q=' . $path . $fragment ;
2003-02-14 19:52:45 +00:00
}
2003-01-06 21:24:21 +00:00
}
else {
2003-02-14 19:52:45 +00:00
if ( isset ( $query )) {
2004-07-13 07:21:14 +00:00
return $base . $script . '?' . $query . $fragment ;
2003-02-14 19:52:45 +00:00
}
else {
2004-07-13 07:21:14 +00:00
return $base . $fragment ;
2003-02-14 19:52:45 +00:00
}
2003-01-06 21:24:21 +00:00
}
}
else {
2004-07-13 07:21:14 +00:00
if ( isset ( $path )) {
2003-02-14 19:52:45 +00:00
if ( isset ( $query )) {
2004-07-13 07:21:14 +00:00
return $base . $path . '?' . $query . $fragment ;
2003-02-14 19:52:45 +00:00
}
else {
2004-07-13 07:21:14 +00:00
return $base . $path . $fragment ;
2003-02-14 19:52:45 +00:00
}
2003-01-06 21:24:21 +00:00
}
2003-01-11 10:46:11 +00:00
else {
2003-02-14 19:52:45 +00:00
if ( isset ( $query )) {
2004-07-13 07:21:14 +00:00
return $base . $script . '?' . $query . $fragment ;
2003-02-14 19:52:45 +00:00
}
else {
2004-07-13 07:21:14 +00:00
return $base . $fragment ;
2003-02-14 19:52:45 +00:00
}
2003-01-11 10:46:11 +00:00
}
2003-01-06 21:24:21 +00:00
}
2002-04-20 11:52:50 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Format an attribute string to insert in a tag .
*
* @ param $attributes
* An associative array of HTML attributes .
* @ return
* An HTML string ready for insertion in a tag .
*/
2004-03-10 19:32:37 +00:00
function drupal_attributes ( $attributes = array ()) {
2005-05-14 09:23:47 +00:00
if ( is_array ( $attributes )) {
2003-08-12 20:37:16 +00:00
$t = array ();
foreach ( $attributes as $key => $value ) {
2005-03-31 09:25:33 +00:00
$t [] = $key . '="' . check_plain ( $value ) . '"' ;
2003-08-12 20:37:16 +00:00
}
2005-07-30 12:52:54 +00:00
return ' ' . implode ( ' ' , $t );
2002-04-27 13:19:37 +00:00
}
2003-08-12 20:37:16 +00:00
}
2003-01-06 19:51:01 +00:00
2004-07-13 07:21:14 +00:00
/**
* Format an internal Drupal link .
*
* This function correctly handles aliased paths , and allows themes to highlight
* links to the current page correctly , so all internal links output by modules
* should be generated by this function if possible .
*
* @ param $text
* The text to be enclosed with the anchor tag .
* @ param $path
2005-05-14 09:23:47 +00:00
* The Drupal path being linked to , such as " admin/node " . Note , this must be a
* system URL as the url () function will generate the alias .
2004-07-13 07:21:14 +00:00
* @ param $attributes
* An associative array of HTML attributes to apply to the anchor tag .
* @ param $query
* A query string to append to the link .
* @ param $fragment
* A fragment identifier ( named anchor ) to append to the link .
* @ param $absolute
* Whether to force the output to be an absolute link ( beginning with http : ) .
* Useful for links that will be displayed outside the site , such as in an RSS feed .
2005-03-31 09:25:33 +00:00
* @ param $html
* Whether the title is HTML , or just plain - text .
2004-07-13 07:21:14 +00:00
* @ return
* an HTML string containing a link to the given path .
*/
2005-03-31 09:25:33 +00:00
function l ( $text , $path , $attributes = array (), $query = NULL , $fragment = NULL , $absolute = FALSE , $html = FALSE ) {
2005-05-14 09:23:47 +00:00
if ( $path == $_GET [ 'q' ]) {
2004-01-19 21:57:42 +00:00
if ( isset ( $attributes [ 'class' ])) {
$attributes [ 'class' ] .= ' active' ;
}
else {
$attributes [ 'class' ] = 'active' ;
}
}
2005-03-31 09:25:33 +00:00
return '<a href="' . check_url ( url ( $path , $query , $fragment , $absolute )) . '"' . drupal_attributes ( $attributes ) . '>' . ( $html ? $text : check_plain ( $text )) . '</a>' ;
2002-04-20 11:52:50 +00:00
}
2004-07-13 07:21:14 +00:00
/**
* Perform end - of - request tasks .
*
* This function sets the page cache if appropriate , and allows modules to
* react to the closing of the page by calling hook_exit () .
*/
2003-05-18 09:48:49 +00:00
function drupal_page_footer () {
2004-07-13 07:21:14 +00:00
if ( variable_get ( 'cache' , 0 )) {
2002-01-05 16:28:34 +00:00
page_set_cache ();
2001-12-01 15:20:48 +00:00
}
2003-01-26 13:22:02 +00:00
2004-07-13 07:21:14 +00:00
module_invoke_all ( 'exit' );
2001-12-01 15:20:48 +00:00
}
2004-02-12 19:37:04 +00:00
/**
2004-07-13 07:21:14 +00:00
* Form an associative array from a linear array .
2004-02-12 19:37:04 +00:00
*
2004-07-13 07:21:14 +00:00
* This function walks through the provided array and constructs an associative
* array out of it . The keys of the resulting array will be the values of the
* input array . The values will be the same as the keys unless a function is
* specified , in which case the output of the function is used for the values
* instead .
*
* @ param $array
* A linear array .
* @ param $function
* The name of a function to apply to all values before output .
* @ result
* An associative array .
2004-02-12 19:37:04 +00:00
*/
function drupal_map_assoc ( $array , $function = NULL ) {
if ( ! isset ( $function )) {
$result = array ();
foreach ( $array as $value ) {
$result [ $value ] = $value ;
}
return $result ;
}
elseif ( function_exists ( $function )) {
$result = array ();
foreach ( $array as $value ) {
$result [ $value ] = $function ( $value );
}
return $result ;
}
}
The Input formats - filter patch has landed. I still need to make update instructions for modules and update the hook docs.
Here's an overview of the changes:
1) Multiple Input formats: they are complete filter configurations (what filters to use, in what order and with which settings). Input formats are admin-definable, and usage of them is role-dependant. For example, you can set it up so that regular users can only use limited HTML, while admins can free HTML without any tag limitations.
The input format can be chosen per content item (nodes, comments, blocks, ...) when you add/edit them. If only a single format is available, there is no choice, and nothing changes with before.
The default install (and the upgrade) contains a basic set of formats which should satisfy the average user's needs.
2) Filters have toggles
Because now you might want to enable a filter only on some input formats, an explicit toggle is provided by the filter system. Modules do not need to worry about it and filters that still have their own on/off switch should get rid of it.
3) Multiple filters per module
This was necessary to accomodate the next change, and it's also a logical extension of the filter system.
4) Embedded PHP is now a filter
Thanks to the multiple input formats, I was able to move the 'embedded PHP' feature from block.module, page.module and book.module into a simple filter which executes PHP code. This filter is part of filter.module, and by default there is an input format 'PHP', restricted to the administrator only, which contains this filter.
This change means that block.module now passes custom block contents through the filter system.
As well as from reducing code duplication and avoiding two type selectors for page/book nodes, you can now combine PHP code with other filters.
5) User-supplied PHP code now requires <?php ?> tags.
This is required for teasers to work with PHP code. Because PHP evaluation is now just another step in the filter process, we can't do this. Also, because teasers are generated before filtering, this would result in errors when the teaser generation would cut off a piece of PHP code.
Also, regular PHP syntax explicitly includes the <?php ?> tags for PHP files, so it makes sense to use the same convention for embedded PHP in Drupal.
6) Filter caching was added.
Benchmarking shows that even for a simple setup (basic html filtering + legacy URL rewriting), filtercache can offer speedups. Unlike the old filtercache, this uses the normal cache table.
7) Filtertips were moved from help into a hook_filter_tips(). This was required to accomodate the fact that there are multiple filters per module, and that filter settings are format dependant. Shoehorning filter tips into _help was ugly and silly. The display of the filter tips is done through the input format selector, so filter_tips_short() no longer exists.
8) A more intelligent linebreak convertor was added, which doesn't stop working if you use block-level tags and which adds <p> tags.
2004-08-10 18:34:29 +00:00
/**
2004-09-09 05:51:08 +00:00
* Evaluate a string of PHP code .
The Input formats - filter patch has landed. I still need to make update instructions for modules and update the hook docs.
Here's an overview of the changes:
1) Multiple Input formats: they are complete filter configurations (what filters to use, in what order and with which settings). Input formats are admin-definable, and usage of them is role-dependant. For example, you can set it up so that regular users can only use limited HTML, while admins can free HTML without any tag limitations.
The input format can be chosen per content item (nodes, comments, blocks, ...) when you add/edit them. If only a single format is available, there is no choice, and nothing changes with before.
The default install (and the upgrade) contains a basic set of formats which should satisfy the average user's needs.
2) Filters have toggles
Because now you might want to enable a filter only on some input formats, an explicit toggle is provided by the filter system. Modules do not need to worry about it and filters that still have their own on/off switch should get rid of it.
3) Multiple filters per module
This was necessary to accomodate the next change, and it's also a logical extension of the filter system.
4) Embedded PHP is now a filter
Thanks to the multiple input formats, I was able to move the 'embedded PHP' feature from block.module, page.module and book.module into a simple filter which executes PHP code. This filter is part of filter.module, and by default there is an input format 'PHP', restricted to the administrator only, which contains this filter.
This change means that block.module now passes custom block contents through the filter system.
As well as from reducing code duplication and avoiding two type selectors for page/book nodes, you can now combine PHP code with other filters.
5) User-supplied PHP code now requires <?php ?> tags.
This is required for teasers to work with PHP code. Because PHP evaluation is now just another step in the filter process, we can't do this. Also, because teasers are generated before filtering, this would result in errors when the teaser generation would cut off a piece of PHP code.
Also, regular PHP syntax explicitly includes the <?php ?> tags for PHP files, so it makes sense to use the same convention for embedded PHP in Drupal.
6) Filter caching was added.
Benchmarking shows that even for a simple setup (basic html filtering + legacy URL rewriting), filtercache can offer speedups. Unlike the old filtercache, this uses the normal cache table.
7) Filtertips were moved from help into a hook_filter_tips(). This was required to accomodate the fact that there are multiple filters per module, and that filter settings are format dependant. Shoehorning filter tips into _help was ugly and silly. The display of the filter tips is done through the input format selector, so filter_tips_short() no longer exists.
8) A more intelligent linebreak convertor was added, which doesn't stop working if you use block-level tags and which adds <p> tags.
2004-08-10 18:34:29 +00:00
*
2004-09-09 05:51:08 +00:00
* This is a wrapper around PHP ' s eval () . It uses output buffering to capture both
* returned and printed text . Unlike eval (), we require code to be surrounded by
* < ? php ?> tags; in other words, we evaluate the code as if it were a stand-alone
* PHP file .
*
2004-09-16 16:12:21 +00:00
* Using this wrapper also ensures that the PHP code which is evaluated can not
* overwrite any variables in the calling code , unlike a regular eval () call .
*
2004-09-09 05:51:08 +00:00
* @ param $code
* The code to evaluate .
* @ return
* A string containing the printed output of the code , followed by the returned
* output of the code .
The Input formats - filter patch has landed. I still need to make update instructions for modules and update the hook docs.
Here's an overview of the changes:
1) Multiple Input formats: they are complete filter configurations (what filters to use, in what order and with which settings). Input formats are admin-definable, and usage of them is role-dependant. For example, you can set it up so that regular users can only use limited HTML, while admins can free HTML without any tag limitations.
The input format can be chosen per content item (nodes, comments, blocks, ...) when you add/edit them. If only a single format is available, there is no choice, and nothing changes with before.
The default install (and the upgrade) contains a basic set of formats which should satisfy the average user's needs.
2) Filters have toggles
Because now you might want to enable a filter only on some input formats, an explicit toggle is provided by the filter system. Modules do not need to worry about it and filters that still have their own on/off switch should get rid of it.
3) Multiple filters per module
This was necessary to accomodate the next change, and it's also a logical extension of the filter system.
4) Embedded PHP is now a filter
Thanks to the multiple input formats, I was able to move the 'embedded PHP' feature from block.module, page.module and book.module into a simple filter which executes PHP code. This filter is part of filter.module, and by default there is an input format 'PHP', restricted to the administrator only, which contains this filter.
This change means that block.module now passes custom block contents through the filter system.
As well as from reducing code duplication and avoiding two type selectors for page/book nodes, you can now combine PHP code with other filters.
5) User-supplied PHP code now requires <?php ?> tags.
This is required for teasers to work with PHP code. Because PHP evaluation is now just another step in the filter process, we can't do this. Also, because teasers are generated before filtering, this would result in errors when the teaser generation would cut off a piece of PHP code.
Also, regular PHP syntax explicitly includes the <?php ?> tags for PHP files, so it makes sense to use the same convention for embedded PHP in Drupal.
6) Filter caching was added.
Benchmarking shows that even for a simple setup (basic html filtering + legacy URL rewriting), filtercache can offer speedups. Unlike the old filtercache, this uses the normal cache table.
7) Filtertips were moved from help into a hook_filter_tips(). This was required to accomodate the fact that there are multiple filters per module, and that filter settings are format dependant. Shoehorning filter tips into _help was ugly and silly. The display of the filter tips is done through the input format selector, so filter_tips_short() no longer exists.
8) A more intelligent linebreak convertor was added, which doesn't stop working if you use block-level tags and which adds <p> tags.
2004-08-10 18:34:29 +00:00
*/
function drupal_eval ( $code ) {
ob_start ();
print eval ( '?>' . $code );
2004-08-21 16:44:05 +00:00
$output = ob_get_contents ();
ob_end_clean ();
return $output ;
The Input formats - filter patch has landed. I still need to make update instructions for modules and update the hook docs.
Here's an overview of the changes:
1) Multiple Input formats: they are complete filter configurations (what filters to use, in what order and with which settings). Input formats are admin-definable, and usage of them is role-dependant. For example, you can set it up so that regular users can only use limited HTML, while admins can free HTML without any tag limitations.
The input format can be chosen per content item (nodes, comments, blocks, ...) when you add/edit them. If only a single format is available, there is no choice, and nothing changes with before.
The default install (and the upgrade) contains a basic set of formats which should satisfy the average user's needs.
2) Filters have toggles
Because now you might want to enable a filter only on some input formats, an explicit toggle is provided by the filter system. Modules do not need to worry about it and filters that still have their own on/off switch should get rid of it.
3) Multiple filters per module
This was necessary to accomodate the next change, and it's also a logical extension of the filter system.
4) Embedded PHP is now a filter
Thanks to the multiple input formats, I was able to move the 'embedded PHP' feature from block.module, page.module and book.module into a simple filter which executes PHP code. This filter is part of filter.module, and by default there is an input format 'PHP', restricted to the administrator only, which contains this filter.
This change means that block.module now passes custom block contents through the filter system.
As well as from reducing code duplication and avoiding two type selectors for page/book nodes, you can now combine PHP code with other filters.
5) User-supplied PHP code now requires <?php ?> tags.
This is required for teasers to work with PHP code. Because PHP evaluation is now just another step in the filter process, we can't do this. Also, because teasers are generated before filtering, this would result in errors when the teaser generation would cut off a piece of PHP code.
Also, regular PHP syntax explicitly includes the <?php ?> tags for PHP files, so it makes sense to use the same convention for embedded PHP in Drupal.
6) Filter caching was added.
Benchmarking shows that even for a simple setup (basic html filtering + legacy URL rewriting), filtercache can offer speedups. Unlike the old filtercache, this uses the normal cache table.
7) Filtertips were moved from help into a hook_filter_tips(). This was required to accomodate the fact that there are multiple filters per module, and that filter settings are format dependant. Shoehorning filter tips into _help was ugly and silly. The display of the filter tips is done through the input format selector, so filter_tips_short() no longer exists.
8) A more intelligent linebreak convertor was added, which doesn't stop working if you use block-level tags and which adds <p> tags.
2004-08-10 18:34:29 +00:00
}
2004-11-24 22:44:01 +00:00
/**
* Returns the path to a system item ( module , theme , etc . ) .
*
* @ param $type
* The type of the item ( i . e . theme , theme_engine , module ) .
* @ param $name
* The name of the item for which the path is requested .
*
* @ return
* The path to the requested item .
*/
function drupal_get_path ( $type , $name ) {
return dirname ( drupal_get_filename ( $type , $name ));
}
2005-02-04 20:14:05 +00:00
/**
* Provide a substitute clone () function for PHP4 .
*/
2005-06-22 20:19:58 +00:00
function drupal_clone ( $object ) {
return version_compare ( phpversion (), '5.0' ) < 0 ? $object : clone ( $object );
2005-02-04 20:14:05 +00:00
}
2005-05-31 21:14:27 +00:00
/**
* Add a < link > tag to the page ' s HEAD .
*/
function drupal_add_link ( $attributes ) {
2005-06-05 19:10:53 +00:00
drupal_set_html_head ( '<link' . drupal_attributes ( $attributes ) . " /> \n " );
2005-05-31 21:14:27 +00:00
}
2005-05-24 06:00:22 +00:00
/**
* Add a JavaScript file to the output .
*
* The first time this function is invoked per page request ,
* it adds " misc/drupal.js " to the output . Other scripts
* depends on the 'killswitch' inside it .
*/
function drupal_add_js ( $file ) {
static $sent = array ();
if ( ! isset ( $sent [ 'misc/drupal.js' ])) {
drupal_set_html_head ( '<script type="text/javascript" src="misc/drupal.js"></script>' );
$sent [ 'misc/drupal.js' ] = true ;
}
if ( ! isset ( $sent [ $file ])) {
drupal_set_html_head ( '<script type="text/javascript" src="' . check_url ( $file ) . '"></script>' );
$sent [ $file ] = true ;
}
}
/**
* Implode a PHP array into a string that can be decoded by the autocomplete JS routines .
*
* Items are separated by double pipes . Each item consists of a key - value pair
* separated by single pipes . Entities are used to ensure pipes in the strings
* pass unharmed .
*
* The key is what is filled in in the text - box ( plain - text ), the value is what
* is displayed in the suggestion list ( HTML ) .
*/
function drupal_implode_autocomplete ( $array ) {
$output = array ();
foreach ( $array as $k => $v ) {
2005-05-24 20:38:00 +00:00
$output [] = str_replace ( '|' , '|' , $k ) . '|' . str_replace ( '|' , '|' , $v );
2005-05-24 06:00:22 +00:00
}
return implode ( '||' , $output );
}
2005-08-17 15:01:14 +00:00
/**
2005-08-17 19:14:08 +00:00
* Performs one or more XML - RPC request ( s ) .
*
* @ param $url
* An absolute URL of the XML - RPC endpoint .
* Example :
* http :// www . domain . com / xmlrpc . php
* @ param ...
* For one request :
* The method name followed by a variable number of arguments to the method .
* For multiple requests ( system . multicall ) :
* An array of call arrays . Each call array follows the pattern of the single
* request : method name followed by the arguments to the method .
* @ return
* For one request :
* Either the return value of the method on success , or FALSE .
* If FALSE is returned , see xmlrpc_errno () and xmlrpc_error_msg () .
* For multiple requests :
* An array of results . Each result will either be the result
* returned by the method called , or an xmlrpc_error object if the call
* failed . See xmlrpc_error () .
*/
2005-08-17 15:01:14 +00:00
function xmlrpc ( $url ) {
require_once './includes/xmlrpc.inc' ;
$args = func_get_args ();
return call_user_func_array ( '_xmlrpc' , $args );
}
2005-06-22 20:19:58 +00:00
function _drupal_bootstrap_full () {
static $called ;
2005-07-27 01:58:43 +00:00
global $locale ;
2003-03-17 07:01:12 +00:00
2005-06-22 20:19:58 +00:00
if ( $called ) {
return ;
}
$called = 1 ;
require_once './includes/theme.inc' ;
require_once './includes/pager.inc' ;
require_once './includes/menu.inc' ;
require_once './includes/tablesort.inc' ;
require_once './includes/file.inc' ;
2005-07-25 20:38:30 +00:00
require_once './includes/unicode.inc' ;
2005-06-22 20:19:58 +00:00
require_once './includes/image.inc' ;
// Set the Drupal custom error handler.
set_error_handler ( 'error_handler' );
// Emit the correct charset HTTP header.
drupal_set_header ( 'Content-Type: text/html; charset=utf-8' );
2005-07-25 20:40:35 +00:00
// Detect string handling method
2005-07-27 01:58:43 +00:00
unicode_check ();
2005-06-22 20:19:58 +00:00
// Initialize $_GET['q'] prior to loading modules and invoking hook_init().
if ( ! empty ( $_GET [ 'q' ])) {
$_GET [ 'q' ] = drupal_get_normal_path ( trim ( $_GET [ 'q' ], '/' ));
}
else {
$_GET [ 'q' ] = drupal_get_normal_path ( variable_get ( 'site_frontpage' , 'node' ));
}
// Initialize all enabled modules.
module_init ();
if ( ! user_access ( 'bypass input data check' )) {
// We can't use $_REQUEST because it consists of the contents of $_POST,
// $_GET and $_COOKIE: if any of the input arrays share a key, only one
// value will be verified.
if ( ! valid_input_data ( $_GET )
|| ! valid_input_data ( $_POST )
|| ! valid_input_data ( $_COOKIE )
|| ! valid_input_data ( $_FILES )) {
die ( 'Terminated request because of suspicious input data.' );
}
2003-11-18 19:44:36 +00:00
}
2005-06-22 20:19:58 +00:00
fix_gpc_magic ();
fix_checkboxes ();
// Initialize the localization system.
$locale = locale_initialize ();
2003-11-18 19:44:36 +00:00
}
2003-03-04 15:10:37 +00:00
?>