// ==UserScript==
// @name WPorg-Dev
// @namespace wordpress
// @description A userscript to help developers access certain WordPress.org plugin information and pages a lot easier.
// @include https://wordpress.org/plugins/*
// @include https://*.wordpress.org/plugins/*
// @version 2.0
// @copyright 2017 Armando Lüscher
// @author Armando Lüscher
// @oujs:author noplanman
// @grant GM_addStyle
// @require https://code.jquery.com/jquery-1.11.3.min.js
// @homepageURL https://github.com/noplanman/WPorg-Dev-Userscript
// @supportURL https://github.com/noplanman/WPorg-Dev-Userscript/issues
// @downloadURL none
// ==/UserScript==
// The base URL of the WordPress plugins pages.
var wpBaseURL = location.protocol + '//' + location.host;
var pluginsBaseURL = wpBaseURL + '/plugins/';
var wodu = {};
/**
* Extract the plugin slug from a URL.
* @param {string} url URL to extract the plugin slug from.
* @return {string} The extracted plugin slug.
*/
wodu.getSlug = function (url) {
var p = url.indexOf(pluginsBaseURL);
if (p >= 0) {
return url.substring(p + pluginsBaseURL.length).split('/')[0];
}
return '';
};
/**
* Generate the failed message and the retry button.
*
* @param {jQuery} $el The error message panel gets appended to this jQuery object.
* @param {Function} cb Callback function when clicking the "Retry" button.
* @return {jQuery} The jQuery object containing the message and button.
*/
wodu.getFailedRetryButton = function ($el, cb) {
$el.removeClass('wodu-loaded');
return $('
Failed.
')
.append($('Retry
').click(cb))
.appendTo($el);
};
/**
* Generate a dropdown menu with all the downloadable versions.
*
* When the selection changes, the selected version download begins.
*
* @param {jQuery} $devPage The page containing the links.
* @return {jQuery} Dropdown menu with all the links.
*/
wodu.getDLLinkDropdown = function ($devPage) {
var $select = $('')
.append('')
.click(function () {
this.value = '';
})
.change(function () {
if (this.value) location.href = this.value;
});
// Force width of first entry. (https://stackoverflow.com/a/27442394)
var $selectTmp = $select.clone().hide();
$selectTmp.appendTo($('body'));
$select.width($selectTmp.width() + 10);
$selectTmp.remove();
$('.previous-versions option', $devPage).each(function () {
$select.append('');
});
return $select;
};
/**
* Get the repository (and admin) links from the developers page.
*
* @param {string} slug Plugin slug.
* @param {jQuery} $devPage The page containing the links.
* @return {array} An array of the developer links.
*/
wodu.getDevLinks = function (slug, $devPage) {
return [
'Developers',
'SVN',
'Trac',
'Log'
];
};
/**
* Load the extra plugin infos for a certain plugin.
*
* @param {jQuery} $card The plugin cart to load the infos for.
*/
wodu.loadPluginCardExtra = function ($card) {
if ($card.hasClass('wodu-loaded')) {
return;
}
$card.addClass('wodu-loaded');
var slug = wodu.getSlug($('.entry-title a', $card).attr('href'));
// Get rid of any error message that may be there.
var $panelInfo = $('.wodu-panel-info', $card).empty();
var $spinner = $('.wodu-spinner', $card).show();
$.get(pluginsBaseURL + slug + '/advanced', function (response) {
// Get rid of all images first, no need to load those.
var $devPage = $(response.replace(/
]*>/g, ''));
var $meta = $('.plugin-meta', $devPage);
// Remove languages section, no need for it.
$('.languages', $meta).parent().remove();
$panelInfo.append($meta);
var $panelDev = $('.wodu-panel-dev', $card)
.append(wodu.getDevLinks(slug, $devPage).join(' - '))
.append(wodu.getDLLinkDropdown($devPage));
})
.fail(function () {
wodu.getFailedRetryButton($panelInfo, function () {
wodu.loadPluginCardExtra($card);
});
})
.always(function () {
$spinner.hide();
});
};
/**
* Add extra plugin information to plugin cards.
*
* @param {jQuery} $pluginCards All the plugin cards displayed on the current page.
*/
wodu.setupPluginCardExtras = function ($pluginCards) {
// Add the CSS.
GM_addStyle(
'.wodu-close { position: absolute; top: 4px; right: 4px; }' +
'.wodu-plugin-card { position: relative; }' +
'.wodu-title { font-size: 1.1em; }' +
'.wodu-dllink-dropdown { float: right; padding: 3px 5px; }' +
'.wodu-plugin-card-extras { box-sizing: border-box; overflow: auto; display: none; position: absolute; width: 100%; height: 100%; margin: -15px; border: 1px solid #ccc; padding: 4px 10px; background-color: rgba(255, 255, 255, 0.9); z-index: 1; }' +
// Little triangle.
'.wodu-plugin-card:before { content: ""; position: absolute; top: 0px; right: 0px; border-width: 36px 0 0 36px; border-style: solid; border-color: #ddd transparent; }' +
'.wodu-extras-button { position: absolute; top: 4px; right: 4px; height: 16px; width: 16px; cursor: pointer; background: url(); }'
);
$pluginCards.each(function () {
var $card = $(this).addClass('wodu-plugin-card');
var $extrasButton = $('')
.click(function () {
$extrasButton.hide();
$extras.show();
wodu.loadPluginCardExtra($card);
})
.prependTo($card);
// Prepare the extras.
var $close = $('')
.click(function () {
$extras.hide();
$extrasButton.show();
});
var $extras = $('', {class: 'wodu-plugin-card-extras'})
.append($close)
.append('')
.append('' + $('.entry-title a', $card).parent().html() + '
')
.append('')
.append('')
.prependTo($card);
});
};
/**
* Start the party.
*/
wodu.init = function () {
// Add the global CSS rules.
GM_addStyle(
'.wodu-spinner { position: absolute; left: 50%; top: 50%; height: 16px; width: 16px; background: no-repeat center center url() }' +
'.wodu-close { height: 16px; width: 16px; cursor: pointer; background: url() }' +
'.wodu-failed { margin: 0; }' +
'.wodu-panel-info .plugin-meta { margin: 0; }' +
'.wodu-panel-info .plugin-meta .tags { width: auto; }' +
'.wodu-panel-info .plugin-meta li { padding: 0; }'
);
var $pluginCards = $('.plugin-card');
if ($pluginCards.length) {
wodu.setupPluginCardExtras($pluginCards);
}
};
// source: https://muffinresearch.co.uk/does-settimeout-solve-the-domcontentloaded-problem/
if (/(?!.*?compatible|.*?webkit)^mozilla|opera/i.test(navigator.userAgent)) { // Feeling dirty yet?
document.addEventListener('DOMContentLoaded', wodu.init, false);
} else {
window.setTimeout(wodu.init, 0);
}