mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/MultimediaViewer
synced 2024-12-13 08:38:55 +00:00
266 lines
7.8 KiB
JavaScript
266 lines
7.8 KiB
JavaScript
/*
|
|
* This file is part of the MediaWiki extension MediaViewer.
|
|
*
|
|
* MediaViewer is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* MediaViewer is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with MediaViewer. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
const { isMediaViewerEnabledOnClick } = require( 'mmv.head' );
|
|
( function () {
|
|
|
|
/**
|
|
* Contains/retrieves configuration/environment information for MediaViewer.
|
|
*/
|
|
class Config {
|
|
/**
|
|
* @param {Object} viewerConfig
|
|
* @param {Map} mwConfig
|
|
* @param {Object} mwUser
|
|
* @param {mw.Api} api
|
|
* @param {mw.SafeStorage} localStorage
|
|
*/
|
|
constructor( viewerConfig, mwConfig, mwUser, api, localStorage ) {
|
|
/**
|
|
* A plain object storing MediaViewer-specific settings
|
|
*
|
|
* @type {Object}
|
|
*/
|
|
this.viewerConfig = viewerConfig;
|
|
|
|
/**
|
|
* The mw.config object, for dependency injection
|
|
*
|
|
* @type {Map}
|
|
*/
|
|
this.mwConfig = mwConfig;
|
|
|
|
/**
|
|
* mw.user object, for dependency injection
|
|
*
|
|
* @type {Object}
|
|
*/
|
|
this.mwUser = mwUser;
|
|
|
|
/**
|
|
* API object, for dependency injection
|
|
*
|
|
* @type {mw.Api}
|
|
*/
|
|
this.api = api;
|
|
|
|
/**
|
|
* The localStorage object, for dependency injection
|
|
*
|
|
* @type {mw.SafeStorage}
|
|
*/
|
|
this.localStorage = localStorage;
|
|
}
|
|
|
|
/**
|
|
* Get value from local storage or fail gracefully.
|
|
*
|
|
* @param {string} key
|
|
* @param {any} [fallback] value to return when key is not set or localStorage is not supported
|
|
* @return {string|null} stored value or fallback or null if neither exists
|
|
*/
|
|
getFromLocalStorage( key, fallback ) {
|
|
const value = this.localStorage.get( key );
|
|
|
|
// localStorage will only store strings; if values `null`, `false` or
|
|
// `0` are set, they'll come out as `"null"`, `"false"` or `"0"`, so we
|
|
// can be certain that an actual null is a failure to locate the item,
|
|
// and false is an issue with localStorage itself
|
|
if ( value !== null && value !== false ) {
|
|
return value;
|
|
}
|
|
|
|
if ( value === null ) {
|
|
mw.log( `Failed to fetch item ${ key } from localStorage` );
|
|
}
|
|
|
|
return fallback !== undefined ? fallback : null;
|
|
}
|
|
|
|
/**
|
|
* Set item in local storage or fail gracefully.
|
|
*
|
|
* @param {string} key
|
|
* @param {any} value
|
|
* @return {boolean} whether storing the item was successful
|
|
*/
|
|
setInLocalStorage( key, value ) {
|
|
return this.localStorage.set( key, value );
|
|
}
|
|
|
|
/**
|
|
* Remove item from local storage or fail gracefully.
|
|
*
|
|
* @param {string} key
|
|
* @return {boolean} whether storing the item was successful
|
|
*/
|
|
removeFromLocalStorage( key ) {
|
|
this.localStorage.remove( key );
|
|
|
|
// mw.storage.remove catches all exceptions and returns false if any
|
|
// occur, so we can't distinguish between actual issues, and
|
|
// localStorage not being supported - however, localStorage.removeItem
|
|
// is not documented to throw any errors, so nothing to worry about;
|
|
// when localStorage is not supported, we'll consider removal successful
|
|
// (it can't have been there in the first place)
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Set user preference via AJAX
|
|
*
|
|
* @param {string} key
|
|
* @param {string} value
|
|
* @return {jQuery.Promise} a deferred which resolves/rejects on success/failure respectively
|
|
*/
|
|
setUserPreference( key, value ) {
|
|
return this.api.saveOption( key, value );
|
|
}
|
|
|
|
/**
|
|
* (Semi-)permanently stores the setting whether MediaViewer should handle thumbnail clicks.
|
|
* - for logged-in users, we use preferences
|
|
* - for anons, we use localStorage
|
|
* - for anons with old browsers, we don't do anything
|
|
*
|
|
* @param {boolean} enabled
|
|
* @return {jQuery.Promise} a deferred which resolves/rejects on success/failure respectively
|
|
*/
|
|
setMediaViewerEnabledOnClick( enabled ) {
|
|
const defaultPrefValue = this.mwConfig.get( 'wgMediaViewerEnabledByDefault' );
|
|
let deferred;
|
|
let newPrefValue;
|
|
let success = true;
|
|
|
|
if ( !this.mwUser.isNamed() ) {
|
|
if ( !enabled ) {
|
|
success = this.setInLocalStorage( 'wgMediaViewerOnClick', '0' ); // localStorage stringifies everything, best use strings in the first place
|
|
} else {
|
|
success = this.removeFromLocalStorage( 'wgMediaViewerOnClick' );
|
|
}
|
|
if ( success ) {
|
|
deferred = $.Deferred().resolve();
|
|
} else {
|
|
deferred = $.Deferred().reject();
|
|
}
|
|
} else {
|
|
// Simulate changing the option in Special:Preferences. Turns out this is quite hard (bug 69942):
|
|
// we need to delete the user_properties row if the new setting is the same as the default,
|
|
// otherwise set '1' for enabled, '' for disabled. In theory the pref API will delete the row
|
|
// if the new value equals the default, but this does not always work.
|
|
if ( defaultPrefValue === true ) {
|
|
newPrefValue = enabled ? '1' : '';
|
|
} else {
|
|
// undefined will cause the API call to omit the optionvalue parameter
|
|
// which in turn will cause the options API to delete the row and revert the pref to default
|
|
newPrefValue = enabled ? '1' : undefined;
|
|
}
|
|
deferred = this.setUserPreference( 'multimediaviewer-enable', newPrefValue );
|
|
}
|
|
|
|
return deferred.done( () => {
|
|
// make the change work without a reload
|
|
this.mwConfig.set( 'wgMediaViewerOnClick', enabled );
|
|
if ( !enabled ) {
|
|
// set flag for showing a popup if this was a first-time disable
|
|
this.maybeEnableStatusInfo();
|
|
}
|
|
} );
|
|
}
|
|
|
|
/**
|
|
* True if info about enable/disable status should be displayed (mingle #719).
|
|
*
|
|
* @return {boolean}
|
|
*/
|
|
shouldShowStatusInfo() {
|
|
return !isMediaViewerEnabledOnClick( this.mwConfig, this.mwUser, this.localStorage ) && this.getFromLocalStorage( 'mmv-showStatusInfo' ) === '1';
|
|
}
|
|
|
|
/**
|
|
* Called when MediaViewer is disabled. If status info was never displayed before, future
|
|
* shouldShowStatusInfo() calls will return true.
|
|
*
|
|
* @private
|
|
*/
|
|
maybeEnableStatusInfo() {
|
|
const currentShowStatusInfo = this.getFromLocalStorage( 'mmv-showStatusInfo' );
|
|
if ( currentShowStatusInfo === null ) {
|
|
this.setInLocalStorage( 'mmv-showStatusInfo', '1' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Called when status info is displayed. Future shouldShowStatusInfo() calls will return false.
|
|
*/
|
|
disableStatusInfo() {
|
|
this.setInLocalStorage( 'mmv-showStatusInfo', '0' );
|
|
}
|
|
|
|
/**
|
|
* Returns file extensions handled by Media Viewer.
|
|
*
|
|
* The object's keys are the file extensions.
|
|
* The object's values are either 'default' when Media Viewer handles that file extension
|
|
* directly or the name of a ResourceLoader module to load when such a file is opened.
|
|
*
|
|
* @return {Object}
|
|
*/
|
|
extensions() {
|
|
return this.viewerConfig.extensions;
|
|
}
|
|
|
|
/**
|
|
* Returns UI language
|
|
*
|
|
* @return {string} Language code
|
|
*/
|
|
language() {
|
|
return this.mwConfig.get( 'wgUserLanguage', false ) || this.mwConfig.get( 'wgContentLanguage', 'en' );
|
|
}
|
|
|
|
/**
|
|
* Returns URI of virtual view beacon or false if not set
|
|
*
|
|
* @return {string|boolean} URI
|
|
*/
|
|
recordVirtualViewBeaconURI() {
|
|
return this.viewerConfig.recordVirtualViewBeaconURI;
|
|
}
|
|
|
|
/**
|
|
* Returns useThumbnailGuessing flag
|
|
*
|
|
* @return {boolean}
|
|
*/
|
|
useThumbnailGuessing() {
|
|
return this.viewerConfig.useThumbnailGuessing;
|
|
}
|
|
|
|
/**
|
|
* Returns imageQueryParameter, if set
|
|
*
|
|
* @return {string|boolean}
|
|
*/
|
|
imageQueryParameter() {
|
|
return this.viewerConfig.imageQueryParameter;
|
|
}
|
|
}
|
|
|
|
module.exports = Config;
|
|
}() );
|