wiki.techinc.nl/includes/Preferences.php

1382 lines
40 KiB
PHP
Raw Normal View History

<?php
/**
General information about this file:
We're now using the HTMLForm object with some customisation to generate the Preferences
form. This object handles generic submission, CSRF protection, layout and other logic
in a reusable manner. We subclass it as a PreferencesForm to make some minor
customisations.
In order to generate the form, the HTMLForm object needs an array structure detailing the
form fields available, and that's what this class is for. Each element of the array is
a basic property-list, including the type of field, the label it is to be given in the
form, callbacks for validation and 'filtering', and other pertinent information. Note that
the 'default' field is named for generic forms, and does not represent the preference's
default (which is stored in $wgDefaultUserOptions), but the default for the form field,
which should be whatever the user has set for that preference. There is no need to
override it unless you have some special storage logic (for instance, those not presently
stored as options, but which are best set from the user preferences view).
Field types are implemented as subclasses of the generic HTMLFormField object, and
typically implement at least getInputHTML, which generates the HTML for the input field
to be placed in the table.
Once fields have been retrieved and validated, submission logic is handed over to the
tryUISubmit static method of this class.
*/
class Preferences {
static $defaultPreferences = null;
static $saveFilters =
array(
'timecorrection' => array( 'Preferences', 'filterTimezoneInput' ),
);
2009-06-21 14:16:11 +00:00
static function getPreferences( $user ) {
2009-06-21 14:16:11 +00:00
if ( self::$defaultPreferences )
return self::$defaultPreferences;
2009-06-21 14:16:11 +00:00
global $wgRCMaxAge;
$defaultPreferences = array();
self::profilePreferences( $user, $defaultPreferences );
self::skinPreferences( $user, $defaultPreferences );
self::filesPreferences( $user, $defaultPreferences );
self::mathPreferences( $user, $defaultPreferences );
self::datetimePreferences( $user, $defaultPreferences );
self::renderingPreferences( $user, $defaultPreferences );
self::editingPreferences( $user, $defaultPreferences );
self::rcPreferences( $user, $defaultPreferences );
self::watchlistPreferences( $user, $defaultPreferences );
self::searchPreferences( $user, $defaultPreferences );
self::miscPreferences( $user, $defaultPreferences );
wfRunHooks( 'GetPreferences', array( $user, &$defaultPreferences ) );
## Remove preferences that wikis don't want to use
global $wgHiddenPrefs;
foreach ( $wgHiddenPrefs as $pref ) {
if ( isset( $defaultPreferences[$pref] ) ) {
2009-06-21 14:16:11 +00:00
unset( $defaultPreferences[$pref] );
}
}
## Prod in defaults from the user
global $wgDefaultUserOptions;
foreach( $defaultPreferences as $name => &$info ) {
$prefFromUser = self::getOptionFromUser( $name, $info, $user );
$field = HTMLForm::loadInputFromParameters( $info ); // For validation
2009-06-09 17:32:33 +00:00
$defaultOptions = User::getDefaultOptions();
2009-06-21 14:16:11 +00:00
$globalDefault = isset( $defaultOptions[$name] )
2009-06-09 17:32:33 +00:00
? $defaultOptions[$name]
: null;
2009-06-21 14:16:11 +00:00
// If it validates, set it as the default
2009-06-21 14:16:11 +00:00
if ( isset( $info['default'] ) ) {
// Already set, no problem
continue;
} elseif ( !is_null( $prefFromUser ) && // Make sure we're not just pulling nothing
$field->validate( $prefFromUser, $user->mOptions ) === true ) {
$info['default'] = $prefFromUser;
} elseif( $field->validate( $globalDefault, $user->mOptions ) === true ) {
$info['default'] = $globalDefault;
} else {
throw new MWException( "Global default '$globalDefault' is invalid for field $name" );
}
}
2009-06-21 14:16:11 +00:00
self::$defaultPreferences = $defaultPreferences;
2009-06-21 14:16:11 +00:00
return $defaultPreferences;
}
2009-06-21 14:16:11 +00:00
// Pull option from a user account. Handles stuff like array-type preferences.
static function getOptionFromUser( $name, $info, $user ) {
$val = $user->getOption( $name );
2009-06-21 14:16:11 +00:00
// Handling for array-type preferences
2009-06-21 14:16:11 +00:00
if ( ( isset( $info['type'] ) && $info['type'] == 'multiselect' ) ||
( isset( $info['class'] ) && $info['class'] == 'HTMLMultiSelectField' ) ) {
2009-06-21 14:16:11 +00:00
$options = HTMLFormField::flattenOptions( $info['options'] );
$prefix = isset( $info['prefix'] ) ? $info['prefix'] : $name;
$val = array();
2009-06-21 14:16:11 +00:00
foreach( $options as $label => $value ) {
2009-06-21 14:16:11 +00:00
if( $user->getOption( "$prefix$value" ) ) {
$val[] = $value;
}
}
}
2009-06-21 14:16:11 +00:00
return $val;
}
2009-06-21 14:16:11 +00:00
static function profilePreferences( $user, &$defaultPreferences ) {
global $wgLang;
## User info #####################################
// Information panel
$defaultPreferences['username'] =
array(
'type' => 'info',
'label-message' => 'username',
'default' => $user->getName(),
'section' => 'personal/info',
);
2009-06-21 14:16:11 +00:00
$defaultPreferences['userid'] =
array(
'type' => 'info',
'label-message' => 'uid',
'default' => $user->getId(),
'section' => 'personal/info',
);
2009-06-21 14:16:11 +00:00
# Get groups to which the user belongs
$userEffectiveGroups = $user->getEffectiveGroups();
$userGroups = $userMembers = array();
foreach( $userEffectiveGroups as $ueg ) {
if( $ueg == '*' ) {
// Skip the default * group, seems useless here
continue;
}
$groupName = User::getGroupName( $ueg );
$userGroups[] = User::makeGroupLinkHTML( $ueg, $groupName );
$memberName = User::getGroupMember( $ueg );
$userMembers[] = User::makeGroupLinkHTML( $ueg, $memberName );
}
asort( $userGroups );
asort( $userMembers );
2009-06-21 14:16:11 +00:00
$defaultPreferences['usergroups'] =
array(
'type' => 'info',
'label' => wfMsgExt( 'prefs-memberingroups', 'parseinline',
$wgLang->formatNum( count($userGroups) ) ),
'default' => wfMsgExt( 'prefs-memberingroups-type', array(),
$wgLang->commaList( $userGroups ),
$wgLang->commaList( $userMembers )
),
'raw' => true,
'section' => 'personal/info',
);
2009-06-21 14:16:11 +00:00
$defaultPreferences['editcount'] =
array(
'type' => 'info',
'label-message' => 'prefs-edits',
'default' => $wgLang->formatNum( $user->getEditCount() ),
'section' => 'personal/info',
);
2009-06-21 14:16:11 +00:00
if( $user->getRegistration() ) {
$defaultPreferences['registrationdate'] =
array(
'type' => 'info',
'label-message' => 'prefs-registration',
'default' => wfMsgExt( 'prefs-registration-date-time', 'parsemag',
$wgLang->timeanddate( $user->getRegistration(), true ),
$wgLang->date( $user->getRegistration(), true ),
$wgLang->time( $user->getRegistration(), true ) ),
'section' => 'personal/info',
);
}
2009-06-21 14:16:11 +00:00
// Actually changeable stuff
global $wgAuth;
$defaultPreferences['realname'] =
array(
'type' => $wgAuth->allowPropChange( 'realname' ) ? 'text' : 'info',
'default' => $user->getRealName(),
'section' => 'personal/info',
'label-message' => 'yourrealname',
'help-message' => 'prefs-help-realname',
);
$defaultPreferences['gender'] =
array(
'type' => 'select',
'section' => 'personal/info',
'options' => array(
2009-06-21 14:16:11 +00:00
wfMsg( 'gender-male' ) => 'male',
wfMsg( 'gender-female' ) => 'female',
wfMsg( 'gender-unknown' ) => 'unknown',
),
'label-message' => 'yourgender',
'help-message' => 'prefs-help-gender',
);
2009-06-21 14:16:11 +00:00
if( $wgAuth->allowPasswordChange() ) {
global $wgUser; // For skin.
2009-05-24 09:23:40 +00:00
$link = $wgUser->getSkin()->link( SpecialPage::getTitleFor( 'Resetpass' ),
2009-06-21 14:16:11 +00:00
wfMsgHtml( 'prefs-resetpass' ), array(),
array( 'returnto' => SpecialPage::getTitleFor( 'Preferences' ) ) );
$defaultPreferences['password'] =
array(
'type' => 'info',
'raw' => true,
'default' => $link,
'label-message' => 'yourpassword',
'section' => 'personal/info',
);
}
2009-06-21 14:16:11 +00:00
$defaultPreferences['rememberpassword'] =
array(
'type' => 'toggle',
'label-message' => 'tog-rememberpassword',
'section' => 'personal/info',
);
2009-06-21 14:16:11 +00:00
// Language
global $wgContLanguageCode;
$languages = array_reverse( Language::getLanguageNames( false ) );
if( !array_key_exists( $wgContLanguageCode, $languages ) ) {
$languages[$wgContLanguageCode] = $wgContLanguageCode;
}
ksort( $languages );
2009-06-21 14:16:11 +00:00
$options = array();
foreach( $languages as $code => $name ) {
$display = wfBCP47( $code ) . ' - ' . $name;
$options[$display] = $code;
}
$defaultPreferences['language'] =
array(
'type' => 'select',
'section' => 'personal/i18n',
'options' => $options,
'label-message' => 'yourlanguage',
);
2009-06-21 14:16:11 +00:00
global $wgContLang, $wgDisableLangConversion;
2009-07-08 17:07:39 +00:00
global $wgDisableTitleConversion;
/* see if there are multiple language variants to choose from*/
$variantArray = array();
2009-06-21 14:16:11 +00:00
if( !$wgDisableLangConversion ) {
$variants = $wgContLang->getVariants();
$languages = Language::getLanguageNames( true );
2009-06-21 14:16:11 +00:00
foreach( $variants as $v ) {
$v = str_replace( '_', '-', strtolower( $v ) );
if( array_key_exists( $v, $languages ) ) {
// If it doesn't have a name, we'll pretend it doesn't exist
$variantArray[$v] = $languages[$v];
}
}
$options = array();
foreach( $variantArray as $code => $name ) {
$display = wfBCP47( $code ) . ' - ' . $name;
$options[$display] = $code;
}
2009-06-21 14:16:11 +00:00
if( count( $variantArray ) > 1 ) {
$defaultPreferences['variant'] =
array(
'label-message' => 'yourvariant',
'type' => 'select',
'options' => $options,
'section' => 'personal/i18n',
);
}
}
2009-06-21 14:16:11 +00:00
if( count( $variantArray ) > 1 && !$wgDisableLangConversion && !$wgDisableTitleConversion ) {
$defaultPreferences['noconvertlink'] =
array(
'type' => 'toggle',
'section' => 'personal/i18n',
'label-message' => 'tog-noconvertlink',
);
}
2009-06-21 14:16:11 +00:00
global $wgMaxSigChars, $wgParser;
// show a preview of the old signature first
$oldsigtext = $wgParser->preSaveTransform( "~~~", new Title , $user, new ParserOptions );
$oldsig = $wgParser->parse( $oldsigtext, new Title , new ParserOptions );
$m = array(); // remove <p> created by the parser (looks better without <p>)
if( preg_match( '/^<p>(.*)\n?<\/p>\n?$/sU', $oldsig->mText, $m ) ) $oldsig->mText = $m[1];
$defaultPreferences['oldsig'] =
array(
'type' => 'info',
'raw' => true,
'label-message' => 'tog-oldsig',
'default' => $oldsig->mText,
'section' => 'personal/signature',
);
$defaultPreferences['nickname'] =
array(
'type' => $wgAuth->allowPropChange( 'nickname' ) ? 'text' : 'info',
'maxlength' => $wgMaxSigChars,
'label-message' => 'yournick',
'validation-callback' =>
array( 'Preferences', 'validateSignature' ),
'section' => 'personal/signature',
'filter-callback' => array( 'Preferences', 'cleanSignature' ),
);
$defaultPreferences['fancysig'] =
array(
'type' => 'toggle',
'label-message' => 'tog-fancysig',
'help-message' => 'prefs-help-signature', // show general help about signature at the bottom of the section
'section' => 'personal/signature'
);
## Email stuff
global $wgEnableEmail;
if ($wgEnableEmail) {
global $wgEmailConfirmToEdit;
$defaultPreferences['emailaddress'] =
array(
'type' => $wgAuth->allowPropChange( 'emailaddress' ) ? 'text' : 'info',
'default' => $user->getEmail(),
'section' => 'personal/email',
'label-message' => 'youremail',
'help-message' => $wgEmailConfirmToEdit
? 'prefs-help-email-required'
: 'prefs-help-email',
'validation-callback' => array( 'Preferences', 'validateEmail' ),
);
global $wgEnableUserEmail, $wgEmailAuthentication;
$disableEmailPrefs = false;
if ( $wgEmailAuthentication ) {
if ( $user->getEmail() ) {
if( $user->getEmailAuthenticationTimestamp() ) {
// date and time are separate parameters to facilitate localisation.
// $time is kept for backward compat reasons.
// 'emailauthenticated' is also used in SpecialConfirmemail.php
$time = $wgLang->timeAndDate( $user->getEmailAuthenticationTimestamp(), true );
$d = $wgLang->date( $user->getEmailAuthenticationTimestamp(), true );
$t = $wgLang->time( $user->getEmailAuthenticationTimestamp(), true );
$emailauthenticated = wfMsgExt( 'emailauthenticated', 'parseinline',
array($time, $d, $t ) ) . '<br />';
$disableEmailPrefs = false;
} else {
$disableEmailPrefs = true;
global $wgUser; // wgUser is okay here, it's for display
$skin = $wgUser->getSkin();
$emailauthenticated = wfMsgExt( 'emailnotauthenticated', 'parseinline' ) . '<br />' .
$skin->link(
SpecialPage::getTitleFor( 'Confirmemail' ),
wfMsg( 'emailconfirmlink' ),
array(),
array(),
array( 'known', 'noclasses' )
) . '<br />';
}
} else {
$disableEmailPrefs = true;
$emailauthenticated = wfMsgHtml( 'noemailprefs' );
}
$defaultPreferences['emailauthentication'] =
array(
'type' => 'info',
'raw' => true,
'section' => 'personal/email',
'label-message' => 'prefs-emailconfirm-label',
'default' => $emailauthenticated,
);
}
2009-06-21 14:16:11 +00:00
if( $wgEnableUserEmail ) {
$defaultPreferences['disablemail'] =
array(
'type' => 'toggle',
'invert' => true,
'section' => 'personal/email',
'label-message' => 'allowemail',
'disabled' => $disableEmailPrefs,
);
$defaultPreferences['ccmeonemails'] =
array(
'type' => 'toggle',
'section' => 'personal/email',
'label-message' => 'tog-ccmeonemails',
'disabled' => $disableEmailPrefs,
);
}
global $wgEnotifWatchlist;
if ( $wgEnotifWatchlist ) {
$defaultPreferences['enotifwatchlistpages'] =
array(
'type' => 'toggle',
'section' => 'personal/email',
'label-message' => 'tog-enotifwatchlistpages',
'disabled' => $disableEmailPrefs,
);
}
global $wgEnotifUserTalk;
if( $wgEnotifUserTalk ) {
$defaultPreferences['enotifusertalkpages'] =
array(
'type' => 'toggle',
'section' => 'personal/email',
'label-message' => 'tog-enotifusertalkpages',
'disabled' => $disableEmailPrefs,
);
}
if( $wgEnotifUserTalk || $wgEnotifWatchlist ) {
$defaultPreferences['enotifminoredits'] =
array(
'type' => 'toggle',
'section' => 'personal/email',
'label-message' => 'tog-enotifminoredits',
'disabled' => $disableEmailPrefs,
);
}
$defaultPreferences['enotifrevealaddr'] =
array(
'type' => 'toggle',
'section' => 'personal/email',
'label-message' => 'tog-enotifrevealaddr',
'disabled' => $disableEmailPrefs,
);
}
}
2009-06-21 14:16:11 +00:00
static function skinPreferences( $user, &$defaultPreferences ) {
## Skin #####################################
$defaultPreferences['skin'] =
array(
'type' => 'radio',
'options' => self::generateSkinOptions( $user ),
'label' => '&nbsp;',
'section' => 'rendering/skin',
);
2009-06-21 14:16:11 +00:00
2009-04-27 01:49:09 +00:00
$selectedSkin = $user->getOption( 'skin' );
if ( in_array( $selectedSkin, array( 'cologneblue', 'standard' ) ) ) {
global $wgLang;
2009-06-21 14:16:11 +00:00
$settings = array_flip( $wgLang->getQuickbarSettings() );
2009-04-27 01:49:09 +00:00
$defaultPreferences['quickbar'] =
array(
'type' => 'radio',
'options' => $settings,
'section' => 'rendering/skin',
2009-04-27 01:49:09 +00:00
'label-message' => 'qbsettings',
);
}
}
2009-06-21 14:16:11 +00:00
static function mathPreferences( $user, &$defaultPreferences ) {
## Math #####################################
global $wgUseTeX, $wgLang;
2009-06-21 14:16:11 +00:00
if( $wgUseTeX ) {
$defaultPreferences['math'] =
array(
'type' => 'radio',
'options' =>
2009-05-22 09:35:48 +00:00
array_flip( array_map( 'wfMsgHtml', $wgLang->getMathNames() ) ),
'label' => '&nbsp;',
'section' => 'rendering/math',
);
}
}
2009-06-21 14:16:11 +00:00
static function filesPreferences( $user, &$defaultPreferences ) {
## Files #####################################
$defaultPreferences['imagesize'] =
array(
'type' => 'select',
'options' => self::getImageSizes(),
'label-message' => 'imagemaxsize',
'section' => 'rendering/files',
);
$defaultPreferences['thumbsize'] =
array(
'type' => 'select',
'options' => self::getThumbSizes(),
'label-message' => 'thumbsize',
'section' => 'rendering/files',
);
}
2009-06-21 14:16:11 +00:00
static function datetimePreferences( $user, &$defaultPreferences ) {
global $wgLang;
2009-06-21 14:16:11 +00:00
## Date and time #####################################
$dateOptions = self::getDateOptions();
2009-06-21 14:16:11 +00:00
if( $dateOptions ) {
$defaultPreferences['date'] =
array(
'type' => 'radio',
'options' => $dateOptions,
'label' => '&nbsp;',
'section' => 'datetime/dateformat',
);
}
2009-06-21 14:16:11 +00:00
// Info
$nowlocal = Xml::element( 'span', array( 'id' => 'wpLocalTime' ),
$wgLang->time( $now = wfTimestampNow(), true ) );
$nowserver = $wgLang->time( $now, false ) .
Xml::hidden( 'wpServerTime', substr( $now, 8, 2 ) * 60 + substr( $now, 10, 2 ) );
2009-06-21 14:16:11 +00:00
$defaultPreferences['nowserver'] =
array(
'type' => 'info',
'raw' => 1,
'label-message' => 'servertime',
'default' => $nowserver,
'section' => 'datetime/timeoffset',
);
2009-06-21 14:16:11 +00:00
$defaultPreferences['nowlocal'] =
array(
'type' => 'info',
'raw' => 1,
'label-message' => 'localtime',
'default' => $nowlocal,
'section' => 'datetime/timeoffset',
);
2009-06-21 14:16:11 +00:00
// Grab existing pref.
$tzOffset = $user->getOption( 'timecorrection' );
$tz = explode( '|', $tzOffset, 2 );
2009-06-21 14:16:11 +00:00
$tzSetting = $tzOffset;
2009-06-21 14:16:11 +00:00
if( count( $tz ) > 1 && $tz[0] == 'Offset' ) {
$minDiff = $tz[1];
2009-06-21 14:16:11 +00:00
$tzSetting = sprintf( '%+03d:%02d', floor( $minDiff/60 ), abs( $minDiff )%60 );
}
2009-06-21 14:16:11 +00:00
$defaultPreferences['timecorrection'] =
array(
'class' => 'HTMLSelectOrOtherField',
'label-message' => 'timezonelegend',
'options' => self::getTimezoneOptions(),
'default' => $tzSetting,
'section' => 'datetime/timeoffset',
);
}
2009-06-21 14:16:11 +00:00
static function renderingPreferences( $user, &$defaultPreferences ) {
## Page Rendering ##############################
$defaultPreferences['underline'] =
array(
'type' => 'select',
'options' => array(
wfMsg( 'underline-never' ) => 0,
wfMsg( 'underline-always' ) => 1,
wfMsg( 'underline-default' ) => 2,
),
'label-message' => 'tog-underline',
'section' => 'rendering/advancedrendering',
);
2009-06-21 14:16:11 +00:00
$stubThresholdValues = array( 0, 50, 100, 500, 1000, 2000, 5000, 10000 );
$stubThresholdOptions = array();
foreach( $stubThresholdValues as $value ) {
$stubThresholdOptions[wfMsg( 'size-bytes', $value )] = $value;
}
2009-06-21 14:16:11 +00:00
$defaultPreferences['stubthreshold'] =
array(
'type' => 'selectorother',
'section' => 'rendering/advancedrendering',
'options' => $stubThresholdOptions,
2009-06-21 14:16:11 +00:00
'label' => wfMsg( 'stub-threshold' ), // Raw HTML message. Yay?
);
$defaultPreferences['highlightbroken'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
2009-06-21 14:16:11 +00:00
'label' => wfMsg( 'tog-highlightbroken' ), // Raw HTML
);
$defaultPreferences['showtoc'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
'label-message' => 'tog-showtoc',
);
$defaultPreferences['nocache'] =
array(
'type' => 'toggle',
'label-message' => 'tog-nocache',
'section' => 'rendering/advancedrendering',
);
$defaultPreferences['showhiddencats'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
'label-message' => 'tog-showhiddencats'
);
$defaultPreferences['showjumplinks'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
'label-message' => 'tog-showjumplinks',
);
$defaultPreferences['justify'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
'label-message' => 'tog-justify',
);
$defaultPreferences['numberheadings'] =
array(
'type' => 'toggle',
'section' => 'rendering/advancedrendering',
'label-message' => 'tog-numberheadings',
);
}
2009-06-21 14:16:11 +00:00
static function editingPreferences( $user, &$defaultPreferences ) {
global $wgUseExternalEditor, $wgLivePreview;
## Editing #####################################
$defaultPreferences['cols'] =
array(
'type' => 'int',
'label-message' => 'columns',
'section' => 'editing/textboxsize',
'min' => 4,
'max' => 1000,
);
$defaultPreferences['rows'] =
array(
'type' => 'int',
'label-message' => 'rows',
'section' => 'editing/textboxsize',
'min' => 4,
'max' => 1000,
);
$defaultPreferences['editfont'] =
array(
'type' => 'select',
'section' => 'editing/advancedediting',
'label-message' => 'editfont-style',
'options' => array(
wfMsg( 'editfont-default' ) => 'default',
wfMsg( 'editfont-monospace' ) => 'monospace',
wfMsg( 'editfont-sansserif' ) => 'sans-serif',
wfMsg( 'editfont-serif' ) => 'serif',
)
);
$defaultPreferences['previewontop'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-previewontop',
);
$defaultPreferences['previewonfirst'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-previewonfirst',
);
$defaultPreferences['editsection'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-editsection',
);
$defaultPreferences['editsectiononrightclick'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-editsectiononrightclick',
);
$defaultPreferences['editondblclick'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-editondblclick',
);
$defaultPreferences['editwidth'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-editwidth',
);
$defaultPreferences['showtoolbar'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-showtoolbar',
);
$defaultPreferences['minordefault'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-minordefault',
);
if ( $wgUseExternalEditor ) {
$defaultPreferences['externaleditor'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-externaleditor',
);
$defaultPreferences['externaldiff'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-externaldiff',
);
}
$defaultPreferences['forceeditsummary'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-forceeditsummary',
);
if ( $wgLivePreview ) {
$defaultPreferences['uselivepreview'] =
array(
'type' => 'toggle',
'section' => 'editing/advancedediting',
'label-message' => 'tog-uselivepreview',
);
}
}
2009-06-21 14:16:11 +00:00
static function rcPreferences( $user, &$defaultPreferences ) {
global $wgRCMaxAge, $wgUseRCPatrol, $wgLang;
## RecentChanges #####################################
$defaultPreferences['rcdays'] =
array(
'type' => 'float',
'label-message' => 'recentchangesdays',
'section' => 'rc/display',
'min' => 1,
2009-06-21 14:16:11 +00:00
'max' => ceil( $wgRCMaxAge / ( 3600*24 ) ),
'help' => wfMsgExt( 'recentchangesdays-max', array( 'parsemag' ), $wgLang->formatNum( ceil( $wgRCMaxAge / ( 3600*24 ) ) ) ),
);
$defaultPreferences['rclimit'] =
array(
'type' => 'int',
'label-message' => 'recentchangescount',
'help-message' => 'prefs-help-recentchangescount',
'section' => 'rc/display',
);
$defaultPreferences['usenewrc'] =
array(
'type' => 'toggle',
'label-message' => 'tog-usenewrc',
'section' => 'rc/advancedrc',
);
$defaultPreferences['hideminor'] =
array(
'type' => 'toggle',
'label-message' => 'tog-hideminor',
'section' => 'rc/advancedrc',
);
2009-06-21 14:16:11 +00:00
global $wgUseRCPatrol;
2009-06-21 14:16:11 +00:00
if( $wgUseRCPatrol ) {
$defaultPreferences['hidepatrolled'] =
array(
'type' => 'toggle',
'section' => 'rc/advancedrc',
'label-message' => 'tog-hidepatrolled',
);
$defaultPreferences['newpageshidepatrolled'] =
array(
'type' => 'toggle',
'section' => 'rc/advancedrc',
'label-message' => 'tog-newpageshidepatrolled',
);
}
2009-06-21 14:16:11 +00:00
global $wgRCShowWatchingUsers;
2009-06-21 14:16:11 +00:00
if( $wgRCShowWatchingUsers ) {
$defaultPreferences['shownumberswatching'] =
array(
'type' => 'toggle',
'section' => 'rc/advancedrc',
'label-message' => 'tog-shownumberswatching',
);
}
}
2009-06-21 14:16:11 +00:00
static function watchlistPreferences( $user, &$defaultPreferences ) {
global $wgUseRCPatrol, $wgEnableAPI;
## Watchlist #####################################
$defaultPreferences['watchlistdays'] =
array(
'type' => 'float',
'min' => 0,
'max' => 7,
'section' => 'watchlist/display',
'help' => wfMsgHtml( 'prefs-watchlist-days-max' ),
'label-message' => 'prefs-watchlist-days',
);
$defaultPreferences['wllimit'] =
array(
'type' => 'int',
'min' => 0,
'max' => 1000,
'label-message' => 'prefs-watchlist-edits',
'help' => wfMsgHtml( 'prefs-watchlist-edits-max' ),
'section' => 'watchlist/display',
);
$defaultPreferences['extendwatchlist'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-extendwatchlist',
);
$defaultPreferences['watchlisthideminor'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthideminor',
);
$defaultPreferences['watchlisthidebots'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthidebots',
);
$defaultPreferences['watchlisthideown'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthideown',
);
$defaultPreferences['watchlisthideanons'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthideanons',
);
$defaultPreferences['watchlisthideliu'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthideliu',
);
if ( $wgEnableAPI ) {
# Some random gibberish as a proposed default
$hash = sha1( mt_rand() . microtime( true ) );
$defaultPreferences['watchlisttoken'] =
array(
'type' => 'text',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'prefs-watchlist-token',
'help' => wfMsgHtml( 'prefs-help-watchlist-token', $hash )
);
}
2009-06-21 14:16:11 +00:00
if ( $wgUseRCPatrol ) {
$defaultPreferences['watchlisthidepatrolled'] =
array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => 'tog-watchlisthidepatrolled',
);
}
2009-06-21 14:16:11 +00:00
$watchTypes = array(
'edit' => 'watchdefault',
'move' => 'watchmoves',
'delete' => 'watchdeletion'
);
// Kinda hacky
if( $user->isAllowed( 'createpage' ) || $user->isAllowed( 'createtalk' ) ) {
$watchTypes['read'] = 'watchcreations';
}
2009-06-21 14:16:11 +00:00
foreach( $watchTypes as $action => $pref ) {
if ( $user->isAllowed( $action ) ) {
$defaultPreferences[$pref] = array(
'type' => 'toggle',
'section' => 'watchlist/advancedwatchlist',
'label-message' => "tog-$pref",
);
}
}
}
2009-06-21 14:16:11 +00:00
static function searchPreferences( $user, &$defaultPreferences ) {
global $wgContLang;
2009-06-21 14:16:11 +00:00
## Search #####################################
$defaultPreferences['searchlimit'] =
array(
'type' => 'int',
'label-message' => 'resultsperpage',
'section' => 'searchoptions/display',
'min' => 0,
);
$defaultPreferences['contextlines'] =
array(
'type' => 'int',
'label-message' => 'contextlines',
'section' => 'searchoptions/display',
'min' => 0,
);
$defaultPreferences['contextchars'] =
array(
'type' => 'int',
'label-message' => 'contextchars',
'section' => 'searchoptions/display',
'min' => 0,
);
global $wgEnableMWSuggest;
2009-06-21 14:16:11 +00:00
if( $wgEnableMWSuggest ) {
$defaultPreferences['disablesuggest'] =
array(
'type' => 'toggle',
'label-message' => 'mwsuggest-disable',
'section' => 'searchoptions/display',
);
}
2009-06-21 14:16:11 +00:00
$defaultPreferences['searcheverything'] =
array(
'type' => 'toggle',
'label-message' => 'searcheverything-enable',
'section' => 'searchoptions/advancedsearchoptions',
);
2009-06-21 14:16:11 +00:00
// Searchable namespaces back-compat with old format
$searchableNamespaces = SearchEngine::searchableNamespaces();
2009-06-21 14:16:11 +00:00
$nsOptions = array();
foreach( $wgContLang->getNamespaces() as $ns => $name ) {
2009-06-21 14:16:11 +00:00
if( $ns < 0 ) continue;
$displayNs = str_replace( '_', ' ', $name );
2009-06-21 14:16:11 +00:00
if( !$displayNs ) $displayNs = wfMsg( 'blanknamespace' );
2009-05-22 09:35:48 +00:00
$displayNs = htmlspecialchars( $displayNs );
$nsOptions[$displayNs] = $ns;
}
2009-06-21 14:16:11 +00:00
$defaultPreferences['searchnamespaces'] =
array(
'type' => 'multiselect',
'label-message' => 'defaultns',
'options' => $nsOptions,
'section' => 'searchoptions/advancedsearchoptions',
'prefix' => 'searchNs',
);
}
2009-06-21 14:16:11 +00:00
static function miscPreferences( $user, &$defaultPreferences ) {
## Misc #####################################
$defaultPreferences['diffonly'] =
array(
'type' => 'toggle',
'section' => 'misc/diffs',
'label-message' => 'tog-diffonly',
);
$defaultPreferences['norollbackdiff'] =
array(
'type' => 'toggle',
'section' => 'misc/diffs',
'label-message' => 'tog-norollbackdiff',
);
2009-06-21 14:16:11 +00:00
// Stuff from Language::getExtraUserToggles()
global $wgContLang;
2009-06-21 14:16:11 +00:00
$toggles = $wgContLang->getExtraUserToggles();
2009-06-21 14:16:11 +00:00
foreach( $toggles as $toggle ) {
$defaultPreferences[$toggle] =
array(
'type' => 'toggle',
'section' => 'personal/i18n',
'label-message' => "tog-$toggle",
);
}
}
2009-06-21 14:16:11 +00:00
static function generateSkinOptions( $user ) {
global $wgDefaultSkin;
$ret = array();
2009-06-21 14:16:11 +00:00
$mptitle = Title::newMainPage();
2009-05-22 09:35:48 +00:00
$previewtext = wfMsgHtml( 'skin-preview' );
# Only show members of Skin::getSkinNames() rather than
# $skinNames (skins is all skin names from Language.php)
$validSkinNames = Skin::getUsableSkins();
# Sort by UI skin name. First though need to update validSkinNames as sometimes
# the skinkey & UI skinname differ (e.g. "standard" skinkey is "Classic" in the UI).
foreach ( $validSkinNames as $skinkey => &$skinname ) {
$msgName = "skinname-{$skinkey}";
$localisedSkinName = wfMsg( $msgName );
2009-06-21 14:16:11 +00:00
if ( !wfEmptyMsg( $msgName, $localisedSkinName ) ) {
$skinname = htmlspecialchars( $localisedSkinName );
}
}
2009-06-21 14:16:11 +00:00
asort( $validSkinNames );
$sk = $user->getSkin();
foreach( $validSkinNames as $skinkey => $sn ) {
$mplink = htmlspecialchars( $mptitle->getLocalURL( "useskin=$skinkey" ) );
$previewlink = "(<a target='_blank' href=\"$mplink\">$previewtext</a>)";
$extraLinks = '';
global $wgAllowUserCss, $wgAllowUserJs;
if( $wgAllowUserCss ) {
2009-06-21 14:16:11 +00:00
$cssPage = Title::makeTitleSafe( NS_USER, $user->getName() . '/' . $skinkey . '.css' );
2009-05-22 09:35:48 +00:00
$customCSS = $sk->link( $cssPage, wfMsgHtml( 'prefs-custom-css' ) );
$extraLinks .= " ($customCSS)";
}
if( $wgAllowUserJs ) {
2009-06-21 14:16:11 +00:00
$jsPage = Title::makeTitleSafe( NS_USER, $user->getName() . '/' . $skinkey . '.js' );
$customJS = $sk->link( $jsPage, wfMsgHtml( 'prefs-custom-js' ) );
$extraLinks .= " ($customJS)";
}
if( $skinkey == $wgDefaultSkin )
2009-05-22 09:35:48 +00:00
$sn .= ' (' . wfMsgHtml( 'default' ) . ')';
$display = "$sn $previewlink{$extraLinks}";
$ret[$display] = $skinkey;
}
2009-06-21 14:16:11 +00:00
return $ret;
}
2009-06-21 14:16:11 +00:00
static function getDateOptions() {
global $wgLang;
$dateopts = $wgLang->getDatePreferences();
2009-06-21 14:16:11 +00:00
$ret = array();
2009-06-21 14:16:11 +00:00
if( $dateopts ) {
if ( !in_array( 'default', $dateopts ) ) {
$dateopts[] = 'default'; // Make sure default is always valid
2009-06-21 14:16:11 +00:00
// Bug 19237
}
2009-06-21 14:16:11 +00:00
$idCnt = 0;
$epoch = '20010115161234'; # Wikipedia day
foreach( $dateopts as $key ) {
if( $key == 'default' ) {
2009-05-22 09:35:48 +00:00
$formatted = wfMsgHtml( 'datedefault' );
} else {
2009-06-21 14:16:11 +00:00
$formatted = htmlspecialchars( $wgLang->timeanddate( $epoch, false, $key ) );
}
$ret[$formatted] = $key;
}
}
return $ret;
}
2009-06-21 14:16:11 +00:00
static function getImageSizes() {
global $wgImageLimits;
2009-06-21 14:16:11 +00:00
$ret = array();
2009-06-21 14:16:11 +00:00
foreach ( $wgImageLimits as $index => $limits ) {
2009-06-21 14:16:11 +00:00
$display = "{$limits[0]}×{$limits[1]}" . wfMsg( 'unit-pixel' );
$ret[$display] = $index;
}
2009-06-21 14:16:11 +00:00
return $ret;
}
2009-06-21 14:16:11 +00:00
static function getThumbSizes() {
global $wgThumbLimits;
2009-06-21 14:16:11 +00:00
$ret = array();
2009-06-21 14:16:11 +00:00
foreach ( $wgThumbLimits as $index => $size ) {
2009-06-21 14:16:11 +00:00
$display = $size . wfMsg( 'unit-pixel' );
$ret[$display] = $index;
}
2009-06-21 14:16:11 +00:00
return $ret;
}
2009-06-21 14:16:11 +00:00
static function validateSignature( $signature, $alldata ) {
global $wgParser, $wgMaxSigChars, $wgLang;
if( mb_strlen( $signature ) > $wgMaxSigChars ) {
return
Xml::element( 'span', array( 'class' => 'error' ),
wfMsgExt( 'badsiglength', 'parsemag',
$wgLang->formatNum( $wgMaxSigChars )
)
);
} elseif( !empty( $alldata['fancysig'] ) &&
false === $wgParser->validateSig( $signature ) ) {
return Xml::element( 'span', array( 'class' => 'error' ), wfMsg( 'badsig' ) );
} else {
return true;
}
}
2009-06-21 14:16:11 +00:00
static function cleanSignature( $signature, $alldata ) {
global $wgParser;
if( $alldata['fancysig'] ) {
$signature = $wgParser->cleanSig( $signature );
} else {
// When no fancy sig used, make sure ~{3,5} get removed.
$signature = $wgParser->cleanSigInSig( $signature );
}
2009-06-21 14:16:11 +00:00
return $signature;
}
2009-06-21 14:16:11 +00:00
static function validateEmail( $email, $alldata ) {
if ( $email && !User::isValidEmailAddr( $email ) ) {
return wfMsgExt( 'invalidemailaddress', 'parseinline' );
}
2009-06-21 14:16:11 +00:00
global $wgEmailConfirmToEdit;
if( $wgEmailConfirmToEdit && !$email ) {
return wfMsgExt( 'noemailtitle', 'parseinline' );
}
return true;
}
2009-06-21 14:16:11 +00:00
static function getFormObject( $user ) {
$formDescriptor = Preferences::getPreferences( $user );
$htmlForm = new PreferencesForm( $formDescriptor, 'prefs' );
2009-06-21 14:16:11 +00:00
$htmlForm->setSubmitText( wfMsg( 'saveprefs' ) );
$htmlForm->setTitle( SpecialPage::getTitleFor( 'Preferences' ) );
$htmlForm->setSubmitID( 'prefsubmit' );
$htmlForm->setSubmitCallback( array( 'Preferences', 'tryFormSubmit' ) );
2009-06-21 14:16:11 +00:00
return $htmlForm;
}
2009-06-21 14:16:11 +00:00
static function getTimezoneOptions() {
$opt = array();
2009-06-21 14:16:11 +00:00
global $wgLocalTZoffset;
$opt[wfMsg( 'timezoneuseserverdefault' )] = "System|$wgLocalTZoffset";
$opt[wfMsg( 'timezoneuseoffset' )] = 'other';
$opt[wfMsg( 'guesstimezone' )] = 'guess';
if ( function_exists( 'timezone_identifiers_list' ) ) {
# Read timezone list
$tzs = timezone_identifiers_list();
sort( $tzs );
$tzRegions = array();
$tzRegions['Africa'] = wfMsg( 'timezoneregion-africa' );
$tzRegions['America'] = wfMsg( 'timezoneregion-america' );
$tzRegions['Antarctica'] = wfMsg( 'timezoneregion-antarctica' );
$tzRegions['Arctic'] = wfMsg( 'timezoneregion-arctic' );
$tzRegions['Asia'] = wfMsg( 'timezoneregion-asia' );
$tzRegions['Atlantic'] = wfMsg( 'timezoneregion-atlantic' );
$tzRegions['Australia'] = wfMsg( 'timezoneregion-australia' );
$tzRegions['Europe'] = wfMsg( 'timezoneregion-europe' );
$tzRegions['Indian'] = wfMsg( 'timezoneregion-indian' );
$tzRegions['Pacific'] = wfMsg( 'timezoneregion-pacific' );
asort( $tzRegions );
2009-06-21 14:16:11 +00:00
$prefill = array_fill_keys( array_values( $tzRegions ), array() );
$opt = array_merge( $opt, $prefill );
$now = date_create( 'now' );
foreach ( $tzs as $tz ) {
$z = explode( '/', $tz, 2 );
# timezone_identifiers_list() returns a number of
# backwards-compatibility entries. This filters them out of the
# list presented to the user.
if ( count( $z ) != 2 || !array_key_exists( $z[0], $tzRegions ) )
continue;
# Localize region
$z[0] = $tzRegions[$z[0]];
$minDiff = floor( timezone_offset_get( timezone_open( $tz ), $now ) / 60 );
$display = str_replace( '_', ' ', $z[0] . '/' . $z[1] );
$value = "ZoneInfo|$minDiff|$tz";
$opt[$z[0]][$display] = $value;
}
}
return $opt;
}
2009-06-21 14:16:11 +00:00
static function filterTimezoneInput( $tz, $alldata ) {
$data = explode( '|', $tz, 3 );
switch ( $data[0] ) {
case 'ZoneInfo':
case 'System':
return $tz;
default:
$data = explode( ':', $tz, 2 );
$minDiff = 0;
if( count( $data ) == 2 ) {
$data[0] = intval( $data[0] );
$data[1] = intval( $data[1] );
$minDiff = abs( $data[0] ) * 60 + $data[1];
if ( $data[0] < 0 ) $minDiff = -$minDiff;
} else {
$minDiff = intval( $data[0] ) * 60;
}
# Max is +14:00 and min is -12:00, see:
# http://en.wikipedia.org/wiki/Timezone
$minDiff = min( $minDiff, 840 ); # 14:00
$minDiff = max( $minDiff, -720 ); # -12:00
return 'Offset|'.$minDiff;
}
}
static function tryFormSubmit( $formData, $entryPoint = 'internal' ) {
global $wgUser, $wgEmailAuthentication, $wgEnableEmail;
2009-06-21 14:16:11 +00:00
$result = true;
2009-06-21 14:16:11 +00:00
// Filter input
2009-06-21 14:16:11 +00:00
foreach( array_keys( $formData ) as $name ) {
if ( isset( self::$saveFilters[$name] ) ) {
$formData[$name] =
call_user_func( self::$saveFilters[$name], $formData[$name], $formData );
}
}
2009-06-21 14:16:11 +00:00
// Stuff that shouldn't be saved as a preference.
$saveBlacklist = array(
2009-06-21 14:16:11 +00:00
'realname',
'emailaddress',
);
if( $wgEnableEmail ) {
$newadr = $formData['emailaddress'];
$oldadr = $wgUser->getEmail();
2009-06-21 14:16:11 +00:00
if( ( $newadr != '' ) && ( $newadr != $oldadr ) ) {
# the user has supplied a new email address on the login page
# new behaviour: set this new emailaddr from login-page into user database record
$wgUser->setEmail( $newadr );
# but flag as "dirty" = unauthenticated
$wgUser->invalidateEmail();
2009-06-21 14:16:11 +00:00
if( $wgEmailAuthentication ) {
# Mail a temporary password to the dirty address.
# User can come back through the confirmation URL to re-enable email.
$result = $wgUser->sendConfirmationMail();
if( WikiError::isError( $result ) ) {
return wfMsg( 'mailerror', htmlspecialchars( $result->getMessage() ) );
2009-06-21 14:16:11 +00:00
} elseif( $entryPoint == 'ui' ) {
$result = 'eauth';
}
}
} else {
$wgUser->setEmail( $newadr );
}
if( $oldadr != $newadr ) {
wfRunHooks( 'PrefsEmailAudit', array( $wgUser, $oldadr, $newadr ) );
}
}
2009-06-21 14:16:11 +00:00
// Fortunately, the realname field is MUCH simpler
global $wgHiddenPrefs;
if ( !in_array( 'realname', $wgHiddenPrefs ) ) {
$realName = $formData['realname'];
$wgUser->setRealName( $realName );
}
2009-06-21 14:16:11 +00:00
foreach( $saveBlacklist as $b )
unset( $formData[$b] );
2009-06-21 14:16:11 +00:00
// Keeps old preferences from interfering due to back-compat
// code, etc.
$wgUser->resetOptions();
2009-06-21 14:16:11 +00:00
foreach( $formData as $key => $value ) {
$wgUser->setOption( $key, $value );
}
2009-06-21 14:16:11 +00:00
$wgUser->saveSettings();
2009-06-21 14:16:11 +00:00
return $result;
}
2009-06-21 14:16:11 +00:00
public static function tryUISubmit( $formData ) {
$res = self::tryFormSubmit( $formData, 'ui' );
2009-06-21 14:16:11 +00:00
if( $res ) {
$urlOptions = array( 'success' );
2009-06-21 14:16:11 +00:00
if( $res === 'eauth' )
$urlOptions[] = 'eauth';
2009-06-21 14:16:11 +00:00
$queryString = implode( '&', $urlOptions );
2009-06-21 14:16:11 +00:00
$url = SpecialPage::getTitleFor( 'Preferences' )->getFullURL( $queryString );
global $wgOut;
$wgOut->redirect( $url );
}
2009-06-21 14:16:11 +00:00
return true;
}
2009-06-21 14:16:11 +00:00
public static function loadOldSearchNs( $user ) {
$searchableNamespaces = SearchEngine::searchableNamespaces();
// Back compat with old format
$arr = array();
2009-06-21 14:16:11 +00:00
foreach( $searchableNamespaces as $ns => $name ) {
if( $user->getOption( 'searchNs' . $ns ) ) {
$arr[] = $ns;
}
}
2009-06-21 14:16:11 +00:00
return $arr;
}
}
/** Some tweaks to allow js prefs to work */
class PreferencesForm extends HTMLForm {
function wrapForm( $html ) {
$html = Xml::tags( 'div', array( 'id' => 'preferences' ), $html );
2009-06-21 14:16:11 +00:00
return parent::wrapForm( $html );
}
2009-06-21 14:16:11 +00:00
function getButtons() {
$html = parent::getButtons();
2009-06-21 14:16:11 +00:00
global $wgUser;
2009-06-21 14:16:11 +00:00
$sk = $wgUser->getSkin();
$t = SpecialPage::getTitleFor( 'Preferences', 'reset' );
2009-06-21 14:16:11 +00:00
2009-05-22 09:35:48 +00:00
$html .= "\n" . $sk->link( $t, wfMsgHtml( 'restoreprefs' ) );
2009-06-21 14:16:11 +00:00
$html = Xml::tags( 'div', array( 'class' => 'mw-prefs-buttons' ), $html );
2009-06-21 14:16:11 +00:00
return $html;
}
2009-06-21 14:16:11 +00:00
function filterDataForSubmit( $data ) {
// Support for separating MultiSelect preferences into multiple preferences
// Due to lack of array support.
foreach( $this->mFlatFields as $fieldname => $field ) {
$info = $field->mParams;
2009-06-21 14:16:11 +00:00
if( $field instanceof HTMLMultiSelectField ) {
$options = HTMLFormField::flattenOptions( $info['options'] );
2009-06-21 14:16:11 +00:00
$prefix = isset( $info['prefix'] ) ? $info['prefix'] : $fieldname;
foreach( $options as $opt ) {
$data["$prefix$opt"] = in_array( $opt, $data[$fieldname] );
}
2009-06-21 14:16:11 +00:00
unset( $data[$fieldname] );
}
}
2009-06-21 14:16:11 +00:00
return $data;
}
}