diff --git a/examples/image.png b/examples/image.png deleted file mode 100644 index 5d81986..0000000 Binary files a/examples/image.png and /dev/null differ diff --git a/icons/aqua.svg b/icons/aqua.svg deleted file mode 100644 index f5c40b9..0000000 --- a/icons/aqua.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/clion.svg b/icons/clion.svg deleted file mode 100644 index 3440b58..0000000 --- a/icons/clion.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/datagrip.svg b/icons/datagrip.svg deleted file mode 100644 index 9cb67c5..0000000 --- a/icons/datagrip.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/dataspell.svg b/icons/dataspell.svg deleted file mode 100644 index 59a22af..0000000 --- a/icons/dataspell.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/dotcover.svg b/icons/dotcover.svg deleted file mode 100644 index e5c7882..0000000 --- a/icons/dotcover.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/dotmemory.svg b/icons/dotmemory.svg deleted file mode 100644 index b92e056..0000000 --- a/icons/dotmemory.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/dottrace.svg b/icons/dottrace.svg deleted file mode 100644 index 436f00c..0000000 --- a/icons/dottrace.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/goland.svg b/icons/goland.svg deleted file mode 100644 index dacd3f7..0000000 --- a/icons/goland.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/intellij-idea.svg b/icons/intellij-idea.svg deleted file mode 100644 index d6de682..0000000 --- a/icons/intellij-idea.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/phpstorm.svg b/icons/phpstorm.svg deleted file mode 100644 index 588e24a..0000000 --- a/icons/phpstorm.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/pycharm.svg b/icons/pycharm.svg deleted file mode 100644 index 0956bbc..0000000 --- a/icons/pycharm.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/rider.svg b/icons/rider.svg deleted file mode 100644 index e84f77b..0000000 --- a/icons/rider.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/rubymine.svg b/icons/rubymine.svg deleted file mode 100644 index 9343cce..0000000 --- a/icons/rubymine.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/rustrover.svg b/icons/rustrover.svg deleted file mode 100644 index 25660a6..0000000 --- a/icons/rustrover.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/icons/webstorm.svg b/icons/webstorm.svg deleted file mode 100644 index 4920237..0000000 --- a/icons/webstorm.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/script/jetbra.user.js b/script/jetbra.user.js deleted file mode 100644 index 725ed6b..0000000 --- a/script/jetbra.user.js +++ /dev/null @@ -1,323 +0,0 @@ -// ==UserScript== -// @name JetBra -// @namespace https://github.com/novice88/jetbra -// @version 5.0 -// @license MIT -// @description Add a button on the plugin homepage and click to get the plugin activation code -// @author novice.li -// @match https://plugins.jetbrains.com/* -// @grant GM_setClipboard -// @grant GM_addStyle -// @grant window.onurlchange -// @downloadURL https://update.greasyfork.org/scripts/480799/JetBra.user.js -// @updateURL https://update.greasyfork.org/scripts/480799/JetBra.meta.js -// ==/UserScript== - -const pemEncodedKey = `-----BEGIN PRIVATE KEY----- -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 -17M9lCizO9o9dONdHsHfNQi6y9Jcnw== ------END PRIVATE KEY-----`; - -const pemEncodedCrt = `-----BEGIN CERTIFICATE----- -MIIEtTCCAp2gAwIBAgIUDyuccmylba71lZQAQic5TJiAhwwwDQYJKoZIhvcNAQEL -BQAwGDEWMBQGA1UEAwwNSmV0UHJvZmlsZSBDQTAeFw0yMzA5MjkxNDA2MTJaFw0z -MzA5MjcxNDA2MTJaMBExDzANBgNVBAMMBk5vdmljZTCCAiIwDQYJKoZIhvcNAQEB -BQADggIPADCCAgoCggIBALenqcGP2ZxGkYqmKA9c4Hzf8+YD1smvmOxKjd+bmTLr -utM/hXv1cj1rW3/lqyDtdDk7K6W8/TDq1CRrEt+Do6l30DxhAiC34aH8DmGwgq77 -xEoLimvH5LpePxflF+tbB1RZtFgFDOIYLdSQaKFH2JDgVKxhLiV3S6jniPhkCtWW -rTs+E6vq4N15Bm3NnM5AJILqjtUbOjNfaxVq6RrOoTc0R3Fqqo6yvxo/+JYa2UnH -IC+r2dbKuDLMUrtgnydEUdJNX0zH9FtcdELvr48uc9mY038TWUsZUK1pnQbxA2bP -yA4qnYJ9IvUgO6LtLXvGFm137YQMS1N41AHDBOrwoNI8UoDX+qI3rM96biFOFvn7 -Edky7rByzybt3H+zxdojfjvpL1E0NO98BT9zfufHAaAxZtlmDOu5LDJe3CGurnyR -MRExbtc+Qjl1mUh6tG4lakAwdsoxry0GdG72yaYyb9it53kaFks/T/s7Z7bRJzVF -zQDV1Y4bzUtk43vKm2vztBVlQkBkZY5f2Jbe5Ig3b8swQzBnOT0mrL5SPUhwmQ6I -xkEWztj55OEujBMmRr92oESuq9ZYMaeLidKWVR3/++HA8BRZaRGEKtSHZCbFEFdi -hDxxJv9Xh6NuT/ewJ6HYp+0NQpFnUnJ72n8wV+tudpam7aKcdzVmz7cNwOhG2Ls7 -AgMBAAEwDQYJKoZIhvcNAQELBQADggIBAIdeaQfKni7tXtcywC3zJvGzaaj242pS -WB1y40HW8jub0uHjTLsBPX27iA/5rb+rNXtUWX/f2K+DU4IgaIiiHhkDrMsw7piv -azqwA9h7/uA0A5nepmTYf/HY4W6P2stbeqInNsFRZXS7Jg4Q5LgEtHKo/H8USjtV -w9apmE3BCElkXRuelXMsSllpR/JEVv/8NPLmnHSY02q4KMVW2ozXtaAxSYQmZswy -P1YnBcnRukoI4igobpcKQXwGoQCIUlec8LbFXYM9V2eNCwgABqd4r67m7QJq31Y/ -1TJysQdMH+hoPFy9rqNCxSq3ptpuzcYAk6qVf58PrrYH/6bHwiYPAayvvdzNPOhM -9OCwomfcazhK3y7HyS8aBLntTQYFf7vYzZxPMDybYTvJM+ClCNnVD7Q9fttIJ6eM -XFsXb8YK1uGNjQW8Y4WHk1MCHuD9ZumWu/CtAhBn6tllTQWwNMaPOQvKf1kr1Kt5 -etrONY+B6O+Oi75SZbDuGz7PIF9nMPy4WB/8XgKdVFtKJ7/zLIPHgY8IKgbx/VTz -6uBhYo8wOf3xzzweMnn06UcfV3JGNvtMuV4vlkZNNxXeifsgzHugCvJX0nybhfBh -fIqVyfK6t0eKJqrvp54XFEtJGR+lf3pBfTdcOI6QFEPKGZKoQz8Ck+BC/WBDtbjc -/uYKczZ8DKZu ------END CERTIFICATE-----`; - -let translations = { - "en": { - "permanentLicense": "Get Permanent License", - "twoYearsLicense": "Get Two Years License", - "notMarketPlugin": "This plugin is not a market-paid plugin!", - "copiedToClipboard": "License key has been copied to your clipboard." - }, - "zh": { - "permanentLicense": "获取永久激活码", - "twoYearsLicense": "获取两年激活码", - "notMarketPlugin": "该插件不是市场付费插件!", - "copiedToClipboard": "激活码已经拷贝到你的剪切板" - }, - "ru": { - "permanentLicense": "Получить постоянную лицензию", - "twoYearsLicense": "Получить лицензию на два года", - "notMarketPlugin": "Этот плагин не является платным на рынке!", - "copiedToClipboard": "Лицензионный ключ скопирован в буфер обмена." - }, - "es": { - "permanentLicense": "Obtener licencia permanente", - "twoYearsLicense": "Obtener licencia por dos años", - "notMarketPlugin": "¡Este plugin no es un plugin de pago en el mercado!", - "copiedToClipboard": "La clave de licencia se ha copiado al portapapeles." - }, - "fr": { - "permanentLicense": "Obtenir une licence permanente", - "twoYearsLicense": "Obtenir une licence de deux ans", - "notMarketPlugin": "Ce plugin n'est pas un plugin payant du marché!", - "copiedToClipboard": "La clé de licence a été copiée dans votre presse-papiers." - }, - "de": { - "permanentLicense": "Dauerlizenz erhalten", - "twoYearsLicense": "Zwei-Jahres-Lizenz erhalten", - "notMarketPlugin": "Dieses Plugin ist kein kostenpflichtiges Markt-Plugin!", - "copiedToClipboard": "Der Lizenzschlüssel wurde in Ihre Zwischenablage kopiert." - }, - "pt": { - "permanentLicense": "Obter licença permanente", - "twoYearsLicense": "Obter licença de dois anos", - "notMarketPlugin": "Este plugin não é um plugin pago do mercado!", - "copiedToClipboard": "A chave de licença foi copiada para a sua área de transferência." - }, - "ja": { - "permanentLicense": "永久ライセンスを取得", - "twoYearsLicense": "2年ライセンスを取得", - "notMarketPlugin": "このプラグインは市場での有料プラグインではありません!", - "copiedToClipboard": "ライセンスキーがクリップボードにコピーされました。" - }, - "ko": { - "permanentLicense": "영구 라이선스 받기", - "twoYearsLicense": "2년 라이선스 받기", - "notMarketPlugin": "이 플러그인은 마켓 유료 플러그인이 아닙니다!", - "copiedToClipboard": "라이선스 키가 클립보드에 복사되었습니다." - }, - "it": { - "permanentLicense": "Ottieni licenza permanente", - "twoYearsLicense": "Ottieni licenza di due anni", - "notMarketPlugin": "Questo plugin non è un plugin a pagamento sul mercato!", - "copiedToClipboard": "La chiave di licenza è stata copiata negli appunti." - }, - "ar": { - "permanentLicense": "الحصول على رخصة دائمة", - "twoYearsLicense": "الحصول على رخصة لمدة عامين", - "notMarketPlugin": "هذا المكون الإضافي ليس مكونًا مدفوعًا في السوق!", - "copiedToClipboard": "تم نسخ مفتاح الترخيص إلى الحافظة الخاصة بك." - } -}; - -function getUserLanguage() { - const userLang = navigator.language || navigator.languages[0]; - const langCode = userLang.split('-')[0]; - - if (translations[langCode]) { - return langCode; - } - - for (let lang in translations) { - if (userLang.startsWith(lang)) { - return lang; - } - } - - return 'en'; -} - -function getTranslatedCulture(key) { - const useralang = getUserLanguage(); - const userTranslations = translations[useralang] || {}; - return userTranslations[key] || translations['en'][key] || key; -} - -function injectStyles() { - GM_addStyle(` - .jetbra-button { - background-color: #04AA6D; border: none; color: white; padding: 8px 24px; - text-align: center; text-decoration: none; display: inline-block; - border-radius: 16px; box-shadow: 0 8px 16px 0 rgba(0,0,0,0.2), 0 6px 20px 0 rgba(0,0,0,0.19); - transition-duration: 0.4s; - margin-left:5px - } - .jetbra-button:hover { background-color: #057e47; color: white; } - `); -} - -async function findElementWithRetry(cssSelector) { - const maxAttempts = 50; - for (let attempts = 0; attempts < maxAttempts; attempts++) { - const element = document.querySelector(cssSelector); - if (element) { - return element; - } - await new Promise(resolve => setTimeout(resolve, 100)); - } - throw new Error(`Element with selector '${cssSelector}' not found after ${maxAttempts} attempts.`); -} - -function pem2base64(pem) { - return pem.split('\n').reduce((base64, line) => line.includes("--") ? base64 : base64 + line, ''); -} - -function arrayBufferToBase64(buffer) { - return btoa([...new Uint8Array(buffer)].map(b => String.fromCharCode(b)).join('')); -} - -function base64ToArrayBuffer(base64) { - return Uint8Array.from(atob(base64), c => c.charCodeAt(0)).buffer; -} - -function genLicenseId() { - const CHARSET = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; - return Array.from({length: 10}, () => { - let idx = Math.floor(Math.random() * CHARSET.length); - return CHARSET[idx]; - }).join(''); -} - -function buildLicensePartJson(productCode, licenseId, twoYears) { - let formattedDateTwoYears = '9999-01-01'; - if (twoYears) { - let futureDateTwoYears = new Date(); - futureDateTwoYears.setFullYear(futureDateTwoYears.getFullYear() + 2); - formattedDateTwoYears = futureDateTwoYears.toISOString().split('T')[0]; - } - - let fallbackDate = formattedDateTwoYears; - let paidUpTo = formattedDateTwoYears; - - return JSON.stringify({ - "licenseId": licenseId, - "licenseeName": "loch", - "assigneeName": "pidor", - "assigneeEmail": "", - "licenseRestriction": "", - "checkConcurrentUse": false, - "products": [{ - code: productCode, - fallbackDate: fallbackDate, - paidUpTo: paidUpTo - }], - "metadata": "0120230102PPAA013009", - "hash": "41472961/0:1563609451", - "gracePeriodDays": 900, - "autoProlongated": true, - "isAutoProlongated": true - }); -} - -async function addButton() { - injectStyles(); - - let url = window.location.href - if (!url.startsWith('https://plugins.jetbrains.com/plugin/')) { - return; - } - - let pluginId = url.split('/')[4].split('-')[0] - console.log('pluginId: ' + pluginId); - - let pluginDetail = await fetch('https://plugins.jetbrains.com/api/plugins/' + pluginId).then(r => r.json()); - - const parentElement = await findElementWithRetry('.plugin-header__controls-panel > div:first-child'); - - if (parentElement.querySelector('.jetbra-button')) { - return; - } - let newElement = document.createElement('div'); - newElement.classList.toggle('wt-col-inline'); - newElement.innerHTML = - `
-
- - -
-
`; - - parentElement.appendChild(newElement) - - newElement.addEventListener('click', async (e) => { - let productCode = pluginDetail?.purchaseInfo?.productCode; - if (productCode === undefined) { - window.alert(`${getTranslatedCulture('notMarketPlugin')}`); - return; - } - let licenseId = genLicenseId() - let licensePartJson = buildLicensePartJson(productCode, licenseId,e.target.id === "twoYearsLicense") - - let privateKey = await window.crypto.subtle.importKey("pkcs8", base64ToArrayBuffer(pem2base64(pemEncodedKey)), { - name: "RSASSA-PKCS1-v1_5", hash: "SHA-1", - }, true, ["sign"]); - - let licensePartBase64 = btoa(unescape(encodeURIComponent(licensePartJson))); - let sigResultsBase64 = arrayBufferToBase64(await window.crypto.subtle.sign("RSASSA-PKCS1-v1_5", privateKey, new TextEncoder().encode(licensePartJson))); - let cert_base64 = pem2base64(pemEncodedCrt); - - GM_setClipboard(`${licenseId}-${licensePartBase64}-${sigResultsBase64}-${cert_base64}`, 'text'); - window.alert(`${getTranslatedCulture('copiedToClipboard')}`); - }) -} - -addButton(); -if (window.onurlchange === null) { - window.addEventListener('urlchange', (ignore) => { - addButton(); - }); -} \ No newline at end of file