mirror of
https://github.com/novitechie/jetbra
synced 2024-12-22 06:15:45 +00:00
122 lines
4.1 KiB
JavaScript
122 lines
4.1 KiB
JavaScript
// ==UserScript==
|
||
// @name JetBra
|
||
// @namespace https://github.com/novice88/jetbra
|
||
// @version 3.2
|
||
// @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 GM_xmlhttpRequest
|
||
// @grant window.onurlchange
|
||
// @connect noviceli.win
|
||
// @connect self
|
||
// @connect localhost
|
||
// ==/UserScript==
|
||
|
||
|
||
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.`);
|
||
}
|
||
|
||
let addButton = async function () {
|
||
console.log('JetBra is running');
|
||
'use strict';
|
||
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;
|
||
}
|
||
|
||
.jetbra-button:hover {
|
||
background-color: #057e47;
|
||
color: white;
|
||
}
|
||
`);
|
||
const backendBaseUrl = 'https://jetbra.noviceli.win'
|
||
|
||
// 获取插件的 id
|
||
// 如果当前url满足 https://plugins.jetbrains.com/plugin/<pluginId>-xxx 的格式,就直接从url中获取
|
||
let url = window.location.href
|
||
if (!url.startsWith('https://plugins.jetbrains.com/plugin/')) {
|
||
return;
|
||
}
|
||
// 提取 pluginId
|
||
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');
|
||
|
||
// 如果 parentElement 的孩子中已经有了按钮,就不再添加
|
||
if (parentElement.querySelector('.jetbra-button')) {
|
||
return;
|
||
}
|
||
let newElement = document.createElement('div');
|
||
newElement.classList.toggle('wt-col-inline');
|
||
newElement.innerHTML = `<button class="jetbra-button" type="button">CLICK TO GENERATE ACTIVATION CODE</button>`;
|
||
parentElement.appendChild(newElement)
|
||
|
||
newElement.addEventListener('click', async () => {
|
||
if (pluginDetail.purchaseInfo === undefined) {
|
||
window.alert('This plugin is not a paid plugin in the market');
|
||
return;
|
||
}
|
||
let data = {
|
||
"licenseeName": "reborn",
|
||
"assigneeName": "reborn",
|
||
"assigneeEmail": "",
|
||
"licenseRestriction": "",
|
||
"checkConcurrentUse": false,
|
||
"products": [{
|
||
"code": pluginDetail.purchaseInfo.productCode,
|
||
"fallbackDate": "2026-12-30",
|
||
"paidUpTo": "2026-12-30",
|
||
"extended": false
|
||
}],
|
||
"metadata": "0120230102PPAA013009",
|
||
"hash": "41472961/0:1563609451",
|
||
"gracePeriodDays": 7,
|
||
"autoProlongated": true,
|
||
"isAutoProlongated": true
|
||
}
|
||
GM_xmlhttpRequest({
|
||
method: 'POST',
|
||
url: backendBaseUrl + '/generateLicense',
|
||
headers: {
|
||
'Content-Type': 'application/json'
|
||
},
|
||
data: JSON.stringify(data),
|
||
onload: function (response) {
|
||
let license = JSON.parse(response.responseText).license
|
||
GM_setClipboard(license, 'text');
|
||
window.alert('The activation code has been copied to your clipboard');
|
||
}
|
||
});
|
||
})
|
||
};
|
||
|
||
addButton();
|
||
if (window.onurlchange === null) {
|
||
window.addEventListener('urlchange', (info) => {
|
||
addButton();
|
||
});
|
||
} |