mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-26 23:31:02 +00:00
1081 lines
37 KiB
PHP
1081 lines
37 KiB
PHP
<?php
|
||
/**
|
||
* VisualEditor extension hooks
|
||
*
|
||
* @file
|
||
* @ingroup Extensions
|
||
* @copyright 2011-2019 VisualEditor Team and others; see AUTHORS.txt
|
||
* @license MIT
|
||
*/
|
||
|
||
use MediaWiki\MediaWikiServices;
|
||
|
||
class VisualEditorHooks {
|
||
|
||
// Known parameters that VE does not handle
|
||
// TODO: Other params too?
|
||
// Known-good parameters: edit, veaction, section, oldid, lintid, preload, preloadparams, editintro
|
||
// Partially-good: preloadtitle (source-mode only)
|
||
private static $unsupportedEditParams = [
|
||
'undo',
|
||
'undoafter',
|
||
'veswitched'
|
||
];
|
||
|
||
/**
|
||
* Initialise the 'VisualEditorAvailableNamespaces' setting, and add content
|
||
* namespaces to it. This will run after LocalSettings.php is processed.
|
||
*/
|
||
public static function onRegistration() {
|
||
global $wgVisualEditorAvailableNamespaces, $wgContentNamespaces;
|
||
|
||
foreach ( $wgContentNamespaces as $contentNamespace ) {
|
||
if ( !isset( $wgVisualEditorAvailableNamespaces[$contentNamespace] ) ) {
|
||
$wgVisualEditorAvailableNamespaces[$contentNamespace] = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Factory to return the relevant API class
|
||
*
|
||
* @param ApiMain $main The ApiMain instance
|
||
* @param string $name The api request name
|
||
* @return ApiVisualEditor|ApiVisualEditorEdit API class
|
||
*/
|
||
public static function getVisualEditorApiFactory( $main, $name ) {
|
||
$config = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
$class = $name === 'visualeditor' ? 'ApiVisualEditor' : 'ApiVisualEditorEdit';
|
||
return new $class( $main, $name, $config );
|
||
}
|
||
|
||
/**
|
||
* Adds VisualEditor JS to the output.
|
||
*
|
||
* This is attached to the MediaWiki 'BeforePageDisplay' hook.
|
||
*
|
||
* @param OutputPage &$output The page view.
|
||
* @param Skin &$skin The skin that's going to build the UI.
|
||
*/
|
||
public static function onBeforePageDisplay( OutputPage &$output, Skin &$skin ) {
|
||
$output->addModules( [
|
||
'ext.visualEditor.desktopArticleTarget.init',
|
||
'ext.visualEditor.targetLoader'
|
||
] );
|
||
$output->addModuleStyles( [ 'ext.visualEditor.desktopArticleTarget.noscript' ] );
|
||
// add scroll offset js variable to output
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
$skinsToolbarScrollOffset = $veConfig->get( 'VisualEditorSkinToolbarScrollOffset' );
|
||
$toolbarScrollOffset = 0;
|
||
$skinName = $skin->getSkinName();
|
||
if ( isset( $skinsToolbarScrollOffset[$skinName] ) ) {
|
||
$toolbarScrollOffset = $skinsToolbarScrollOffset[$skinName];
|
||
}
|
||
$output->addJsConfigVars( 'wgVisualEditorToolbarScrollOffset', $toolbarScrollOffset );
|
||
$output->addJsConfigVars( 'wgVisualEditorUnsupportedEditParams', self::$unsupportedEditParams );
|
||
|
||
$output->addJsConfigVars(
|
||
'wgEditSubmitButtonLabelPublish',
|
||
$veConfig->get( 'EditSubmitButtonLabelPublish' )
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Handler for the DiffViewHeader hook, to add visual diffs code as configured
|
||
*
|
||
* @param DifferenceEngine $diff The difference engine
|
||
* @param Revision|null $oldRev The old revision
|
||
* @param Revision|null $newRev The new revision
|
||
* @return void
|
||
*/
|
||
public static function onDiffViewHeader(
|
||
DifferenceEngine $diff,
|
||
Revision $oldRev = null,
|
||
Revision $newRev = null
|
||
) {
|
||
$config = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
$output = RequestContext::getMain()->getOutput();
|
||
$user = RequestContext::getMain()->getUser();
|
||
|
||
if ( !(
|
||
// Enabled globally on wiki
|
||
$config->get( 'VisualEditorEnableDiffPage' ) ||
|
||
// Enabled as user beta feature
|
||
$user->getOption( 'visualeditor-visualdiffpage' ) ||
|
||
// Enabled by query param
|
||
$output->getRequest()->getVal( 'visualdiff' ) !== null
|
||
) ) {
|
||
return;
|
||
}
|
||
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
if ( !ApiVisualEditor::isAllowedContentType( $veConfig, $diff->getTitle()->getContentModel() ) ) {
|
||
return;
|
||
}
|
||
|
||
$output->addModuleStyles( [
|
||
'ext.visualEditor.diffPage.init.styles',
|
||
'oojs-ui.styles.icons-alerts',
|
||
'oojs-ui.styles.icons-editing-advanced'
|
||
] );
|
||
$output->addModules( 'ext.visualEditor.diffPage.init' );
|
||
$output->enableOOUI();
|
||
$output->addHtml(
|
||
'<div class="ve-init-mw-diffPage-diffMode">' .
|
||
// Will be replaced by a ButtonSelectWidget in JS
|
||
new OOUI\ButtonGroupWidget( [
|
||
'items' => [
|
||
new \OOUI\ButtonWidget( [
|
||
'data' => 'visual',
|
||
'icon' => 'eye',
|
||
'disabled' => true,
|
||
'label' => $output->msg( 'visualeditor-savedialog-review-visual' )->plain()
|
||
] ),
|
||
new \OOUI\ButtonWidget( [
|
||
'data' => 'source',
|
||
'icon' => 'wikiText',
|
||
'active' => true,
|
||
'label' => $output->msg( 'visualeditor-savedialog-review-wikitext' )->plain()
|
||
] )
|
||
]
|
||
] ) .
|
||
'</div>'
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Detect incompatibile browsers which we can't expect to load VE
|
||
*
|
||
* @param WebRequest $req The web request to check the details of
|
||
* @param Config $config VE config object
|
||
* @return bool True if the User Agent is blacklisted
|
||
*/
|
||
private static function isUABlacklisted( WebRequest $req, $config ) {
|
||
if ( $req->getVal( 'vewhitelist' ) ) {
|
||
return false;
|
||
}
|
||
$blacklist = $config->get( 'VisualEditorBrowserBlacklist' );
|
||
$ua = strtolower( $req->getHeader( 'User-Agent' ) );
|
||
foreach ( $blacklist as $uaSubstr => $rules ) {
|
||
if ( !strpos( $ua, $uaSubstr . '/' ) ) {
|
||
continue;
|
||
}
|
||
if ( !is_array( $rules ) ) {
|
||
return true;
|
||
}
|
||
|
||
$matches = [];
|
||
$ret = preg_match( '/' . $uaSubstr . '\/([0-9\.]*) ?/i', $ua, $matches );
|
||
if ( $ret !== 1 ) {
|
||
continue;
|
||
}
|
||
$version = $matches[1];
|
||
foreach ( $rules as $rule ) {
|
||
list( $op, $matchVersion ) = $rule;
|
||
if (
|
||
( $op === '<' && $version < $matchVersion ) ||
|
||
( $op === '>' && $version > $matchVersion ) ||
|
||
( $op === '<=' && $version <= $matchVersion ) ||
|
||
( $op === '>=' && $version >= $matchVersion )
|
||
) {
|
||
return true;
|
||
}
|
||
}
|
||
|
||
}
|
||
return false;
|
||
}
|
||
|
||
private static function isSupportedEditPage( Title $title, User $user, WebRequest $req ) {
|
||
if ( $req->getVal( 'action' ) !== 'edit' || !$title->quickUserCan( 'edit' ) ) {
|
||
return false;
|
||
}
|
||
|
||
foreach ( self::$unsupportedEditParams as $param ) {
|
||
if ( $req->getVal( $param ) !== null ) {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
if ( $req->getVal( 'wteswitched' ) ) {
|
||
return self::isVisualAvailable( $title );
|
||
}
|
||
|
||
switch ( self::getPreferredEditor( $user, $req ) ) {
|
||
case 'visualeditor':
|
||
return self::isVisualAvailable( $title ) || self::isWikitextAvailable( $title, $user );
|
||
case 'wikitext':
|
||
return self::isWikitextAvailable( $title, $user );
|
||
}
|
||
return false;
|
||
}
|
||
|
||
private static function isVisualAvailable( $title ) {
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
return ApiVisualEditor::isAllowedNamespace( $veConfig, $title->getNamespace() ) &&
|
||
ApiVisualEditor::isAllowedContentType( $veConfig, $title->getContentModel() );
|
||
}
|
||
|
||
private static function isWikitextAvailable( $title, $user ) {
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
return $user->getOption( 'visualeditor-newwikitext' ) &&
|
||
$title->getContentModel() === 'wikitext';
|
||
}
|
||
|
||
/**
|
||
* Decide whether to bother showing the wikitext editor at all.
|
||
* If not, we expect the VE initialisation JS to activate.
|
||
*
|
||
* @param Article $article The article being viewed.
|
||
* @param User $user The user-specific settings.
|
||
* @return bool Whether to show the wikitext editor or not.
|
||
*/
|
||
public static function onCustomEditor( Article $article, User $user ) {
|
||
$req = $article->getContext()->getRequest();
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
|
||
if (
|
||
!$user->getOption( 'visualeditor-enable' ) ||
|
||
$user->getOption( 'visualeditor-betatempdisable' ) ||
|
||
$user->getOption( 'visualeditor-autodisable' ) ||
|
||
( $veConfig->get( 'VisualEditorDisableForAnons' ) && $user->isAnon() ) ||
|
||
self::isUABlacklisted( $req, $veConfig )
|
||
) {
|
||
return true;
|
||
}
|
||
|
||
$title = $article->getTitle();
|
||
|
||
if ( $req->getVal( 'venoscript' ) ) {
|
||
$req->response()->setCookie( 'VEE', 'wikitext', 0, [ 'prefix' => '' ] );
|
||
$user->setOption( 'visualeditor-editor', 'wikitext' );
|
||
if ( !wfReadOnly() && !$user->isAnon() ) {
|
||
DeferredUpdates::addCallableUpdate( function () use ( $user ) {
|
||
$user->saveSettings();
|
||
} );
|
||
}
|
||
return true;
|
||
}
|
||
|
||
if ( self::isSupportedEditPage( $title, $user, $req ) ) {
|
||
$params = $req->getValues();
|
||
$params['venoscript'] = '1';
|
||
$url = wfScript() . '?' . wfArrayToCgi( $params );
|
||
$escapedUrl = htmlspecialchars( $url );
|
||
|
||
$out = $article->getContext()->getOutput();
|
||
$titleMsg = $title->exists() ? 'editing' : 'creating';
|
||
$out->setPageTitle( wfMessage( $titleMsg, $title->getPrefixedText() ) );
|
||
$out->addWikiMsg( 'visualeditor-toload', wfExpandUrl( $url ) );
|
||
|
||
// Redirect if the user has no JS (<noscript>)
|
||
$out->addHeadItem(
|
||
've-noscript-fallback',
|
||
"<noscript><meta http-equiv=\"refresh\" content=\"0; url=$escapedUrl\"></noscript>"
|
||
);
|
||
// Redirect if the user has no ResourceLoader
|
||
$out->addScript( Html::inlineScript(
|
||
"(window.NORLQ=window.NORLQ||[]).push(" .
|
||
"function(){" .
|
||
"location.href=\"$url\";" .
|
||
"}" .
|
||
");"
|
||
) );
|
||
$out->setRevisionId( $req->getInt( 'oldid', $article->getRevIdFetched() ) );
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
private static function getPreferredEditor( User $user, WebRequest $req ) {
|
||
$config = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
// On dual-edit-tab wikis, the edit page must mean the user wants wikitext
|
||
if ( !$config->get( 'VisualEditorUseSingleEditTab' ) ) {
|
||
return 'wikitext';
|
||
}
|
||
switch ( $user->getOption( 'visualeditor-tabs' ) ) {
|
||
case 'prefer-ve':
|
||
return 'visualeditor';
|
||
case 'prefer-wt':
|
||
return 'wikitext';
|
||
case 'remember-last':
|
||
return self::getLastEditor( $user, $req );
|
||
case 'multi-tab':
|
||
// May have got here by switching from VE
|
||
// TODO: Make such an action explicitly request wikitext
|
||
// so we can use getLastEditor here instead.
|
||
return 'wikitext';
|
||
}
|
||
return null;
|
||
}
|
||
|
||
private static function getLastEditor( User $user, WebRequest $req ) {
|
||
// This logic matches getLastEditor in:
|
||
// modules/ve-mw/init/targets/ve.init.mw.DesktopArticleTarget.init.js
|
||
$editor = $req->getCookie( 'VEE', '' );
|
||
// Set editor to user's preference or site's default if …
|
||
if (
|
||
// … user is logged in,
|
||
!$user->isAnon() ||
|
||
// … no cookie is set, or
|
||
!$editor ||
|
||
// value is invalid.
|
||
!( $editor === 'visualeditor' || $editor === 'wikitext' )
|
||
) {
|
||
$editor = $user->getOption( 'visualeditor-editor' );
|
||
}
|
||
return $editor;
|
||
}
|
||
|
||
/**
|
||
* Changes the Edit tab and adds the VisualEditor tab.
|
||
*
|
||
* This is attached to the MediaWiki 'SkinTemplateNavigation' hook.
|
||
*
|
||
* @param SkinTemplate &$skin The skin template on which the UI is built.
|
||
* @param array &$links Navigation links.
|
||
*/
|
||
public static function onSkinTemplateNavigation( SkinTemplate &$skin, array &$links ) {
|
||
$config = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
|
||
// Exit if there's no edit link for whatever reason (e.g. protected page)
|
||
if ( !isset( $links['views']['edit'] ) ) {
|
||
return;
|
||
}
|
||
|
||
$user = $skin->getUser();
|
||
if (
|
||
$config->get( 'VisualEditorUseSingleEditTab' ) &&
|
||
$user->getOption( 'visualeditor-tabs' ) === 'prefer-wt'
|
||
) {
|
||
return;
|
||
}
|
||
|
||
if (
|
||
$config->get( 'VisualEditorUseSingleEditTab' ) &&
|
||
!$user->isAnon() &&
|
||
!$user->getOption( 'visualeditor-autodisable' ) &&
|
||
!$user->getOption( 'visualeditor-betatempdisable' ) &&
|
||
!$user->getOption( 'visualeditor-hidetabdialog' ) &&
|
||
$user->getOption( 'visualeditor-tabs' ) === 'remember-last'
|
||
) {
|
||
$dbr = wfGetDB( DB_REPLICA );
|
||
$revWhere = ActorMigration::newMigration()->getWhere( $dbr, 'rev_user', $user );
|
||
foreach ( $revWhere['orconds'] as $key => $cond ) {
|
||
$tsField = $key === 'actor' ? 'revactor_timestamp' : 'rev_timestamp';
|
||
if (
|
||
$dbr->select(
|
||
[ 'revision' ] + $revWhere['tables'],
|
||
'1',
|
||
[
|
||
$cond,
|
||
$tsField . ' < ' . $dbr->addQuotes(
|
||
$config->get( 'VisualEditorSingleEditTabSwitchTime' )
|
||
)
|
||
],
|
||
__METHOD__,
|
||
[ 'LIMIT' => 1 ],
|
||
$revWhere['joins']
|
||
)->numRows() === 1
|
||
) {
|
||
$links['views']['edit']['class'] .= ' visualeditor-showtabdialog';
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Exit if the user doesn't have VE enabled
|
||
if (
|
||
!$user->getOption( 'visualeditor-enable' ) ||
|
||
$user->getOption( 'visualeditor-betatempdisable' ) ||
|
||
$user->getOption( 'visualeditor-autodisable' ) ||
|
||
( $config->get( 'VisualEditorDisableForAnons' ) && $user->isAnon() )
|
||
) {
|
||
return;
|
||
}
|
||
|
||
$title = $skin->getRelevantTitle();
|
||
$namespaceEnabled = ApiVisualEditor::isAllowedNamespace( $config, $title->getNamespace() );
|
||
$pageContentModel = $title->getContentModel();
|
||
$contentModelEnabled = ApiVisualEditor::isAllowedContentType( $config, $pageContentModel );
|
||
// Don't exit if this page isn't VE-enabled, since we should still
|
||
// change "Edit" to "Edit source".
|
||
$isAvailable = $namespaceEnabled && $contentModelEnabled;
|
||
|
||
$tabMessages = $config->get( 'VisualEditorTabMessages' );
|
||
// Rebuild the $links['views'] array and inject the VisualEditor tab before or after
|
||
// the edit tab as appropriate. We have to rebuild the array because PHP doesn't allow
|
||
// us to splice into the middle of an associative array.
|
||
$newViews = [];
|
||
foreach ( $links['views'] as $action => $data ) {
|
||
if ( $action === 'edit' ) {
|
||
// Build the VisualEditor tab
|
||
$existing = $title->exists() || (
|
||
$title->inNamespace( NS_MEDIAWIKI ) &&
|
||
$title->getDefaultMessageText() !== false
|
||
);
|
||
$action = $existing ? 'edit' : 'create';
|
||
$veParams = $skin->editUrlOptions();
|
||
// Remove action=edit
|
||
unset( $veParams['action'] );
|
||
// Set veaction=edit
|
||
$veParams['veaction'] = 'edit';
|
||
$veTabMessage = $tabMessages[$action];
|
||
$veTabText = $veTabMessage === null ? $data['text'] :
|
||
$skin->msg( $veTabMessage )->text();
|
||
$veTab = [
|
||
'href' => $title->getLocalURL( $veParams ),
|
||
'text' => $veTabText,
|
||
'primary' => true,
|
||
'class' => '',
|
||
];
|
||
|
||
// Alter the edit tab
|
||
$editTab = $data;
|
||
if (
|
||
$title->inNamespace( NS_FILE ) &&
|
||
WikiPage::factory( $title ) instanceof WikiFilePage &&
|
||
!WikiPage::factory( $title )->isLocal()
|
||
) {
|
||
$editTabMessage = $tabMessages[$action . 'localdescriptionsource'];
|
||
} else {
|
||
$editTabMessage = $tabMessages[$action . 'source'];
|
||
}
|
||
|
||
if ( $editTabMessage !== null ) {
|
||
$editTab['text'] = $skin->msg( $editTabMessage )->text();
|
||
}
|
||
|
||
$editor = self::getLastEditor( $user, $skin->getRequest() );
|
||
if (
|
||
$isAvailable &&
|
||
$config->get( 'VisualEditorUseSingleEditTab' ) &&
|
||
(
|
||
$user->getOption( 'visualeditor-tabs' ) === 'prefer-ve' ||
|
||
(
|
||
$user->getOption( 'visualeditor-tabs' ) === 'remember-last' &&
|
||
$editor === 'visualeditor'
|
||
)
|
||
)
|
||
) {
|
||
$editTab['text'] = $veTabText;
|
||
$newViews['edit'] = $editTab;
|
||
} elseif (
|
||
$isAvailable &&
|
||
(
|
||
!$config->get( 'VisualEditorUseSingleEditTab' ) ||
|
||
$user->getOption( 'visualeditor-tabs' ) === 'multi-tab'
|
||
)
|
||
) {
|
||
// Inject the VE tab before or after the edit tab
|
||
if ( $config->get( 'VisualEditorTabPosition' ) === 'before' ) {
|
||
$editTab['class'] .= ' collapsible';
|
||
$newViews['ve-edit'] = $veTab;
|
||
$newViews['edit'] = $editTab;
|
||
} else {
|
||
$veTab['class'] .= ' collapsible';
|
||
$newViews['edit'] = $editTab;
|
||
$newViews['ve-edit'] = $veTab;
|
||
}
|
||
} elseif (
|
||
!$config->get( 'VisualEditorUseSingleEditTab' ) ||
|
||
!$isAvailable ||
|
||
$user->getOption( 'visualeditor-tabs' ) === 'multi-tab' ||
|
||
(
|
||
$user->getOption( 'visualeditor-tabs' ) === 'remember-last' &&
|
||
$editor === 'wikitext'
|
||
)
|
||
) {
|
||
// Don't add ve-edit, but do update the edit tab (e.g. "Edit source").
|
||
$newViews['edit'] = $editTab;
|
||
} else {
|
||
// This should not happen.
|
||
}
|
||
} else {
|
||
// Just pass through
|
||
$newViews[$action] = $data;
|
||
}
|
||
}
|
||
$links['views'] = $newViews;
|
||
}
|
||
|
||
/**
|
||
* Called when the normal wikitext editor is shown.
|
||
* Inserts a 'veswitched' hidden field if requested by the client
|
||
*
|
||
* @param EditPage $editPage The edit page view.
|
||
* @param OutputPage $output The page view.
|
||
*/
|
||
public static function onEditPageShowEditFormFields( EditPage $editPage, OutputPage $output ) {
|
||
$request = $output->getRequest();
|
||
if ( $request->getBool( 'veswitched' ) ) {
|
||
$output->addHTML( Html::hidden( 'veswitched', '1' ) );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Called when an edit is saved
|
||
* Adds 'visualeditor-switched' tag to the edit if requested
|
||
*
|
||
* @param RecentChange $rc The new RC entry.
|
||
*/
|
||
public static function onRecentChangeSave( RecentChange $rc ) {
|
||
$request = RequestContext::getMain()->getRequest();
|
||
if ( $request->getBool( 'veswitched' ) && $rc->getAttribute( 'rc_this_oldid' ) ) {
|
||
$rc->addTags( 'visualeditor-switched' );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Changes the section edit links to add a VE edit link.
|
||
*
|
||
* This is attached to the MediaWiki 'SkinEditSectionLinks' hook.
|
||
*
|
||
* @param Skin $skin Skin being used to render the UI
|
||
* @param Title $title Title being used for request
|
||
* @param string $section The name of the section being pointed to.
|
||
* @param string $tooltip The default tooltip.
|
||
* @param array &$result All link detail arrays.
|
||
* @param Language $lang The user interface language.
|
||
*/
|
||
public static function onSkinEditSectionLinks( Skin $skin, Title $title, $section,
|
||
$tooltip, &$result, $lang
|
||
) {
|
||
$config = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
|
||
// Exit if we're in parserTests
|
||
if ( isset( $GLOBALS[ 'wgVisualEditorInParserTests' ] ) ) {
|
||
return;
|
||
}
|
||
|
||
$user = $skin->getUser();
|
||
// Exit if the user doesn't have VE enabled
|
||
if (
|
||
!$user->getOption( 'visualeditor-enable' ) ||
|
||
$user->getOption( 'visualeditor-betatempdisable' ) ||
|
||
$user->getOption( 'visualeditor-autodisable' ) ||
|
||
( $config->get( 'VisualEditorDisableForAnons' ) && $user->isAnon() )
|
||
) {
|
||
return;
|
||
}
|
||
|
||
// Exit if we're on a foreign file description page
|
||
if (
|
||
$title->inNamespace( NS_FILE ) &&
|
||
WikiPage::factory( $title ) instanceof WikiFilePage &&
|
||
!WikiPage::factory( $title )->isLocal()
|
||
) {
|
||
return;
|
||
}
|
||
|
||
$editor = self::getLastEditor( $user, RequestContext::getMain()->getRequest() );
|
||
if (
|
||
!$config->get( 'VisualEditorUseSingleEditTab' ) ||
|
||
$user->getOption( 'visualeditor-tabs' ) === 'multi-tab' ||
|
||
(
|
||
$user->getOption( 'visualeditor-tabs' ) === 'remember-last' &&
|
||
$editor === 'wikitext'
|
||
)
|
||
) {
|
||
// Don't add ve-edit, but do update the edit tab (e.g. "Edit source").
|
||
$tabMessages = $config->get( 'VisualEditorTabMessages' );
|
||
$sourceEditSection = $tabMessages['editsectionsource'];
|
||
$result['editsection']['text'] = $skin->msg( $sourceEditSection )->inLanguage( $lang )->text();
|
||
}
|
||
|
||
// Exit if we're using the single edit tab.
|
||
if (
|
||
$config->get( 'VisualEditorUseSingleEditTab' ) &&
|
||
$user->getOption( 'visualeditor-tabs' ) !== 'multi-tab'
|
||
) {
|
||
return;
|
||
}
|
||
|
||
// add VE edit section in VE available namespaces
|
||
if ( ApiVisualEditor::isAllowedNamespace( $config, $title->getNamespace() ) ) {
|
||
$veEditSection = $tabMessages['editsection'];
|
||
$veLink = [
|
||
'text' => $skin->msg( $veEditSection )->inLanguage( $lang )->text(),
|
||
'targetTitle' => $title,
|
||
'attribs' => $result['editsection']['attribs'] + [
|
||
'class' => 'mw-editsection-visualeditor'
|
||
],
|
||
'query' => [ 'veaction' => 'edit', 'section' => $section ],
|
||
'options' => [ 'noclasses', 'known' ]
|
||
];
|
||
|
||
$result['veeditsection'] = $veLink;
|
||
if ( $config->get( 'VisualEditorTabPosition' ) === 'before' ) {
|
||
krsort( $result );
|
||
// TODO: This will probably cause weird ordering if any other extensions added something
|
||
// already.
|
||
// ... wfArrayInsertBefore?
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Convert a namespace index to the local text for display to the user.
|
||
*
|
||
* @param int $nsIndex
|
||
* @return string
|
||
*/
|
||
private static function convertNs( $nsIndex ) {
|
||
global $wgLang;
|
||
if ( $nsIndex ) {
|
||
return $wgLang->convertNamespace( $nsIndex );
|
||
} else {
|
||
return wfMessage( 'blanknamespace' )->text();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Handler for the GetPreferences hook, to add and hide user preferences as configured
|
||
*
|
||
* @param User $user The user object
|
||
* @param array &$preferences Their preferences object
|
||
*/
|
||
public static function onGetPreferences( User $user, array &$preferences ) {
|
||
global $wgLang;
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
|
||
if ( !class_exists( 'BetaFeatures' ) ) {
|
||
// Config option for visual editing "alpha" state (no Beta Feature)
|
||
$namespaces = ApiVisualEditor::getAvailableNamespaceIds( $veConfig );
|
||
|
||
$visualEnablePreference = [
|
||
'type' => 'toggle',
|
||
'label-message' => [
|
||
'visualeditor-preference-enable',
|
||
$wgLang->commaList( array_map(
|
||
[ 'self', 'convertNs' ],
|
||
$namespaces
|
||
) ),
|
||
count( $namespaces )
|
||
],
|
||
'section' => 'editing/editor'
|
||
];
|
||
if ( $user->getOption( 'visualeditor-autodisable' ) ) {
|
||
$visualEnablePreference['default'] = false;
|
||
}
|
||
$preferences['visualeditor-enable'] = $visualEnablePreference;
|
||
}
|
||
|
||
// Config option for visual editing "deployed" state (opt-out)
|
||
$preferences['visualeditor-betatempdisable'] = [
|
||
'type' => 'toggle',
|
||
'label-message' => 'visualeditor-preference-betatempdisable',
|
||
'section' => 'editing/editor',
|
||
'default' => $user->getOption( 'visualeditor-betatempdisable' ) ||
|
||
$user->getOption( 'visualeditor-autodisable' )
|
||
];
|
||
|
||
// Config option for wikitext editing "deployed" state (opt-out)
|
||
if (
|
||
$veConfig->get( 'VisualEditorEnableWikitext' )
|
||
) {
|
||
$preferences['visualeditor-newwikitext'] = [
|
||
'type' => 'toggle',
|
||
'label-message' => 'visualeditor-preference-newwikitexteditor-enable',
|
||
'section' => 'editing/editor'
|
||
];
|
||
}
|
||
|
||
// Config option for Single Edit Tab
|
||
if (
|
||
$veConfig->get( 'VisualEditorUseSingleEditTab' ) &&
|
||
!$user->getOption( 'visualeditor-autodisable' ) &&
|
||
!$user->getOption( 'visualeditor-betatempdisable' )
|
||
) {
|
||
$preferences['visualeditor-tabs'] = [
|
||
'type' => 'select',
|
||
'label-message' => 'visualeditor-preference-tabs',
|
||
'section' => 'editing/editor',
|
||
'options' => [
|
||
wfMessage( 'visualeditor-preference-tabs-remember-last' )->escaped() => 'remember-last',
|
||
wfMessage( 'visualeditor-preference-tabs-prefer-ve' )->escaped() => 'prefer-ve',
|
||
wfMessage( 'visualeditor-preference-tabs-prefer-wt' )->escaped() => 'prefer-wt',
|
||
wfMessage( 'visualeditor-preference-tabs-multi-tab' )->escaped() => 'multi-tab'
|
||
]
|
||
];
|
||
}
|
||
|
||
$api = [ 'type' => 'api' ];
|
||
$preferences['visualeditor-autodisable'] = $api;
|
||
// The diff mode is persisted for each editor mode separately,
|
||
// e.g. use visual diffs for visual mode only.
|
||
$preferences['visualeditor-diffmode-source'] = $api;
|
||
$preferences['visualeditor-diffmode-visual'] = $api;
|
||
$preferences['visualeditor-diffmode-historical'] = $api;
|
||
$preferences['visualeditor-editor'] = $api;
|
||
$preferences['visualeditor-hidebetawelcome'] = $api;
|
||
$preferences['visualeditor-hidetabdialog'] = $api;
|
||
$preferences['visualeditor-hidesourceswitchpopup'] = $api;
|
||
$preferences['visualeditor-hidevisualswitchpopup'] = $api;
|
||
$preferences['visualeditor-hideusered'] = $api;
|
||
$preferences['visualeditor-findAndReplace-diacritic'] = $api;
|
||
$preferences['visualeditor-findAndReplace-findText'] = $api;
|
||
$preferences['visualeditor-findAndReplace-replaceText'] = $api;
|
||
$preferences['visualeditor-findAndReplace-regex'] = $api;
|
||
$preferences['visualeditor-findAndReplace-matchCase'] = $api;
|
||
$preferences['visualeditor-findAndReplace-word'] = $api;
|
||
}
|
||
|
||
/**
|
||
* Handler for the GetBetaPreferences hook, to add and hide user beta preferences as configured
|
||
*
|
||
* @param User $user The user object
|
||
* @param array &$preferences Their preferences object
|
||
*/
|
||
public static function onGetBetaPreferences( User $user, array &$preferences ) {
|
||
$coreConfig = RequestContext::getMain()->getConfig();
|
||
$iconpath = $coreConfig->get( 'ExtensionAssetsPath' ) . "/VisualEditor/images";
|
||
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
$preferences['visualeditor-enable'] = [
|
||
'version' => '1.0',
|
||
'label-message' => 'visualeditor-preference-core-label',
|
||
'desc-message' => 'visualeditor-preference-core-description',
|
||
'screenshot' => [
|
||
'ltr' => "$iconpath/betafeatures-icon-VisualEditor-ltr.svg",
|
||
'rtl' => "$iconpath/betafeatures-icon-VisualEditor-rtl.svg",
|
||
],
|
||
'info-message' => 'visualeditor-preference-core-info-link',
|
||
'discussion-message' => 'visualeditor-preference-core-discussion-link',
|
||
'requirements' => [
|
||
'javascript' => true,
|
||
'blacklist' => $veConfig->get( 'VisualEditorBrowserBlacklist' ),
|
||
]
|
||
];
|
||
|
||
if (
|
||
$veConfig->get( 'VisualEditorEnableWikitextBetaFeature' ) &&
|
||
// Don't try to register as a beta feature if enabled by default
|
||
!$veConfig->get( 'VisualEditorEnableWikitext' )
|
||
) {
|
||
$preferences['visualeditor-newwikitext'] = [
|
||
'version' => '1.0',
|
||
'label-message' => 'visualeditor-preference-newwikitexteditor-label',
|
||
'desc-message' => 'visualeditor-preference-newwikitexteditor-description',
|
||
'screenshot' => [
|
||
'ltr' => "$iconpath/betafeatures-icon-WikitextEditor-ltr.svg",
|
||
'rtl' => "$iconpath/betafeatures-icon-WikitextEditor-rtl.svg",
|
||
],
|
||
'info-message' => 'visualeditor-preference-newwikitexteditor-info-link',
|
||
'discussion-message' => 'visualeditor-preference-newwikitexteditor-discussion-link',
|
||
'requirements' => [
|
||
'javascript' => true,
|
||
'blacklist' => $veConfig->get( 'VisualEditorBrowserBlacklist' ),
|
||
]
|
||
];
|
||
}
|
||
|
||
if (
|
||
$veConfig->get( 'VisualEditorEnableDiffPageBetaFeature' ) &&
|
||
// Don't try to register as a beta feature if enabled by default
|
||
!$veConfig->get( 'VisualEditorEnableDiffPage' )
|
||
) {
|
||
$preferences['visualeditor-visualdiffpage'] = [
|
||
'version' => '1.0',
|
||
'label-message' => 'visualeditor-preference-visualdiffpage-label',
|
||
'desc-message' => 'visualeditor-preference-visualdiffpage-description',
|
||
'screenshot' => [
|
||
'ltr' => "$iconpath/betafeatures-icon-VisualDiffPage-ltr.svg",
|
||
'rtl' => "$iconpath/betafeatures-icon-VisualDiffPage-rtl.svg",
|
||
],
|
||
'info-message' => 'visualeditor-preference-visualdiffpage-info-link',
|
||
'discussion-message' => 'visualeditor-preference-visualdiffpage-discussion-link',
|
||
'requirements' => [
|
||
'javascript' => true,
|
||
'blacklist' => $veConfig->get( 'VisualEditorBrowserBlacklist' ),
|
||
]
|
||
];
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Implements the PreferencesFormPreSave hook, to remove the 'autodisable' flag
|
||
* when the user it was set on explicitly enables VE.
|
||
*
|
||
* @param array $data User-submitted data
|
||
* @param PreferencesForm $form A ContextSource
|
||
* @param User $user User with new preferences already set
|
||
* @param bool &$result Success or failure
|
||
*/
|
||
public static function onPreferencesFormPreSave( $data, $form, $user, &$result ) {
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
// On a wiki where enable is hidden and set to 1, if user sets betatempdisable=0
|
||
// then set autodisable=0
|
||
// On a wiki where betatempdisable is hidden and set to 0, if user sets enable=1
|
||
// then set autodisable=0
|
||
if (
|
||
$user->getOption( 'visualeditor-autodisable' ) &&
|
||
$user->getOption( 'visualeditor-enable' ) &&
|
||
!$user->getOption( 'visualeditor-betatempdisable' )
|
||
) {
|
||
$user->setOption( 'visualeditor-autodisable', false );
|
||
} elseif (
|
||
// On a wiki where betatempdisable is hidden and set to 0, if user sets enable=0,
|
||
// then set autodisable=1
|
||
$veConfig->get( 'VisualEditorTransitionDefault' ) &&
|
||
!$user->getOption( 'visualeditor-betatempdisable' ) &&
|
||
!$user->getOption( 'visualeditor-enable' ) &&
|
||
!$user->getOption( 'visualeditor-autodisable' )
|
||
) {
|
||
$user->setOption( 'visualeditor-autodisable', true );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Implements the ListDefinedTags and ChangeTagsListActive hooks, to populate
|
||
* core Special:Tags with the change tags in use by VisualEditor.
|
||
*
|
||
* @param array &$tags Available change tags.
|
||
*/
|
||
public static function onListDefinedTags( &$tags ) {
|
||
$tags[] = 'visualeditor';
|
||
// No longer in active use:
|
||
$tags[] = 'visualeditor-needcheck';
|
||
$tags[] = 'visualeditor-switched';
|
||
$tags[] = 'visualeditor-wikitext';
|
||
}
|
||
|
||
/**
|
||
* Adds extra variables to the page config.
|
||
*
|
||
* @param array &$vars Global variables object
|
||
* @param OutputPage $out The page view.
|
||
*/
|
||
public static function onMakeGlobalVariablesScript( array &$vars, OutputPage $out ) {
|
||
$pageLanguage = ApiVisualEditor::getPageLanguage( $out->getTitle() );
|
||
|
||
$fallbacks = $pageLanguage->getConverter()->getVariantFallbacks(
|
||
$pageLanguage->getPreferredVariant()
|
||
);
|
||
|
||
$vars['wgVisualEditor'] = [
|
||
'pageLanguageCode' => $pageLanguage->getHtmlCode(),
|
||
'pageLanguageDir' => $pageLanguage->getDir(),
|
||
'pageVariantFallbacks' => $fallbacks,
|
||
'usePageImages' => ExtensionRegistry::getInstance()->isLoaded( 'PageImages' ),
|
||
'usePageDescriptions' => defined( 'WBC_VERSION' ),
|
||
];
|
||
}
|
||
|
||
/**
|
||
* Adds extra variables to the global config
|
||
*
|
||
* @param array &$vars Global variables object
|
||
*/
|
||
public static function onResourceLoaderGetConfigVars( array &$vars ) {
|
||
$coreConfig = RequestContext::getMain()->getConfig();
|
||
$defaultUserOptions = $coreConfig->get( 'DefaultUserOptions' );
|
||
$thumbLimits = $coreConfig->get( 'ThumbLimits' );
|
||
$veConfig = ConfigFactory::getDefaultInstance()->makeConfig( 'visualeditor' );
|
||
$availableNamespaces = ApiVisualEditor::getAvailableNamespaceIds( $veConfig );
|
||
$availableContentModels = array_filter(
|
||
array_merge(
|
||
ExtensionRegistry::getInstance()->getAttribute( 'VisualEditorAvailableContentModels' ),
|
||
$veConfig->get( 'VisualEditorAvailableContentModels' )
|
||
)
|
||
);
|
||
|
||
$vars['wgVisualEditorConfig'] = [
|
||
'disableForAnons' => $veConfig->get( 'VisualEditorDisableForAnons' ),
|
||
'preloadModules' => $veConfig->get( 'VisualEditorPreloadModules' ),
|
||
'preferenceModules' => $veConfig->get( 'VisualEditorPreferenceModules' ),
|
||
'namespaces' => $availableNamespaces,
|
||
'contentModels' => $availableContentModels,
|
||
'pluginModules' => array_merge(
|
||
ExtensionRegistry::getInstance()->getAttribute( 'VisualEditorPluginModules' ),
|
||
// @todo deprecate the global setting
|
||
$veConfig->get( 'VisualEditorPluginModules' )
|
||
),
|
||
'defaultUserOptions' => [
|
||
'defaultthumbsize' => $thumbLimits[ $defaultUserOptions['thumbsize'] ]
|
||
],
|
||
'galleryOptions' => $coreConfig->get( 'GalleryOptions' ),
|
||
'blacklist' => $veConfig->get( 'VisualEditorBrowserBlacklist' ),
|
||
'tabPosition' => $veConfig->get( 'VisualEditorTabPosition' ),
|
||
'tabMessages' => $veConfig->get( 'VisualEditorTabMessages' ),
|
||
'singleEditTab' => $veConfig->get( 'VisualEditorUseSingleEditTab' ),
|
||
'enableVisualSectionEditing' => $veConfig->get( 'VisualEditorEnableVisualSectionEditing' ),
|
||
'showBetaWelcome' => $veConfig->get( 'VisualEditorShowBetaWelcome' ),
|
||
'enableTocWidget' => $veConfig->get( 'VisualEditorEnableTocWidget' ),
|
||
'enableWikitext' => (
|
||
$veConfig->get( 'VisualEditorEnableWikitext' ) ||
|
||
$veConfig->get( 'VisualEditorEnableWikitextBetaFeature' )
|
||
),
|
||
'svgMaxSize' => $coreConfig->get( 'SVGMaxSize' ),
|
||
'namespacesWithSubpages' => $coreConfig->get( 'NamespacesWithSubpages' ),
|
||
'specialBooksources' => urldecode( SpecialPage::getTitleFor( 'Booksources' )->getPrefixedURL() ),
|
||
'rebaserUrl' => $coreConfig->get( 'VisualEditorRebaserURL' ),
|
||
'restbaseUrl' => $coreConfig->get( 'VisualEditorRestbaseURL' ),
|
||
'fullRestbaseUrl' => $coreConfig->get( 'VisualEditorFullRestbaseURL' ),
|
||
'feedbackApiUrl' => $veConfig->get( 'VisualEditorFeedbackAPIURL' ),
|
||
'feedbackTitle' => $veConfig->get( 'VisualEditorFeedbackTitle' ),
|
||
'sourceFeedbackTitle' => $veConfig->get( 'VisualEditorSourceFeedbackTitle' ),
|
||
'enableBlockNoticeStats' => $coreConfig->get( 'EnableBlockNoticeStats' ),
|
||
];
|
||
}
|
||
|
||
/**
|
||
* Conditionally register the jquery.uls.data and jquery.i18n modules, in case they've already
|
||
* been registered by the UniversalLanguageSelector extension or the TemplateData extension.
|
||
*
|
||
* @param ResourceLoader &$resourceLoader Client-side code and assets to be loaded.
|
||
*/
|
||
public static function onResourceLoaderRegisterModules( ResourceLoader &$resourceLoader ) {
|
||
$resourceModules = $resourceLoader->getConfig()->get( 'ResourceModules' );
|
||
|
||
$veResourceTemplate = [
|
||
'localBasePath' => dirname( __DIR__ ),
|
||
'remoteExtPath' => 'VisualEditor',
|
||
];
|
||
|
||
// Only pull in VisualEditor core's local version of jquery.uls.data if it hasn't been
|
||
// installed locally already (presumably, by the UniversalLanguageSelector extension).
|
||
if (
|
||
!isset( $resourceModules[ 'jquery.uls.data' ] ) &&
|
||
!$resourceLoader->isModuleRegistered( 'jquery.uls.data' )
|
||
) {
|
||
$resourceLoader->register( [
|
||
'jquery.uls.data' => $veResourceTemplate + [
|
||
'scripts' => [
|
||
'lib/ve/lib/jquery.uls/src/jquery.uls.data.js',
|
||
'lib/ve/lib/jquery.uls/src/jquery.uls.data.utils.js',
|
||
],
|
||
'targets' => [ 'desktop', 'mobile' ],
|
||
] ] );
|
||
}
|
||
|
||
$extensionMessages = [];
|
||
if ( class_exists( 'ConfirmEditHooks' ) ) {
|
||
$extensionMessages[] = 'captcha-edit';
|
||
$extensionMessages[] = 'captcha-label';
|
||
|
||
if ( class_exists( 'QuestyCaptcha' ) ) {
|
||
$extensionMessages[] = 'questycaptcha-edit';
|
||
}
|
||
|
||
if ( class_exists( 'FancyCaptcha' ) ) {
|
||
$extensionMessages[] = 'fancycaptcha-edit';
|
||
$extensionMessages[] = 'fancycaptcha-reload-text';
|
||
}
|
||
}
|
||
$resourceLoader->register( [
|
||
'ext.visualEditor.mwextensionmessages' => $veResourceTemplate + [
|
||
'messages' => $extensionMessages,
|
||
'targets' => [ 'desktop', 'mobile' ],
|
||
]
|
||
] );
|
||
}
|
||
|
||
/**
|
||
* Ensures that we know whether we're running inside a parser test.
|
||
*
|
||
* @param array &$settings The settings with which MediaWiki is being run.
|
||
*/
|
||
public static function onParserTestGlobals( array &$settings ) {
|
||
$settings['wgVisualEditorInParserTests'] = true;
|
||
}
|
||
|
||
/**
|
||
* @param array &$redirectParams Parameters preserved on special page redirects
|
||
* to wiki pages
|
||
*/
|
||
public static function onRedirectSpecialArticleRedirectParams( &$redirectParams ) {
|
||
array_push( $redirectParams, 'veaction' );
|
||
}
|
||
|
||
/**
|
||
* If the user has specified that they want to edit the page with VE, suppress any redirect.
|
||
*
|
||
* @param Title $title Title being used for request
|
||
* @param Article|null $article The page being viewed.
|
||
* @param OutputPage $output The page view.
|
||
* @param User $user The user-specific settings.
|
||
* @param WebRequest $request The request.
|
||
* @param MediaWiki $mediaWiki Helper class.
|
||
*/
|
||
public static function onBeforeInitialize(
|
||
Title $title, $article, OutputPage $output,
|
||
User $user, WebRequest $request, MediaWiki $mediaWiki
|
||
) {
|
||
if ( $request->getVal( 'veaction' ) ) {
|
||
$request->setVal( 'redirect', 'no' );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Set user preferences for new and auto-created accounts if so configured.
|
||
*
|
||
* Sets user preference to enable the VisualEditor account for new auto-
|
||
* created ('auth') accounts, if $wgVisualEditorAutoAccountEnable is set.
|
||
*
|
||
* Sets user preference to enable the VisualEditor account for new non-auto-
|
||
* created accounts, if the account's userID matches, modulo the value of
|
||
* $wgVisualEditorNewAccountEnableProportion, if set. If set to '1', all new
|
||
* accounts would have VisualEditor enabled; at '2', 50% would; at '20',
|
||
* 5% would, and so on.
|
||
*
|
||
* To be removed once no longer needed.
|
||
*
|
||
* @param User $user The user-specific settings.
|
||
* @param bool $autocreated True if the user was auto-created (not a new global user).
|
||
*/
|
||
public static function onLocalUserCreated( $user, $autocreated ) {
|
||
$config = RequestContext::getMain()->getConfig();
|
||
$enableProportion = $config->get( 'VisualEditorNewAccountEnableProportion' );
|
||
|
||
if (
|
||
// Only act on actual accounts (avoid race condition bugs)
|
||
$user->isLoggedin() &&
|
||
// Only act if the default isn't already set
|
||
!User::getDefaultOption( 'visualeditor-enable' ) &&
|
||
// Act if either …
|
||
(
|
||
// … this is an auto-created account and we're configured so to do
|
||
(
|
||
$autocreated &&
|
||
$config->get( 'VisualEditorAutoAccountEnable' )
|
||
) ||
|
||
// … this is a real new account that matches the modulo and we're configured so to do
|
||
(
|
||
!$autocreated &&
|
||
$enableProportion &&
|
||
( ( $user->getId() % $enableProportion ) === 0 )
|
||
)
|
||
)
|
||
) {
|
||
$user->setOption( 'visualeditor-enable', 1 );
|
||
$user->saveSettings();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* On login, if user has a VEE cookie, set their preference equal to it.
|
||
*
|
||
* @param User $user The user-specific settings.
|
||
*/
|
||
public static function onUserLoggedIn( $user ) {
|
||
$cookie = RequestContext::getMain()->getRequest()->getCookie( 'VEE', '' );
|
||
if ( $cookie === 'visualeditor' || $cookie === 'wikitext' ) {
|
||
$lb = MediaWikiServices::getInstance()->getDBLoadBalancer();
|
||
DeferredUpdates::addUpdate( new AtomicSectionUpdate(
|
||
$lb->getLazyConnectionRef( DB_MASTER ),
|
||
__METHOD__,
|
||
function () use ( $user, $cookie ) {
|
||
if ( wfReadOnly() ) {
|
||
return;
|
||
}
|
||
|
||
$uLatest = $user->getInstanceForUpdate();
|
||
$uLatest->setOption( 'visualeditor-editor', $cookie );
|
||
$uLatest->saveSettings();
|
||
}
|
||
) );
|
||
}
|
||
}
|
||
}
|