// ==UserScript==
// @name Direct download from Google Play
// @name:it Download diretto dal Google Play Store
// @namespace StephenP
// @version 3.5.0
// @description Adds APK-DL, APKPure, APKCombo, APKPremier, APKMirror and Evozi download buttons to Google Play Store when browsing apps.
// @description:it Aggiunge i tasti di download di APK-DL, APKPure, APKCombo, APKPremier, APKMirror e Evozi al Google Play Store quando si naviga tra le applicazioni.
// @author StephenP
// @icon 
// @match https://play.google.com/*
// @match http://play.google.com/*
// @match http://apkfind.com/store/captcha?app=*
// @grant GM.xmlHttpRequest
// @grant GM.getValue
// @grant GM.setValue
// @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
// @connect self
// @connect apkpure.com
// @connect apkfind.com
// @connect apk-cloud.com
// @connect winudf.com
// @connect apkcombo.com
// @connect down-apk.com
// @connect play.googleapis.com
// @connect gvt1.com
// @connect apkpremier.com
// @connect web-api.aptoide.com
// @connect apk.support
// @contributionURL https://buymeacoffee.com/stephenp_greasyfork
// ==/UserScript==
var ui;
var wlButton;
var pageURL;
var title;
var appCwiz;
var done=[0];
var useGS;
(async function(){
useGS=await GM.getValue("useGS", false);
starter();
})();
function starter() {
if(document.location.href.includes("apkfind")===true){
setInterval(unredirect,100);
}
else{
try{
'use strict';
var site=window.location.href.toString();
ui=checkUI();
pageURL=location.href;
if(ui>0){
title=document.getElementById("main-title").innerHTML;
var buttonsStyle=document.createElement("style");
var styleString;
styleString='.ddlButton:visited{color: white;} .ddlButton:hover{opacity: 0.8;} .ddlButton:active{opacity: 0.6;} .ddlButton{font-family: "Google Sans",Roboto,Arial,sans-serif; line-height: 3rem; font-size: 1rem; font-weight: 500; height: 44px; margin-right: 0.5em; min-height: 44px; min-width: 200px; padding: 10px 16px; border-radius: 0.5em; color: white; position: relative; z-index: 0;}';
buttonsStyle.textContent=styleString;
document.body.appendChild(buttonsStyle);
}
if((pageURL.includes("details?id="))||(pageURL.includes("/store/search?q="))){
addButtons();
}
setInterval(checkReload, 2000);
}
catch(err){
console.log("main(): "+err);
}
}
}
function unredirect(){
var tot=document.body.children.length-1;
if(parseInt(document.body.children[tot].style.zIndex, 10)>2){
if(document.body.children[tot].id==""){
document.body.children[tot].style.zIndex="1";
document.body.children[tot-1].style.zIndex="-1000";
}
else{
document.body.children[tot].style.zIndex="-1000";
}
}
}
function waitForRemovingButtons(){
//if(title!=document.getElementById("main-title").innerHTML){
if((pageURL!=location.href)||(isButtonVisible()===false)){
title=document.getElementById("main-title").innerHTML;
pageURL=location.href;
wlButton=null;
if((location.href.includes("details?id="))||(location.href.includes("/store/search?q="))){
if((ui>0)&&(document.getElementsByClassName("ddlButton").length>0)){
try{
removePreviousCwiz();
}
catch(err){
console.log(err+"; I was probably just trying to remove buttons that weren't there...");
}
}
addButtons();
}
}
else{
setTimeout(waitForRemovingButtons, 1000);
}
}
function checkReload(){
if((pageURL!=location.href)||(isButtonVisible()===false)){
waitForRemovingButtons();
}
}
function isButtonVisible(){
var allButtons=document.getElementsByClassName("ddlButton");
//console.log("how many buttons: "+allButtons.length);
if(allButtons.length>0){
for(var i=0;i<allButtons.length;i++){
if(allButtons[i].offsetParent!=null){
//console.log(i+true);
return true;
}
}
//console.log(i+false);
return false;
}
else{
if(document.location.href.includes("play.google.com/store/apps/details")){
console.log("apppage//"+false);
return false;
}
//console.log("notapppage//"+false);
return true;
}
}
function addButtons(){
var price=-1;
var installButton=null;
var instWishButtons=[];
if(ui>0){
installButton=get2022UIButtons();
while(installButton.tagName!="C-WIZ"){
installButton=installButton.parentNode;
}
try{
price=installButton.querySelector("meta[itemprop=price]").content;
//alert("Price: "+price);
}
catch(err){
console.error("Price not found. Maybe the app is already installed or not officially available?");
price=0;
}
//determina c-wiz dell'app per poterlo radere al suolo al cambio di pagina
var currentNode;
currentNode=installButton.parentNode;
do{
if(currentNode.tagName=="C-WIZ"){
appCwiz=currentNode;
}
currentNode=currentNode.parentNode;
}while(currentNode.tagName!="BODY");
}
else{//this is the part for when you land on a missing application.
document.getElementById("search-section").lastChild.remove();
let searchSection=document.getElementById("search-section");
let x=document.createElement("SPAN");
x.style="margin-top: 32px; float: left";
x.textContent="or ";
searchSection.appendChild(x);
let y=document.createElement("SPAN");
let z=document.createElement("A");
z.style="background-color: #FF8B14; font-weight: bold; text-decoration: none; padding: 1em; margin: 17px; float: left; color: white; cursor: pointer;";
z.textContent="Search on APKMirror";
z.className="rounded";
z.id="apkMirrorBtn";
//z.href='https://www.apkmirror.com/?post_type=app_release&searchtype=apk&s='+location.search.match(/id=(.*)/)[1].split("&", 1);
let apkmirrorURL='https://www.apkmirror.com/?post_type=app_release&searchtype=apk&s='+location.search.match(/id=(.*)/)[1].split("&", 1);
z.addEventListener("click",function opn(){window.open(apkmirrorURL)});
y.appendChild(z);
searchSection.appendChild(y);
let w=document.createElement("DIV");
w.style="clear:both";
searchSection.appendChild(w);
GM.xmlHttpRequest({
method: "GET",
url: 'https://www.apkmirror.com/?post_type=app_release&searchtype=apk&s='+location.search.match(/id=(.*)/)[1].split("&", 1),
timeout: 5000,
onload: function(response){
let parser = new DOMParser();
let doc = parser.parseFromString(response.responseText, "text/html");
if(doc.getElementById("content").getElementsByClassName("appRow").length>0){
z.textContent="Available on APKMirror";
}
else{
let w=z.cloneNode(true);
w.textContent="Not available on APKMirror";
w.style.backgroundColor="#CCCCCC";
w.style.cursor="not-allowed";
z.parentNode.replaceChild(w, z);
}
}
});
}
if(price==0){
var html;
var buttonslist;
var id;
if(location.href.includes("details?id=")){
id=location.search.match(/id=(.*)/)[1].split("&", 1);
}
else if(location.href.includes("/store/search?q=")){
let idAttr=installButton.querySelector("[data-item-id^=\"%.@.\"]").getAttribute("data-item-id");
id=idAttr.match(/%\.@\.".+"/)[0].replace("%.@.\"","").replace("\"","");
}
var apkpureURL='https://d.apkpure.com/b/APK/'+id+'?version=latest';
var evoziURL='https://apps.evozi.com/apk-downloader/?id='+id;
var apkdlURL='http://apkfind.com/store/download?id='+id;
var apkmirrorURL='https://www.apkmirror.com/?post_type=app_release&searchtype=apk&s='+id;
var apkleecherURL='https://apkleecher.com/download/dl.php?dl='+id;
var apkcomboURL='https://apkcombo.com/genericApp/'+id+'/download/apk';
var apkpremierURL='https://apkpremier.com/download/'+id.toString().replace(/[.]/g,"-");
var aptoideURL="https://web-api.aptoide.com/search?query="+id;
var apksupportURL="https://apk.support/download-app/"+id;
wlButton = document.createDocumentFragment();
var wishListButton;
var cn="";
buttonslist = installButton.parentNode;
cn="ddlButton";
let b1=document.createElement("span");
b1.id="apkdlbutton";
let b1a=document.createElement("A");
b1a.style.backgroundColor="#009688";
b1a.className=cn;
b1a.textContent="APK-DL";
b1.appendChild(b1a);
//
let b2=document.createElement("span");
b2.id="apkpurebutton";
let b2a=document.createElement("A");
b2a.style.backgroundColor="#24cd77";
b2a.className=cn;
b2a.textContent="APKPure";
b2.appendChild(b2a);
//
let b3=document.createElement("span");
b3.id="apkcombobutton";
let b3a=document.createElement("A");
b3a.style.backgroundColor="#00875f";
b3a.className=cn;
b3a.textContent="APKCombo";
b3a.href=apkcomboURL;
b3.appendChild(b3a);
//
let b4=document.createElement("span");
b4.id="apkpremierbutton";
let b4a=document.createElement("A");
b4a.style.backgroundColor="#3740ff";
b4a.className=cn;
b4a.textContent="APKPremier";
b4.appendChild(b4a);
//
let b5=document.createElement("span");
let b5a=document.createElement("A");
b5a.addEventListener("click",function(){window.open(evoziURL)});
b5a.style.backgroundColor="#286090";
b5a.className=cn;
b5a.textContent="Evozi";
b5.appendChild(b5a);
//
let b6=document.createElement("span");
let b6a=document.createElement("A");
b6a.addEventListener("click",function(){window.open(apkmirrorURL)});
b6a.style.backgroundColor="#FF8B14";
b6a.className=cn;
b6a.textContent="APKMirror";
b6.appendChild(b6a);
//
let b7=document.createElement("span");
b7.id="aptoidebutton";
let b7a=document.createElement("A");
b7a.style.backgroundColor="#fe6446";
b7a.className=cn;
b7a.textContent="Aptoide";
b7.appendChild(b7a);
//
let b8=document.createElement("span");
b8.id="apksupportbutton";
let b8a=document.createElement("A");
b8a.style.backgroundColor="#3740ff";
b8a.className=cn;
b8a.textContent="APKSupport";
b8a.href=apksupportURL;
b8.appendChild(b8a);
//
buttonslist.appendChild(b1);
buttonslist.appendChild(b2);
buttonslist.appendChild(b3);
buttonslist.appendChild(document.createElement("BR"));
buttonslist.appendChild(b4);
buttonslist.appendChild(b5);
buttonslist.appendChild(b6);
buttonslist.appendChild(b7);
buttonslist.appendChild(b8);
if(typeof wishListButton!=='undefined'){
wlButton.appendChild(wishListButton.firstChild.firstChild);
}
buttonslist.appendChild(wlButton);
var ddlButton1=document.getElementById("apkdlbutton");
ddlButton1.onclick=function(){ddl(this,apkdlURL,id);};
var ddlButton2=document.getElementById("apkpurebutton");
ddlButton2.onclick=function(){ddl(this,apkpureURL,id);};
var ddlButton3=document.getElementById("apkcombobutton");
ddlButton3.onclick=function(){ddl(this,apkcomboURL,id); return false};
var ddlButton4=document.getElementById("apkpremierbutton");
ddlButton4.onclick=function(){ddl(this,apkpremierURL,id);};
var ddlButton7=document.getElementById("aptoidebutton");
ddlButton7.onclick=function(){ddl(this,aptoideURL,id);};
var ddlButton8=document.getElementById("apksupportbutton");
ddlButton8.onclick=function(){ddl(this,apksupportURL,id); return false};
//document.getElementById("useGoogleServers").checked=useGS;
//document.getElementById("gsExpl").addEventListener("click",function(){alert("If you choose the option \"Use Google\'s servers when downloading from APKCombo\", packages are directly downloaded form Play Store servers, but file names are randomized. Otherwise files are downloaded from APKCombo\'s own servers, with correct names.")});
}
}
function openLink(link){
window.open(link.replace("http://","https://"),"_self");
}
function ddlFinalApk(link,ddlButton,i){
if(link!=""){
done[i]=0;
GM.xmlHttpRequest({
method: "GET",
url: link,
timeout: 5000,
ontimeout: function(response) {
if(done[i]==0){
ddlButton.firstChild.textContent="Retry";
}
else{
done[i]=0;
}
},
onprogress: function(response){
//console.log(response.finalUrl);
if((response.finalUrl.includes("winudf.com"))||(response.finalUrl.includes("down-apk.com"))||(response.finalUrl.includes("/play-apps-download-default/"))){
if(done[i]==0){
console.log("downloading file n."+i);
done[i]=1;
if(link.includes("apkpure")){
window.open(response.finalUrl,"_self");
ddlButton.onclick=function(){openLink(response.finalUrl);};
ddlButton.firstChild.textContent="Ready!";
}
else if(link.includes("apkpremier")){
window.open(response.finalUrl,"_self");
ddlButton.onclick=function(){openLink(response.finalUrl);};
ddlButton.firstChild.textContent="Ready!";
}
else{
window.open(response.finalUrl,i);
ddlButton.firstChild.textContent="APKCombo";
}
}
}
},
onload: function(response){
if(done[i]==0){
ddlButton.firstChild.textContent="Retry";
}
else{
done[i]=0;
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
return 0
}
else{
buttonError(ddlButton,"Failed!");
return -1
}
}
function ddl(ddlButton,ddlURL,packageId){
ddlButton.firstChild.textContent="Loading...";
if(ddlURL.includes("apkfind")){
try {
var apkDlRequest1=GM.xmlHttpRequest({
method: "GET",
url: ddlURL,
onload: function(response) {
console.log(response);
if(response.finalUrl.includes("/captcha?")){
ddlButton.firstChild.addEventListener("click",function(){window.open(response.finalUrl)});
//ddlButton.firstChild.setAttribute("href",response.finalUrl);
ddlButton.firstChild.textContent="CAPTCHA";
ddlButton.onclick=null;
}
else if(response.finalUrl.includes("app/removed")){
buttonError(ddlButton,"Removed!");
}
else{
try{
let parser = new DOMParser();
var linkIntermediary = parser.parseFromString(response.response, 'text/html');
var link="http:"+linkIntermediary.getElementsByClassName("mdl-button")[0].getAttribute("href");
ddlButton.firstChild.textContent="Ready!";
openLink(link);
ddlButton.onclick=function(){openLink(link);};
}
catch(err){
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
}
catch (err) {
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
else if(ddlURL.includes("apkpure")){
if(ddlFinalApk(ddlURL,ddlButton,0)==-1){//if 0 is returned (request completed), the download has started and the following won't be run. Otherwise, the script will try the following method to extract the link.
try{
GM.xmlHttpRequest({
method: "GET",
url: ddlURL,
onload: function(response) {
switch (response.status) {
case 410:
buttonError(ddlButton, "Removed!");
break;
case 404:
buttonError(ddlButton, "Not found!");
break;
default:
var apklink=response.responseText.substr(response.responseText.indexOf('https://download.apkpure.com/b/'),response.responseText.length-1);
apklink=apklink.substr(0,apklink.indexOf('"'));
console.log(ddlURL);
ddlButton.firstChild.textContent="Wait...";
//ddlButton.onclick=function(){GM.openInTab(apklink,"open_in_background");};
ddlFinalApk(apklink,ddlButton,0);
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
}
catch(err){
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
}
else if(ddlURL.includes("apkcombo")){
try{
var checkin;
GM.xmlHttpRequest({
method: "POST",
url: "https://apkcombo.com/checkin",
headers: {
"Referer": ddlURL,
"Origin": "https://apkcombo.com"
},
onload: function(response) {
checkin=response.responseText;
GM.xmlHttpRequest({
method: "GET",
url: ddlURL,
onload: function(response) {
switch (response.status) {
case 410:
buttonError(ddlButton, "Removed!");
break;
case 404:
buttonError(ddlButton, "Not found!");
break;
default:
try {
var i;
var parser = new DOMParser();
var resp = parser.parseFromString(response.responseText, 'text/html');
var combo = resp.getElementsByClassName("file-list")[0];
if (combo !== undefined){
var combolinks = combo.getElementsByTagName("a");
for (i = 0; i < combolinks.length; i++) {
let fileLink=combolinks[i].getAttribute("href");
if(fileLink.startsWith("/r2?u=")){
try{
window.open(decodeURIComponent(fileLink).split("?u=")[1]);
}
catch(e){
console.log(e);
buttonError(ddlButton, "Error!");
}
}
else{
window.open(fileLink+"&"+checkin);
}
}
ddlButton.firstChild.textContent="APKCombo";
}
else{ //if loading the main download page results in an empty list of apks, tries to read the token to request directly the urls from apkcombo server
var tokenStart=response.responseText.indexOf("/dl?token=")+4;
var tokenEnd=response.responseText.indexOf("\"",tokenStart);
var token = response.responseText.substring(tokenStart,tokenEnd);
ddlURL=response.finalUrl;
GM.xmlHttpRequest({
method: "POST",
url: ddlURL.replace("/download/apk", "/dl")+"?"+token,
onload: function(response) {
var parser2 = new DOMParser();
var resp2 = parser2.parseFromString(response.responseText, 'text/html');
combo = resp2.getElementsByClassName("file-list")[0];
if (combo !== null) {
var combolinks = combo.getElementsByTagName("a");
for (i = 0; i < combolinks.length; i++) {
window.open(combolinks[i].getAttribute("href")+"&"+checkin);
}
ddlButton.firstChild.textContent="APKCombo";
} else {
ddlButton.firstChild.addEventListener("click",function(){window.open(ddlURL)});
//ddlButton.firstChild.setAttribute("href", ddlURL);
ddlButton.firstChild.textContent = "New tab >";
ddlButton.onclick = null;
}
},
onerror: function(response) {
buttonError(ddlButton, "Error!");
}
});
}
} catch (err) {
console.log(err);
}
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
}
});
}
catch(err){
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
else if(ddlURL.includes("apkpremier")){
try{
GM.xmlHttpRequest({
method: "POST",
url: ddlURL,
data: "cmd=apk&gc=0",
//data: "pa=xapk&gid="+ddlURL.substr(32).replace(/[-]/g,"."),
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function(response) {
switch (response.status) {
case 410:
buttonError(ddlButton, "Removed!");
break;
case 404:
buttonError(ddlButton, "Not found!");
break;
default:
let parser = new DOMParser();
const respDom = parser.parseFromString(response.responseText, "text/html");
let apkLinks=respDom.getElementsByClassName("bdlinks");
if(response.responseText=="error capchar"){
ddlButton.firstChild.addEventListener("click",function(){window.open(ddlURL)});
ddlButton.firstChild.textContent="CAPTCHA";
ddlButton.onclick=null;
}
else{
if(apkLinks.length>0){
for(let apkLink of apkLinks){
window.open(apkLink.firstElementChild.getAttribute("href"),"_self");
}
ddlButton.firstChild.textContent="Ready!";
}
else{
ddlButton.firstChild.textContent="Failed!";
}
}
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
}
catch(err){
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
else if(ddlURL.includes("aptoide")){
try{
GM.xmlHttpRequest({
method: "GET",
url: ddlURL,
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function(response) {
switch (response.status) {
case 410:
buttonError(ddlButton, "Removed!");
break;
case 404:
buttonError(ddlButton, "Not found!");
break;
default:
let searchResults=JSON.parse(response.responseXML);
if(searchResults.datalist.total>0){
let packageId=ddlURL.split("?query=")[1];
for(let app of searchResults.datalist.list){
if(app.package===packageId){
//should dowload splits and obb. I'm absolutely not sure wether this works properly or not once installed
if(app.aab){
for(let split of app.aab.splits){
window.open(split.path);
}
}
if(app.obb){
window.open(app.obb.main.path);
}
//
openLink(searchResults.datalist.list[0].file.path);
ddlButton.firstChild.textContent="Ready!";
}
}
if(ddlButton.firstChild.textContent!="Ready!"){
buttonError(ddlButton,"Not found!");
}
}
else{
ddlButton.firstChild.textContent="Failed!";
}
}
},
onerror: function(){
buttonError(ddlButton,"Offline!");
}
});
}
catch(err){
buttonError(ddlButton,"Failed!");
console.log(err);
}
}
else if(ddlURL.includes("apk.support")){
try{
GM.xmlHttpRequest({
method: "POST",
url: ddlURL,
data: "cmd=csapk&pkg="+packageId+"&arch=default&tbi=default&device_id=&model=default&language=en&dpi=480&av=default",
headers: {
"Referer": ddlURL,
"Origin": "https://apk.support",
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function(response) {
if(response.responseXML){
let links=response.responseXML.querySelectorAll("[href*='.androidcontents.com/']");
if(links.length==0){
links=response.responseXML.querySelectorAll("[href*='.googleapis.com/download/']");
}
if(links.length>0){
ddlButton.firstChild.textContent="Ready!";
for(let l of links){
window.open(l);
}
}
else{
buttonError(ddlButton, "Not found!");
}
}
},
onerror: function(response) {
buttonError(ddlButton, "Error!");
}
});
}
catch (err) {
console.log(err);
}
}
}
function get2022UIButtons(){
var matchingElements=[];
matchingElements=document.querySelectorAll("[data-item-id^=\"%.@.\"]");
if(matchingElements.length==0){
matchingElements=document.querySelectorAll("[data-p^=\"%.@.[\"]");
for(let element of matchingElements){
if((element.querySelector("[fill-rule=evenodd]"))&&(element.getAttribute("data-p").includes("\",7]]"))){//hard but hopefully durable way to find the wishlist button if install button is not on the page.
return element;
}
}
}
console.log("Install buttons:");
console.log(matchingElements);
return matchingElements[0];
}
function checkUI(){
//Different UIs:
//0=Missing APK page
//5=2022UI
var check=0;
try{
if(document.getElementsByTagName("header").length>0){
check=5;
}
else{
check=0;
}
}
catch(err){
console.error('The user interface of Google Play Store was not recognized by "Direct Download from Google Play" script. This might result in unexpected behaviour of the page. Please report the error to the author on Greasyfork. Error: '+err);
}
console.log("PLAY STORE INTERFACE: "+check);
return check;
}
function removePreviousCwiz(){
appCwiz.parentNode.removeChild(appCwiz);
}
function buttonError(ddlButton,error){
ddlButton.firstChild.textContent=error;
ddlButton.firstChild.style.backgroundColor="#CCCCCC";
ddlButton.onclick=null;
}
async function setUseGS(check){
useGS=check;
GM.setValue("useGS", check);
}