// ==UserScript==
// @name Aptoide Market (Direct) APK Downloader
// @name:es Descargador directo de Apps de Aptoide Market a su PC
// @namespace https://www.facebook.com/edgargerardo.chinchillamazate
// @description Download Apps from Aptoide Market to your PC directly
// @description:es Descarga Apps desde Aptoide directamente a tu PC
// @author edgerch@live
// @include *.aptoide.com/*
// @version 10.0.0
// @released 2014-10-10
// @updated 2021-02-12
// @encoding utf-8
// @homepageURL https://github.com/edgarchinchilla/aptoidemarketapkdownloader#readme
// @supportURL https://github.com/edgarchinchilla/aptoidemarketapkdownloader/issues
// @icon https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/aptoide-icon-2017-t1-300x300.png
// @grant metadata
// @grant GM.xmlHttpRequest
// @license Creative Commons Attribution License
// ==/UserScript==
// Updated & Enhanced by Edgar Gerardo Chinchilla Mazate
// [email protected]
// https://greasyfork.org/es/scripts/6952-aptoide-market-direct-apk-downloader
// Thanks to the creator of original script:
// http://userscripts-mirror.org/scripts/show/180436.html
// Thanks to the language contributors
// RU: BullFFm (https://github.com/bullffm)
/*
* GLOBAL VARS
*/
var rawJSON = null;
var userLangCode = navigator.languages ? navigator.languages[0] : (navigator.language || navigator.userLanguage || navigator.browserLanguage);
var url = window.location.toString().split('/');
var protocol = window.location.toString().split(':')[0];
var src = document.documentElement.innerHTML;
var domainDownload = 'https://pool.apk.aptoide.com/';
var divAppDown = document.createElement('div');
var divContainer = document.createElement('div');
var divElementClassName = null;
var divWrapperLevel1A = null;
var divWrapperLevel2A = null;
var divWrapperLevel2B = null;
var divWrapperLevel3A = null;
var btnDownChild = document.createElement('div');
var loadingAnimationChild = document.createElement('span');
var btnJSONChild = document.createElement('div');
var btnStrings = null;
var btnDefaultStyle = null;
var storeName = null;
var appFileNameExtended = null;
var appId = null;
var appJSONURL = null;
var appMD5 = null;
var appObbFiles = null;
var appOnSiteJSON = null;
var appPackageName = null;
var appState = null;
var appVer = null;
var apkDownloadURL = null;
/*
* DEBUG CONTROL
*/
var debugEnabled = true;
/*
* Aptoide API (v7)
*/
// https://www.aptoide.com/webservices/docs/7/app/getMeta
// https://ws2.aptoide.com/api/7/getAppMeta?apk_id=<value>&app_id=<value>&package_name=<value>&store_name=<value>&apk_md5sum=<value>
// apk_id Apk file ID. +INT31 Example: 2422529
// app_id Application ID. +INT31 Example: 7306243
// package_name Apk package name. PACKAGE_NAME Example: "cm.aptoide.pt"
// store_name Store name. DOMAIN_LABEL Example: "apps"
// apk_md5sum Apk file md5sum. HEXADECIMAL Example: "aefde962dae50881d4353089fa8039f1"
var domainWebService = 'https://ws2.aptoide.com/api/7/getAppMeta?';
/*
* STYLE & LANG STUFF
*/
// Icons and Colors
var icons = {
trusted : { color : '#01D300', imagen : 'https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/trusted.png' },
unknown : { color : '#6D6D6D', imagen : 'https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/unknown.png' },
warning : { color : '#DDB03B', imagen : 'https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/warning.png' },
critical : { color : '#C9391E', imagen : 'https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/critical.png' },
generic : { color : '#FFFFFF', imagen : 'https://github.com/edgarchinchilla/aptoidemarketapkdownloader/raw/master/resources/download.gif' },
loading : { color : '#000000', imagen : ''}
};
// Buttons generic style definition
btnDefaultStyle = document.createElement('style');
btnDefaultStyle.type = 'text/css';
// REF http://buttonoptimizer.com/
btnDefaultStyle.innerHTML = '.button {' +
'margin-bottom: 0px;' +
'display: inline-block;' +
'text-align: center;' +
'vertical-align: middle;' +
'padding: 17px 24px;' +
'border: 1px solid #fe7b4b;' +
'border-radius: 10px;' +
'background: #fe7b4b;' +
'background: -webkit-gradient(linear, left top, left bottom, from(#fe7b4b), to(#fe7b4b));' +
'background: -moz-linear-gradient(top, #fe7b4b, #fe7b4b);' +
'background: linear-gradient(to bottom, #fe7b4b, #fe7b4b);' +
'font: normal normal normal 18px tahoma;' +
'color: #ffffff;' +
'text-decoration: none;' +
'}' +
'.button:hover,' +
'.button:focus {' +
'border: 1px solid ##fe7b4b;' +
'-webkit-box-shadow: #fe7b4b 0px 0px 15px 0px;' +
'-moz-box-shadow: #fe7b4b 0px 0px 15px 0px;' +
'box-shadow: #fe7b4b 0px 0px 15px 0px;' +
'color: #ffffff;' +
'text-decoration: none;' +
'}' +
'.button:active {' +
'background: #fe7b4b;' +
'background: -webkit-gradient(linear, left top, left bottom, from(#fe7b4b), to(#fe7b4b));' +
'background: -moz-linear-gradient(top, #fe7b4b, #fe7b4b);' +
'background: linear-gradient(to bottom, #fe7b4b, #fe7b4b);' +
'}';
document.getElementsByTagName('head')[0].appendChild(btnDefaultStyle);
// Lang strings
var langStrings = {
es : { downloadAPK : 'Descargar APK', viewJSON : 'Ver JSON de información' },
en : { downloadAPK : 'Download APK', viewJSON : 'View JSON information' },
de : { downloadAPK : 'Herunterladen APK', viewJSON : 'JSON-Informationen anzeigen' },
it : { downloadAPK : 'Scarica APK', viewJSON : 'Visualizzare le informazioni JSON' },
fr : { downloadAPK : 'Télécharger APK', viewJSON : 'Voir les informations JSON' },
ru : { downloadAPK : 'Скачать APK', viewJSON : 'Просмотр JSON-информации' },
pl : { downloadAPK : 'Pobierz APK', viewJSON : 'Wyswietl informacje JSON' }
};
// Set the lang strings for the current user language
btnStrings = getLangStrings(userLangCode)
/*
* GLOBAL CHECKS
*/
// ONLY EXECUTY THE COMPLETE SCRIPT BEHAVIOR IF THE CURRENT PAGE IS FOR AN APP,
// OTHERWISE (POSSIBLY WE ARE IN THE HOME PAGE, SEARCH PAGE, ETC) DO NOTHING.
var isAnAppPage = window.location.toString().match(new RegExp(protocol + ':\/\/([A-Z]|[a-z]|\.|[0-9])*.?aptoide.com\/app', 'gi')) || [];
if (debugEnabled) { console.debug('¿App Page?: ' + isAnAppPage.length); }
var isAnErrorPage = src.toString().match(new RegExp('error_404_container', 'gi')) || [];
if (debugEnabled) { console.debug('¿Error Page?: ' + isAnErrorPage.length); }
setTimeout(function(){
if ((isAnAppPage.length > 0) && (isAnErrorPage.length <= 0)) {
/*
* APTOIDE DESKTOP AND MOBILE UNIFIED BEHAVIOR
*/
if (debugEnabled) { console.debug('APTOIDE: Processing On Site JSON'); }
// Getting RAW JSON and cleaning it
appOnSiteJSON = document.getElementById('__NEXT_DATA__').innerHTML;
appOnSiteJSON = appOnSiteJSON.replace(/\\"|"(?:\\"|[^"])*"|(\/\/.*|\/\*[\s\S]*?\*\/)/g, (m, g) => g ? "" : m);
appOnSiteJSON = JSON.parse(appOnSiteJSON);
if (debugEnabled) { console.debug('' + JSON.stringify(appOnSiteJSON)); }
// Getting required information from parsed JSON
// Get the App MD5
appMD5 = appOnSiteJSON['props']['pageProps']['app']['file']['md5sum'];
// TODO: Debug
if (debugEnabled) { console.debug('File MD5 SUM: ' + appMD5); }
// Determine the store name
storeName = appOnSiteJSON['props']['pageProps']['app']['store']['name'];
// TODO: Debug
if (debugEnabled) { console.debug('Store Name: ' + storeName); }
// Determine package name
appId = appOnSiteJSON['props']['pageProps']['app']['id'];
// TODO: Debug
if (debugEnabled) { console.debug('App ID: ' + appId); }
// Determine package name
appPackageName = appOnSiteJSON['props']['pageProps']['app']['package'];
// TODO: Debug
if (debugEnabled) { console.debug('Package Name: ' + appPackageName); }
// Determine the Application version
appVer = appOnSiteJSON['props']['pageProps']['app']['file']['vercode'];
// TODO: Debug
if (debugEnabled) { console.debug('App Version: ' + appVer); }
// Determine the App state
appState = appOnSiteJSON['props']['pageProps']['app']['file']['malware']['rank'];
// TODO: Debug
if (debugEnabled) { console.debug('App State: ' + appState); }
// Update the download link to include the store
domainDownload = domainDownload + storeName + '/';
// TODO: Debug
if (debugEnabled) { console.debug('Domain to Download: ' + domainDownload); }
// APP Full download URL
appFileNameExtended = appPackageName.replace(new RegExp("\\.","g"), "-") + '-' + appVer + '-' + appId;
apkDownloadURL = domainDownload + appFileNameExtended + '-' + appMD5 + '.apk';
// TODO: Debug
if (debugEnabled) { console.debug('APK URL: ' + apkDownloadURL); }
// JSON URL (App Metadata)
appJSONURL = domainWebService + "store_name=" + storeName + "&package_name=" + appPackageName + "&apk_md5sum=" + appMD5;
// TODO: Debug
if (debugEnabled) { console.debug('JSON URL: ' + appJSONURL); }
// Get the Aptoide Download Button Block
// header-desktop__ActionsContainer
// header-desktop__ShareWrapper
// gradient-button__GradientButton
divElementClassName = src.match(new RegExp("header-desktop__ActionsContainer-([a-z]|[A-Z]|[0-9]|-| )*","g"))[0].toString();
if (debugEnabled) { console.debug('DIV Element Class Name: ' + divElementClassName); }
divAppDown = document.getElementsByClassName(divElementClassName)[0];
if (debugEnabled) { console.debug('DIV Element HTML: ' + divAppDown); }
// Remove all the current download buttons
// Instead of remove, add a "Loading" animation and then "Direct downaload" button
// while (divAppDown.hasChildNodes()) { divAppDown.removeChild(divAppDown.firstChild); }
// Set the default loading animation
divWrapperLevel1A = src.match(new RegExp("header-desktop__HeaderContainer-([a-z]|[A-Z]|[0-9]|-| )*","g"))[0].toString();
divWrapperLevel2A = src.match(new RegExp("header-desktop__HeaderContent-([a-z]|[A-Z]|[0-9]|-| )*","g"))[0].toString();
loadingAnimationChild.innerHTML = '<div class="' + divWrapperLevel1A + '"><div class="' + divWrapperLevel2A + '"><span></span><span><img src="'+ icons.loading.imagen +'" /></span></div></div>';
//loadingAnimationChild.setAttribute('style', btnDefaultStyle);
// Show a loading animation (a timeout is needed because the Aptoide scripts reload the original HTML elements after page loads)
divAppDown.appendChild(loadingAnimationChild);
// After the "Loading" animation is shown, start the procesing of downlad APK and view JSON buttons
// btnDownChild.innerHTML = getButton(appState, apkDownloadURL, btnStrings.downloadAPK);
// Add the custom APK download button
//divAppDown.appendChild(btnDownChild);
// Read the App Information and Create the download buttons
getJSON(appJSONURL);
} else {
// Do nothing, whe are in the home, search, etc. page
// Show site Scope
if (debugEnabled) { console.debug('APTOIDE: Non-App page'); }
}
}, 3000);
/*
* PUBLIC FUNCTIONS
*/
// Replace HREF URL for matching elements
function replaceHREFURLForMatchingElements(attribute,newURL)
{
var ret = false;
var allElements = document.getElementsByTagName('*');
for (var i = 0; i < allElements.length; i++)
{
if (allElements[i].hasAttribute(attribute))
{
// If the element has an "OnClick" attribute, remove it
if (allElements[i].hasAttribute('OnClick')) {
allElements[i].removeAttribute('OnClick');
}
var childElements = allElements[i].children;
for (var j = 0; j < childElements.length; j++) {
// Replace the URL for the direct APK download one
if (childElements[j].hasAttribute('href')) {
childElements[j].setAttribute('href',newURL);
if (debugEnabled) { console.debug('Element ' + childElements[j]); }
}
// If the element has an "OnClick" attribute, remove it
if (childElements[j].hasAttribute('OnClick')) {
childElements[j].removeAttribute('OnClick');
}
}
}
}
return ret;
}
// Get an return a list of elements with matching attribute
function getAllElementsWithAttribute(attribute)
{
var matchingElements = [];
var allElements = document.getElementsByTagName('*');
for (var i = 0, n = allElements.length; i < n; i++)
{
if (allElements[i].getAttribute(attribute) !== null)
{
// Element exists with attribute. Add to array.
matchingElements.push(allElements[i]);
}
}
return matchingElements;
}
// Get and return an attribute value
function getElementAttributeValue(attribute)
{
var attributeValue = null;
var matchingElements = [];
var allElements = document.getElementsByTagName('*');
for (var i = 0, n = allElements.length; i < n; i++)
{
if (allElements[i].getAttribute(attribute) !== null)
{
// Element exists with attribute. Add to array.
attributeValue = allElements[i].getAttribute(attribute);
}
}
return attributeValue;
}
// Lang strings generator
// LANGS List: https://gist.github.com/JamieMason/3748498
function getLangStrings(langCode) {
var buttonStrings = new Object();
switch (langCode.toString().toLowerCase())
{
case 'es':
case 'es-es':
case 'es-es':
case 'es-ar':
case 'es-bo':
case 'es-cl':
case 'es-co':
case 'es-cr':
case 'es-do':
case 'es-ec':
case 'es-sv':
case 'es-gt':
case 'es-hn':
case 'es-mx':
case 'es-ni':
case 'es-pa':
case 'es-py':
case 'es-pe':
case 'es-pr':
case 'es-uy':
case 'es-ve':
default:
buttonStrings = { downloadAPK : langStrings.es.downloadAPK, viewJSONFile : langStrings.es.viewJSON }
break;
case 'en':
case 'en-us':
case 'en-gb':
case 'en-au':
case 'en-bz':
case 'en-ca':
case 'en-ie':
case 'en-jm':
case 'en-nz':
case 'en-ph':
case 'en-za':
case 'en-tt':
case 'en-zw':
buttonStrings = { downloadAPK : langStrings.en.downloadAPK, viewJSONFile : langStrings.en.viewJSON }
break;
case 'de':
case 'de-de':
case 'de-at':
case 'de-li':
case 'de-lu':
case 'de-ch':
buttonStrings = { downloadAPK : langStrings.de.downloadAPK, viewJSONFile : langStrings.de.viewJSON }
break;
case 'it':
case 'it-ch':
buttonStrings = { downloadAPK : langStrings.it.downloadAPK, viewJSONFile : langStrings.it.viewJSON }
break;
case 'fr':
case 'fr-be':
case 'fr-ca':
case 'fr-fr':
case 'fr-lu':
case 'fr-mc':
case 'fr-ch':
buttonStrings = { downloadAPK : langStrings.fr.downloadAPK, viewJSONFile : langStrings.fr.viewJSON }
break;
case 'ru':
case 'ru-ru':
buttonStrings = { downloadAPK : langStrings.ru.downloadAPK, viewJSONFile : langStrings.ru.viewJSON }
break;
case 'pl':
case 'pl-pl':
buttonStrings = { downloadAPK : langStrings.pl.downloadAPK, viewJSONFile : langStrings.pl.viewJSON }
break;
}
return buttonStrings;
}
// Generate a custom download button for every app state
function getButton(currentState, downloadURL, appDownloadString) {
var img = null;
var retButton = null;
if (currentState.toLowerCase() == 'trusted') {
// btnDefaultStyle = btnDefaultStyle + "background-color: " + icons.trusted.color + "; ";
img = icons.generic.imagen;
}
if (currentState.toLowerCase() == 'unknown') {
// btnDefaultStyle = btnDefaultStyle + "background-color: " + icons.unknown.color + "; ";
img = icons.generic.imagen;
}
if ((currentState.toLowerCase() == 'warn') || (currentState == 'warning')) {
// btnDefaultStyle = btnDefaultStyle + "background-color: " + icons.warning.color + "; ";
img = icons.generic.imagen;
}
if (currentState.toLowerCase() == 'critical') {
// btnDefaultStyle = btnDefaultStyle + "background-color: " + icons.critical.color + "; ";
img = icons.generic.imagen;
}
if (null == img) {
// btnDefaultStyle = btnDefaultStyle + "background-color: " + icons.generic.color + "; ";
img = icons.generic.imagen;
}
if (debugEnabled) { console.debug('Button Style: ' + btnDefaultStyle); }
// Formated download button
divWrapperLevel1A = src.match(new RegExp("header-desktop__HeaderContainer-([a-z]|[A-Z]|[0-9]|-| )*","g"))[0].toString();
divWrapperLevel2A = src.match(new RegExp("header-desktop__HeaderContent-([a-z]|[A-Z]|[0-9]|-| )*","g"))[0].toString();
loadingAnimationChild.innerHTML = '<div class="' + divWrapperLevel1A + '"><div class="' + divWrapperLevel2A + '"><span></span><span><img src="'+ icons.loading.imagen +'" /></span></div></div>';
retButton = '<div class="' + divWrapperLevel1A + '">' +
'<div class="' + divWrapperLevel2A + '">' +
'<span></span>' +
'<span>' +
'<a class="button" href="' + downloadURL + '">' + appDownloadString + '</a>' +
'</span>' +
'</div>' +
'</div>';
if (debugEnabled) { console.debug('Return button: ' + retButton); }
return retButton;
}
// Get the App JSON using Greasemonkey's CORS
function getJSON(jsonURL) {
var xmlCall = GM.xmlHttpRequest({
method: "GET",
headers: {"Accept": "application/json"},
ignoreCache: true,
url: jsonURL,
onload: function(res) {
rawJSON =JSON.parse(res.responseText);
createJSONButton();
},
onerror: function(res) {
rawJSON = JSON.parse('{ "error": { "message": "' + res.responseText + '" } }');
createJSONButton();
}
});
}
/*
* CREATE THE JSON LINK IF THE APP HAS AN OBB FILE
*/
// Replace the Aptoide "download" button for one that links to the direct APK donwload
var createJSONButton = function() {
// Remove the loading animation
divAppDown.removeChild(divAppDown.lastChild);
// Construct the apk filename with the format 'nombreapp-xxx-xxxxxxxx-'
appFileNameExtended = rawJSON['data']['name'];
// TODO: Debug
if (debugEnabled) { console.debug('App File Name: ' + appFileNameExtended); }
// Update other aptoide links that point to the Store APK with the actual APP full URL
// replaceHREFURLForMatchingElements("data-desktop-download-popup",rawJSON['data']['file']['path']);
// document.querySelector("#directDownloadUrl").setAttribute("href",rawJSON['data']['file']['path']);
// Construct the APK download button
btnDownChild.innerHTML = getButton(appState, rawJSON['data']['file']['path'], btnStrings.downloadAPK);
// Add the custom APK download button
divAppDown.appendChild(btnDownChild);
divAppDown.parentElement.setAttribute("href",rawJSON['data']['file']['path']);
// Construct de JSON view button
btnJSONChild.innerHTML = getButton(appState, appJSONURL, btnStrings.viewJSONFile);
// If the APP has obb files, show a button to see its JSON Info Page
if (JSON.stringify(rawJSON['data']['obb']) != 'null') {
divAppDown.appendChild(btnJSONChild);
}
}
/* FUENTES/RECURSOS
http://upload.wikimedia.org/wikipedia/commons/3/37/Aptoide_Logo.png
https://regex101.com/
http://www.javascripter.net/faq/rgbtohex.htm
http://icons8.com/web-app/for/all/download
http://dataurl.net/#dataurlmaker
http://www.developingwebs.net/html/hexgen.php
http://www.w3schools.com/jsref/jsref_trim_string.asp
http://www.w3schools.com/js/js_timing.asp
http://www.w3schools.com/jsref/jsref_obj_regexp.asp
https://www.w3schools.com/jsref/met_element_setattribute.asp
http://www.w3schools.com/js/js_cookies.asp
http://www.w3schools.com/js/js_popup.asp
http://stackoverflow.com/questions/3466356/alert-json-object
http://stackoverflow.com/questions/3172985/javascript-use-variable-in-string-match
http://stackoverflow.com/questions/9496427/get-elements-by-attribute-when-queryselectorall-is-not-available-without-using-l
http://stackoverflow.com/questions/2934137/how-to-insert-an-element-between-the-two-elements-dynamically
http://stackoverflow.com/questions/5671451/creating-a-javascript-cookie-on-a-domain-and-reading-it-across-sub-domains
http://stackoverflow.com/questions/2155737/remove-css-class-from-element-with-javascript-no-jquery
http://stackoverflow.com/questions/8074295/is-there-a-way-to-load-a-xml-file-from-another-domain-using-just-javascript
http://stackoverflow.com/questions/3926451/how-to-match-but-not-capture-part-of-a-regex
http://stackoverflow.com/questions/17883692/how-to-set-time-delay-in-javascript
http://stackoverflow.com/questions/10125561/yql-request-not-working-when-using-url
http://stackoverflow.com/questions/859024/how-can-i-use-jquery-in-greasemonkey
http://stackoverflow.com/questions/1043339/javascript-for-detecting-browser-language-preference
http://jsfiddle.net/3s7Ly/20/
http://www.aptoide.com/webservices/getApkInfo/nameofstore/name.of.app/version.of.app/xml
http://ajaxpatterns.org/XMLHttpRequest_Call#Creating_XMLHttpRequest_Objects
http://www.jibbering.com/2002/4/httprequest.html
https://developer.yahoo.com/yql/guide/yql-code-examples.html
https://developer.yahoo.com/javascript/howto-proxy.html
https://github.com/yql
http://www.360doc.com/content/06/1119/16/14389_265247.shtml
https://code.google.com/p/download-data-uri/
http://www.metamodpro.com/browser-language-codes
http://userscripts-mirror.org/scripts/review/119798
https://stackoverflow.com/questions/22191576/javascript-createelement-and-setattribute
https://stackoverflow.com/questions/4777077/removing-elements-by-class-name
https://stackoverflow.com/questions/1162529/javascript-replace-regex
https://stackoverflow.com/questions/1720320/how-to-dynamically-create-css-class-in-javascript-and-apply
Example of Trusted, Warning, Critical and Unknown Apps:
Trusted: https://pingtools-pro.en.aptoide.com/app
Warning: https://real-tiger-shark-hungry-attack.en.aptoide.com/
Critical: https://com-hoquamind-flamliesai.en.aptoide.com/
Unknown: https://com-devboy-mygame.en.aptoide.com/
Example of an API Response
https://ws2.aptoide.com/api/7/getAppMeta?app_id=53093246&store_name=aptoide-web
https://ws2.aptoide.com/api/7/getAppMeta?store_name=aptoide-web&package_name=ua.com.streamsoft.pingtoolspro&apk_md5sum=a8425618383331ddc78c5ab0e33efcf1
FUENTES/RECURSOS */