Greasy Fork は 日本語 で閲覧できます。

Aptoide Market (Direct) APK Downloader

Download Apps from Aptoide Market to your PC directly

// ==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         9.7.2
// @released        2014-10-10
// @updated         2019-04-23
// @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
// edgerch@live.com
//  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 regEx1                  = null;
var regEx2                  = null;
var rawJSON                 = null;
var isMobile                = false;
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          = 'http://pool.apk.aptoide.com/';
var divAppDown              = document.createElement('div');
var divContainer            = document.createElement('div');
var btnDownChild            = document.createElement('div');
var loadingAnimationChild   = document.createElement('span');
var btnJSONChild            = document.createElement('div');
var btnStrings              = null;
var downBtnStyle            = null;
var md5                     = null;
var appVer                  = null;
var storeName               = null;
var appState                = null;
var appId                   = null;
var appFileNameExtended     = null;
var appObbFiles             = null;
var apkDownloadURL          = null;
var appJSONURL              = null;
var homeOrSearchPage        = null;

/*
 * DEBUG CONTROL
 */
var debugEnabled = false;

/*
 * 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
    downBtnStyle = "text-align: center; " +
                   "color: #FFF; " +
                   "box-shadow: none; " +
                   "border: none; " +
                   "display: -webkit-box; " +
                   "display: -ms-flexbox; " +
                   "display: flex; " +
                   "-webkit-box-pack: center; " +
                   "-ms-flex-pack: center; " +
                   "justify-content: center; " +
                   "margin-bottom: 16px; ";
    
// 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 default loading animation
loadingAnimationChild.innerHTML = '<img style="border: none;" src="'+ icons.loading.imagen +'" />';
loadingAnimationChild.setAttribute('style', downBtnStyle);

// 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.
regEx1 = new RegExp(protocol + ':\/\/[A-Za-z]*.?aptoide.com\/.*', 'gi');
var homeOrSearchPage = window.location.toString().match(regEx1) || [];
if (homeOrSearchPage.length > 0) {
    // Do nothing, whe are in the home, search, etc. page
    // Show site Scope
    if (debugEnabled) { console.debug('APTOIDE: Homepage'); }
} else {
    // Determine if we are in the mobile version of the site
    // Even if the URL in the browser ends with *aptoide.com, javascript retrieves *aptoide.com/
    try
        {
            regEx1 = new RegExp(protocol + ':\/\/(a-z|A-Z|0-9|-|.)*.aptoide.com\/$', 'gi');
            var mobileMatch = window.location.toString().match(regEx1) || [];
            if (mobileMatch.length > 0)
            {
                isMobile = true;
            } else {
                mobileMatch = window.location.toString().match(/aptoide.com\/[a-z]*\/[a-z]*\/[a-zA-Z0-9-_.]*/gi) || [];
                if (mobileMatch.length == 0) { isMobile = true; }
            }
        }
    catch(err)
        { isMobile = false; }

    /*
     * APTOIDE DESKTOP AND MOBILE SPECIFIC BEHAVIOR
     */

    // DESKTOP
    if (!isMobile)
        {
            // Show site Scope
            if (debugEnabled) { console.debug('APTOIDE: Desktop'); }
            // Get the App MD5
            md5 = src.match(/MD5:<\/strong> [A-Za-z0-9]*/).toString().slice(14);
            // Determine the store name
            regEx1 = new RegExp(protocol + ':\/\/[A-Za-z0-9-_]*\.', 'gi');
            regEx2 = new RegExp(protocol + ':\/\/', 'gi');
            storeName = window.location.toString().match(regEx1).toString().replace(regEx2,'').replace(/.$/,'');
            // TODO: Debug
            if (debugEnabled) { console.debug('Store Name: ' + storeName); }
            // Determine the Application version
            appVer = document.getElementsByClassName('app_meta')[0].innerHTML.split("\n")[4].match(/<\/b>[a-zA-Z0-9-_.]*/gi).toString().split('>')[1].toString();
            // TODO: Debug
            if (debugEnabled) { console.debug('App Version: ' + appVer); }
            // Determine the App state
            appState = src.match(/app_install [a-z]* [a-z]*/).toString().toLowerCase().split(' ');
            appState = appState[appState.length-1].toString();
            // TODO: Debug
            if (debugEnabled) { console.debug('App State: ' + appState); }
            // Update the download link to include the store
            domainDownload = domainDownload + storeName + '/';
            // APP Full download URL
            apkDownloadURL = domainDownload + appFileNameExtended + md5 + '.apk';
            // JSON URL (App Metadata)
            appJSONURL = domainWebService + "store_name=" + storeName + "&package_name=" + url[url.length-4].toString() + "&apk_md5sum=" + md5;
            // TODO: Debug
            if (debugEnabled) { console.debug('JSON Url: ' + appJSONURL); }
            // Get the Aptoide Download Button Block
            divAppDown = document.getElementsByClassName('app_install')[0];
            // Remove all the current download buttons
            while (divAppDown.hasChildNodes()) { divAppDown.removeChild(divAppDown.firstChild); }
            btnDownChild.innerHTML = getButton(appState, apkDownloadURL, btnStrings.downloadAPK);
            // Add the custom APK download button
            divAppDown.appendChild(btnDownChild);
            // Show a loading animation
            divAppDown.appendChild(loadingAnimationChild);
            // Read the App Information and Create the download buttons
            getJSON(appJSONURL);
        }
    // MOBILE
    else
        {
            // Show site Scope
            if (debugEnabled) { console.debug('APTOIDE: Mobile'); }
            // Determine the store name
            storeName = document.getElementsByClassName('header__store-name')[0].innerHTML.toString();
            // TODO: Debug
            if (debugEnabled) { console.debug('Store Name: ' + storeName); }
            // Determine the Application version
            appVer = document.getElementsByClassName('header__stats__item')[1].getElementsByTagName('span')[1].innerHTML.toString();
            // TODO: Debug
            if (debugEnabled) { console.debug('App Version: ' + appVer); }
            // Determine the App ID
            appId = getElementAttributeValue('data-popup-url').match(/app_id=[0-9]*/gi).toString().split('=')[1];
            // TODO: Debug
            if (debugEnabled) { console.debug('App ID: ' + appId); }
            // Determine the App state
            appState = getAllElementsWithAttribute('itemscope')[0].innerHTML.match(/data-popup-badge="badge-[a-zA-Z0-9]*(?=")/gi).toString().split('-');
            appState = appState[appState.length-1].toString();
            // TODO: Debug
            if (debugEnabled) { console.debug('App State: ' + appState); }
            // JSON URL (App Metadata)
            appJSONURL = domainWebService + "store_name=" + storeName + "&app_id=" + appId;
            // TODO: Debug
            if (debugEnabled) { console.debug('JSON Url: ' + appJSONURL); }
            // Get the Aptoide Download Button Block
            divAppDown.setAttribute('class', "customDownloadBtn aptweb-button aptweb-button--big");
            document.getElementsByClassName('aptweb-button--app')[0].parentNode.removeChild(document.getElementsByClassName('aptweb-button--app')[0]);
            divContainer = document.getElementsByClassName('text-box text-box--element')[0];
            divContainer.parentNode.insertBefore(divAppDown, divContainer.nextSibling);
            // Show a loading animation
            divAppDown.appendChild(loadingAnimationChild);
            // Read the App Information and Create the download buttons
            getJSON(appJSONURL);
        }
}

/*
 * 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 == 'trusted') {
        downBtnStyle = downBtnStyle + "background-color: " + icons.trusted.color + "; ";
        img = icons.generic.imagen;
    }
    if (currentState == 'unknown') {
        downBtnStyle = downBtnStyle + "background-color: " + icons.unknown.color + "; ";
        img = icons.generic.imagen;
    }
    if ((currentState == 'warn') || (currentState == 'warning')) {
        downBtnStyle = downBtnStyle + "background-color: " + icons.warning.color + "; ";
        img = icons.generic.imagen;
    }
    if (currentState == 'critical') {
        downBtnStyle = downBtnStyle + "background-color: " + icons.critical.color + "; ";
        img = icons.generic.imagen;
    }
    if (null == img) {
        downBtnStyle = downBtnStyle + "background-color: " + icons.generic.color + "; ";
        img = icons.generic.imagen;
    }

    if (debugEnabled) { console.debug('Button Style: ' + downBtnStyle); }
    
    // Formated download button
    retButton = '<span style="vertical-align: middle;"><a style="color: ' + icons.generic.color + '; text-decoration: none;" href="' + downloadURL + '" download="' + appFileNameExtended + ' v' + appVer + '.apk' + '">' + appDownloadString + '</a></span>';
    
    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']);

    if (isMobile) {
        // 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']);
        // Customize the btnJSONChild for Mobile
        btnJSONChild.innerHTML = '<div class="aptweb-button aptweb-button--big" style="' + downBtnStyle + '"><span><a href="' + appJSONURL + '" style="color: #FFF;">' + btnStrings.viewJSONFile + '</a></span></div>';
    } else {
        // Customize the btnJSONChild for Desktop
        btnJSONChild.innerHTML = '<a href="' + appJSONURL + '" id="show_app_malware_data" data-fancybox class="app_install_badge_label">' + btnStrings.viewJSONFile + '</a>';
    }

    // Apply button customization
    divAppDown.setAttribute('style', downBtnStyle);
    if (debugEnabled) { console.debug('Download Button: ' + divAppDown); }
    
    // If the APP has obb files, show a button to see its JSON Info Page
    if (JSON.stringify(rawJSON['data']['obb']) != 'null') {
        if (isMobile)
            divAppDown.parentNode.insertBefore(btnJSONChild, divAppDown.nextSibling);
        else
            divAppDown.appendChild(btnJSONChild);
        if (debugEnabled) { console.debug('JSON Button: ' + 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
Example of Trusted, Warning, Critical and Unknown Apps:
    Trusted:  https://moto-robot-angry-shark.en.aptoide.com/
    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/
    FUENTES/RECURSOS  */
X