Dateien nach "/" hochladen
This commit is contained in:
parent
931304b808
commit
8b9260a195
@ -1,2 +1,7 @@
|
|||||||
# CRX-Extractor-Downloader
|
# crx-download
|
||||||
|
Download CRX files as zip or directly.
|
||||||
|
|
||||||
|
### Store Links
|
||||||
|
|
||||||
|
1. [Chrome Store](https://chrome.google.com/webstore/detail/crx-extractordownloader/ajkhmmldknmfjnmeedkbkkojgobmljda)
|
||||||
|
2. [Microsoft Edge Store](https://microsoftedge.microsoft.com/addons/detail/crx-extractordownloader/gfgehnhkaggeillajnpegcanbdjcbeja)
|
||||||
|
191
background.js
Normal file
191
background.js
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
let chromeURLPattern = /^https?:\/\/chrome.google.com\/webstore\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
let microsoftURLPattern = /^https?:\/\/microsoftedge.microsoft.com\/addons\/detail\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
let chromeNewURLPattern = /^https?:\/\/chromewebstore.google.com\/detail\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
|
||||||
|
|
||||||
|
function getChromeVersion() {
|
||||||
|
var pieces = navigator.userAgent.match(/Chrom(?:e|ium)\/([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)/);
|
||||||
|
if (pieces == null || pieces.length != 5) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
pieces = pieces.map(piece => parseInt(piece, 10));
|
||||||
|
return {
|
||||||
|
major: pieces[1],
|
||||||
|
minor: pieces[2],
|
||||||
|
build: pieces[3],
|
||||||
|
patch: pieces[4]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNaclArch() {
|
||||||
|
var nacl_arch = 'arm';
|
||||||
|
if (navigator.userAgent.indexOf('x86') > 0) {
|
||||||
|
nacl_arch = 'x86-32';
|
||||||
|
} else if (navigator.userAgent.indexOf('x64') > 0) {
|
||||||
|
nacl_arch = 'x86-64';
|
||||||
|
}
|
||||||
|
return nacl_arch;
|
||||||
|
}
|
||||||
|
let currentVersion = getChromeVersion();
|
||||||
|
let version = currentVersion.major + "." + currentVersion.minor + "." + currentVersion.build + "." + currentVersion.patch;
|
||||||
|
const nacl_arch = getNaclArch();
|
||||||
|
|
||||||
|
function getTabTitle(title, currentEXTId, url) {
|
||||||
|
if (!chromeNewURLPattern.exec(url)) {
|
||||||
|
title = title.match(/^(.*[-])/);
|
||||||
|
if (title) {
|
||||||
|
title = title[0].split(' - ').join("");
|
||||||
|
} else {
|
||||||
|
title = currentEXTId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Ѐ-ӿ matches cyrillic characters
|
||||||
|
return (title).replace(/[&\/\\#,+()$~%.'":*?<>|{}\sЀ-ӿ]/g, '-').replace(/-*$/g, '').replace(/-+/g, '-');
|
||||||
|
}
|
||||||
|
|
||||||
|
function download(downloadAs, tab) {
|
||||||
|
|
||||||
|
var query = {
|
||||||
|
active: true,
|
||||||
|
currentWindow: true
|
||||||
|
};
|
||||||
|
result = chromeURLPattern.exec(tab.url);
|
||||||
|
if (!result) {
|
||||||
|
result = chromeNewURLPattern.exec(tab.url);
|
||||||
|
}
|
||||||
|
if (result && result[1]) {
|
||||||
|
var name = getTabTitle(tab.title, result[1], tab.url);
|
||||||
|
if (downloadAs === "zip") {
|
||||||
|
url = `https://clients2.google.com/service/update2/crx?response=redirect&prodversion=${version}&x=id%3D${result[1]}%26installsource%3Dondemand%26uc&nacl_arch=${nacl_arch}&acceptformat=crx2,crx3`;
|
||||||
|
convertURLToZip(url, function (urlVal) {
|
||||||
|
downloadFile(urlVal, name + ".zip");
|
||||||
|
|
||||||
|
});
|
||||||
|
} else if (downloadAs === "crx") {
|
||||||
|
url = `https://clients2.google.com/service/update2/crx?response=redirect&prodversion=${version}&acceptformat=crx2,crx3&x=id%3D${result[1]}%26uc&nacl_arch=${nacl_arch}`;
|
||||||
|
downloadFile(url, name + ".crx", result[1] + ".crx");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var edgeId = microsoftURLPattern.exec(tab.url);
|
||||||
|
if (edgeId && edgeId[1] && downloadAs === "crx") {
|
||||||
|
var name = getTabTitle(tab.title, edgeId[1], tab.url);
|
||||||
|
url = `https://edge.microsoft.com/extensionwebstorebase/v1/crx?response=redirect&prod=chromiumcrx&prodchannel=&x=id%3D${edgeId[1]}%26installsource%3Dondemand%26uc`;
|
||||||
|
downloadFile(url, name + ".crx", edgeId[1] + ".crx");
|
||||||
|
}
|
||||||
|
// });
|
||||||
|
}
|
||||||
|
|
||||||
|
function ArrayBufferToBlob(arraybuffer, callback) {
|
||||||
|
|
||||||
|
var data = arraybuffer;
|
||||||
|
var buf = new Uint8Array(data);
|
||||||
|
var publicKeyLength, signatureLength, header, zipStartOffset;
|
||||||
|
if (buf[4] === 2) {
|
||||||
|
header = 16;
|
||||||
|
publicKeyLength = 0 + buf[8] + (buf[9] << 8) + (buf[10] << 16) + (buf[11] << 24);
|
||||||
|
signatureLength = 0 + buf[12] + (buf[13] << 8) + (buf[14] << 16) + (buf[15] << 24);
|
||||||
|
zipStartOffset = header + publicKeyLength + signatureLength;
|
||||||
|
} else {
|
||||||
|
publicKeyLength = 0 + buf[8] + (buf[9] << 8) + (buf[10] << 16) + (buf[11] << 24 >>> 0);
|
||||||
|
zipStartOffset = 12 + publicKeyLength;
|
||||||
|
}
|
||||||
|
// 16 = Magic number (4), CRX format version (4), lengths (2x4)
|
||||||
|
|
||||||
|
return new Blob([
|
||||||
|
new Uint8Array(arraybuffer, zipStartOffset)
|
||||||
|
], {
|
||||||
|
type: 'application/zip'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertURLToZip(url, callback, xhrProgressListener) {
|
||||||
|
var requestUrl = url;
|
||||||
|
fetch(requestUrl).then(function (response) {
|
||||||
|
return (response.arrayBuffer())
|
||||||
|
}).then((res) => {
|
||||||
|
var zipFragment = ArrayBufferToBlob(res);
|
||||||
|
var reader = new FileReader();
|
||||||
|
reader.readAsDataURL(zipFragment);
|
||||||
|
reader.onloadend = function () {
|
||||||
|
var base64data = reader.result;
|
||||||
|
callback(base64data);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function downloadFile(url, fileName, currentEXTId = "unknown", _fails = 0) {
|
||||||
|
chrome.downloads.download({
|
||||||
|
url: url,
|
||||||
|
filename: fileName,
|
||||||
|
saveAs: true
|
||||||
|
}, function () {
|
||||||
|
if (chrome.runtime.lastError) {
|
||||||
|
if (chrome.runtime.lastError.message === "Invalid filename" && _fails < 1) {
|
||||||
|
downloadFile(url, currentEXTId, currentEXTId, _fails + 1);
|
||||||
|
} else {
|
||||||
|
alert('An error occurred while trying to save ' + fileName + ':\n\n' +
|
||||||
|
chrome.runtime.lastError.message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
function onClickEvent(info, tab) {
|
||||||
|
|
||||||
|
if (info.menuItemId === "crx" || info.menuItemId === "crxmicrosoft") {
|
||||||
|
download("crx", tab)
|
||||||
|
} else if (info.menuItemId === "zip") {
|
||||||
|
download("zip", tab)
|
||||||
|
}
|
||||||
|
console.log(info)
|
||||||
|
|
||||||
|
}
|
||||||
|
chrome.contextMenus.onClicked.addListener(onClickEvent);
|
||||||
|
|
||||||
|
|
||||||
|
chrome.runtime.setUninstallURL("https://thebyteseffect.com/posts/reason-for-uninstall-crx-extractor/", null);
|
||||||
|
chrome.runtime.onInstalled.addListener(function (details) {
|
||||||
|
if (details.reason == "install") {
|
||||||
|
chrome.tabs.create({
|
||||||
|
url: "https://thebyteseffect.com/posts/crx-extractor-features/"
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
const parent = chrome.contextMenus.create({
|
||||||
|
'title': 'Download CRX for this extension',
|
||||||
|
'contexts': ['all'],
|
||||||
|
'id': "parent",
|
||||||
|
'documentUrlPatterns': ['https://chrome.google.com/webstore/detail/*', 'https://chromewebstore.google.com/detail/*']
|
||||||
|
});
|
||||||
|
chrome.contextMenus.create({
|
||||||
|
'title': 'Download CRX for this extension',
|
||||||
|
'contexts': ['all'],
|
||||||
|
id: "crx",
|
||||||
|
parentId: parent,
|
||||||
|
'documentUrlPatterns': ['https://chrome.google.com/webstore/detail/*', 'https://chromewebstore.google.com/detail/*']
|
||||||
|
});
|
||||||
|
|
||||||
|
chrome.contextMenus.create({
|
||||||
|
'title': 'Download CRX for this extension',
|
||||||
|
'contexts': ['all'],
|
||||||
|
parentId: parent,
|
||||||
|
id: "crxmicrosoft",
|
||||||
|
'documentUrlPatterns': ['https://microsoftedge.microsoft.com/addons/detail/*']
|
||||||
|
});
|
||||||
|
chrome.contextMenus.create({
|
||||||
|
'title': 'Download ZIP for this extension',
|
||||||
|
'contexts': ['all'],
|
||||||
|
id: "zip",
|
||||||
|
parentId: parent,
|
||||||
|
'documentUrlPatterns': ['https://chrome.google.com/webstore/detail/*', 'https://chromewebstore.google.com/detail/*']
|
||||||
|
});
|
||||||
|
});
|
||||||
|
chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) {
|
||||||
|
download(request.download, request.tab);
|
||||||
|
});
|
31
manifest.json
Normal file
31
manifest.json
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{
|
||||||
|
"update_url": "https://clients2.google.com/service/update2/crx",
|
||||||
|
"name": "CRX Extractor/Downloader",
|
||||||
|
"version": "1.5.8",
|
||||||
|
"manifest_version": 3,
|
||||||
|
"description": "Download CRX Files directly as crx or zip file depending upon your choice",
|
||||||
|
"permissions": [
|
||||||
|
"downloads",
|
||||||
|
"contextMenus",
|
||||||
|
"activeTab",
|
||||||
|
"downloads"
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"service_worker": "background.js"
|
||||||
|
},
|
||||||
|
"host_permissions": [
|
||||||
|
"*://clients2.google.com/service/update2/crx*",
|
||||||
|
"*://clients2.googleusercontent.com/crx/download/*"
|
||||||
|
],
|
||||||
|
"icons": {
|
||||||
|
"16": "16.png",
|
||||||
|
"48": "48.png",
|
||||||
|
"128": "128.png"
|
||||||
|
},
|
||||||
|
|
||||||
|
"action": {
|
||||||
|
"default_icon": "128.png",
|
||||||
|
"default_popup": "popup.html"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
157
popup.html
Normal file
157
popup.html
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<title>Download CRX</title>
|
||||||
|
<style>
|
||||||
|
.buttons {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid green;
|
||||||
|
width: 200px;
|
||||||
|
height: 30px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
border-radius: 5px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:nth-child(1) {
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer {
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer * {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rating>span {
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
width: 1.1em;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rating>span:hover:before,
|
||||||
|
.rating>span:hover~span:before {
|
||||||
|
content: "\2605";
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rating {
|
||||||
|
unicode-bidi: bidi-override;
|
||||||
|
direction: rtl;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rate-container {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 14px;
|
||||||
|
margin-right: 10px;
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
|
||||||
|
"Helvetica Neue", Arial, sans-serif, "Apple Color Emoji",
|
||||||
|
"Segoe UI Emoji", "Segoe UI Symbol";
|
||||||
|
}
|
||||||
|
label{
|
||||||
|
display: inline-block;
|
||||||
|
width: 200px;
|
||||||
|
height: 32px;
|
||||||
|
text-align: center;
|
||||||
|
padding-top: 6px;
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 13.333px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
position: relative;
|
||||||
|
text-align: center;
|
||||||
|
width: 50px;
|
||||||
|
height: 20px;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.loader .dot {
|
||||||
|
display: inline-block;
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
border-radius: 50%;
|
||||||
|
margin-right: 3px;
|
||||||
|
background: green;
|
||||||
|
animation: loadanim 1.3s linear infinite;
|
||||||
|
}
|
||||||
|
.loader .dot:nth-child(2) {
|
||||||
|
animation-delay: -1.1s;
|
||||||
|
}
|
||||||
|
.loader .dot:nth-child(3) {
|
||||||
|
animation-delay: -0.9s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes loadanim {
|
||||||
|
0%, 60%, 100% {
|
||||||
|
transform: initial;
|
||||||
|
}
|
||||||
|
30% {
|
||||||
|
transform: translateY(-10px);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<button class="buttons defaultBtn" id="downloadZIP" title="Download extension in zip">
|
||||||
|
Download as ZIP
|
||||||
|
</button>
|
||||||
|
<button class="buttons defaultBtn" id="downloadCRX" title="Download extension in CRX">
|
||||||
|
Download as CRX
|
||||||
|
</button>
|
||||||
|
|
||||||
|
|
||||||
|
<p style="display: none" id="info">
|
||||||
|
Please go to
|
||||||
|
<a href="https://chrome.google.com/webstore/category/extensions" target="_blank"
|
||||||
|
rel="noreferrer noopener">Chrome Extension Store</a> or <a href="https://microsoftedge.microsoft.com/addons/Microsoft-Edge-Extensions-Home" target="_blank"
|
||||||
|
rel="noreferrer noopener">Microsoft Addon Store</a>
|
||||||
|
and go to extension page to download chrome extensions.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="buttons">
|
||||||
|
<label for="convertCRXToZip">Convert from CRX to ZIP file</label>
|
||||||
|
<input id="convertCRXToZip" title="Download extension in CRX" accept=".crx" type="file" style="display: none;height:100%"> </input>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div class="loader" id='loader'>
|
||||||
|
<span class="dot"></span>
|
||||||
|
<span class="dot"></span>
|
||||||
|
<span class="dot"></span>
|
||||||
|
</div>
|
||||||
|
<button class="buttons" id="downloadCRXToZip" title="Download ZIP File" style="display: none;">
|
||||||
|
Download ZIP File
|
||||||
|
</button>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="container footer">
|
||||||
|
<div class="rate-container">
|
||||||
|
<span>Rate Us</span>
|
||||||
|
<div class="rating" id="rating">
|
||||||
|
<span>☆</span><span>☆</span><span>☆</span><span>☆</span><span>☆</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
<script src="popup.js"></script>
|
||||||
|
|
||||||
|
</html>
|
96
popup.js
Normal file
96
popup.js
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
|
||||||
|
|
||||||
|
let chromeURLPattern = /^https?:\/\/chrome.google.com\/webstore\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
let chromeNewURLPattern = /^https?:\/\/chromewebstore.google.com\/detail\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
let microsoftURLPattern = /^https?:\/\/microsoftedge.microsoft.com\/addons\/detail\/.+?\/([a-z]{32})(?=[\/#?]|$)/;
|
||||||
|
|
||||||
|
|
||||||
|
function ready() {
|
||||||
|
document.getElementById('downloadZIP').onclick = async function () {
|
||||||
|
let queryOptions = { active: true, currentWindow: true };
|
||||||
|
let [tab] = await chrome.tabs.query(queryOptions);
|
||||||
|
chrome.runtime.sendMessage({download:"zip", tab: tab});
|
||||||
|
};
|
||||||
|
document.getElementById('downloadCRX').onclick = async function () {
|
||||||
|
let queryOptions = { active: true, currentWindow: true };
|
||||||
|
let [tab] = await chrome.tabs.query(queryOptions);
|
||||||
|
chrome.runtime.sendMessage({download:"crx",tab:tab});
|
||||||
|
};
|
||||||
|
|
||||||
|
document.getElementById("convertCRXToZip").onchange = function (files) {
|
||||||
|
setTimeout(() => {
|
||||||
|
document.getElementById("loader").style.display = "none";
|
||||||
|
document.getElementById("downloadCRXToZip").style.display = "block";
|
||||||
|
}, 2000);
|
||||||
|
document.getElementById("loader").style.display = "block";
|
||||||
|
document.getElementById("downloadCRXToZip").style.display = "none";
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
document.getElementById("downloadCRXToZip").onclick = function () {
|
||||||
|
var file = document.getElementById("convertCRXToZip").files[0];
|
||||||
|
|
||||||
|
var reader = new FileReader();
|
||||||
|
reader.onload = function (e) {
|
||||||
|
var data = reader.result;
|
||||||
|
var buf = new Uint8Array(data);
|
||||||
|
var publicKeyLength, signatureLength, header, zipStartOffset;
|
||||||
|
if (buf[4] === 2) {
|
||||||
|
header = 16;
|
||||||
|
publicKeyLength = 0 + buf[8] + (buf[9] << 8) + (buf[10] << 16) + (buf[11] << 24);
|
||||||
|
signatureLength = 0 + buf[12] + (buf[13] << 8) + (buf[14] << 16) + (buf[15] << 24);
|
||||||
|
zipStartOffset = header + publicKeyLength + signatureLength;
|
||||||
|
} else {
|
||||||
|
publicKeyLength = 0 + buf[8] + (buf[9] << 8) + (buf[10] << 16) + (buf[11] << 24 >>> 0);
|
||||||
|
zipStartOffset = 12 + publicKeyLength;
|
||||||
|
}
|
||||||
|
// 16 = Magic number (4), CRX format version (4), lengths (2x4)
|
||||||
|
var zip = buf.slice(zipStartOffset, buf.length);
|
||||||
|
var fileName = file.name.replace(".crx", ".zip")
|
||||||
|
var blob = new Blob([zip], { type: "application/octet-stream" });
|
||||||
|
var url = window.URL.createObjectURL(blob);
|
||||||
|
chrome.downloads.download({
|
||||||
|
url: url,
|
||||||
|
filename: fileName,
|
||||||
|
saveAs: true
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
reader.readAsArrayBuffer(file);
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
document.getElementById('rating').onclick = function () {
|
||||||
|
window.open("https://chrome.google.com/webstore/detail/crx-extractordownloader/ajkhmmldknmfjnmeedkbkkojgobmljda/reviews");
|
||||||
|
};
|
||||||
|
chrome.tabs.query({ active: true, currentWindow: true }, function (tab) {
|
||||||
|
tab = tab[0];
|
||||||
|
var id = chromeURLPattern.exec(tab.url);
|
||||||
|
|
||||||
|
if (!id) {
|
||||||
|
id = chromeNewURLPattern.exec(tab.url);
|
||||||
|
}
|
||||||
|
var edgeId = microsoftURLPattern.exec(tab.url);
|
||||||
|
|
||||||
|
|
||||||
|
document.getElementById("info").style.display = "block";
|
||||||
|
var elements = document.getElementsByClassName('defaultBtn');
|
||||||
|
var length = elements.length;
|
||||||
|
for (var i = 0; i < length; i++) {
|
||||||
|
elements[i].style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (edgeId !== null && edgeId[1] !== null) {
|
||||||
|
document.getElementById("info").style.display = "none";
|
||||||
|
document.getElementById('downloadCRX').style.display = 'block';
|
||||||
|
} else if (id !== null && id[1] !== null) {
|
||||||
|
document.getElementById("info").style.display = "none";
|
||||||
|
var elements = document.getElementsByClassName('defaultBtn');
|
||||||
|
var length = elements.length;
|
||||||
|
for (var i = 0; i < length; i++) {
|
||||||
|
elements[i].style.display = 'block';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
ready();
|
Loading…
x
Reference in New Issue
Block a user