支持PC 用于统计购物车域名列表,方便统计计算
// ==UserScript==
// @name Namecheap 购物车域名列表
// @icon 
// @match *://www.namecheap.com/shoppingcart/*
// @grant GM_addStyle
// @version 2025.7.25
// @author ayersltd
// @description 支持PC 用于统计购物车域名列表,方便统计计算
// @license GPL-3.0-only
// @require https://cdnjs.cloudflare.com/ajax/libs/fuse.js/6.6.2/fuse.min.js
// @namespace https://greasyfork.org/users/1497923
// ==/UserScript==
(function() {
// 添加自定义样式
GM_addStyle(`
#domainListPanel {
position: fixed;
top: 50px;
right: 50px;
background: #fff;
border: 1px solid #e1e1e1;
border-radius: 8px;
box-shadow: 0 8px 30px rgba(0,0,0,0.12);
z-index: 99999;
width: 380px;
max-height: 85vh;
display: flex;
flex-direction: column;
font-family: 'Segoe UI', system-ui, sans-serif;
}
.panel-header {
padding: 16px;
background: #f8f8f8;
border-bottom: 1px solid #eee;
display: flex;
justify-content: space-between;
align-items: center;
border-radius: 8px 8px 0 0;
}
.panel-title {
font-weight: 600;
font-size: 16px;
color: #333;
margin: 0;
}
.panel-close {
background: none;
border: none;
font-size: 18px;
cursor: pointer;
color: #888;
}
.panel-actions {
padding: 12px 16px;
display: flex;
gap: 8px;
border-bottom: 1px solid #eee;
}
.search-box {
flex: 1;
padding: 8px 12px;
border: 1px solid #ddd;
border-radius: 4px;
font-size: 14px;
}
.action-btn {
padding: 6px 12px;
background: #f2f2f2;
border: 1px solid #ddd;
border-radius: 4px;
cursor: pointer;
font-size: 13px;
}
.action-btn:hover {
background: #e8e8e8;
}
.panel-content {
flex: 1;
overflow: auto;
padding: 0;
}
.domain-list {
list-style: none;
padding: 0;
margin: 0;
}
.domain-item {
padding: 12px 16px;
border-bottom: 1px solid #f5f5f5;
font-size: 14px;
display: flex;
justify-content: space-between;
align-items: center;
}
.domain-name {
word-break: break-all;
padding-right: 10px;
}
.domain-count {
background: #f0f5ff;
color: #2f5bea;
border-radius: 12px;
padding: 3px 8px;
font-size: 12px;
font-weight: 500;
min-width: 60px;
text-align: center;
}
.panel-footer {
padding: 12px 16px;
text-align: center;
background: #f8f8f8;
border-top: 1px solid #eee;
font-size: 13px;
color: #666;
border-radius: 0 0 8px 8px;
}
.view-toggle {
display: flex;
gap: 8px;
margin-bottom: 12px;
justify-content: center;
}
.view-option {
padding: 4px 12px;
border-radius: 15px;
background: #f0f0f0;
cursor: pointer;
font-size: 12px;
}
.view-option.active {
background: #e1ecff;
color: #2f5bea;
font-weight: 500;
}
.section-title {
padding: 10px 16px;
background: #f8f8f8;
font-weight: 600;
font-size: 13px;
color: #666;
border-bottom: 1px solid #eee;
position: sticky;
top: 0;
}
.tld-group {
margin-bottom: 15px;
}
`);
// 主功能
function initDomainList() {
// 创建面板容器
const panel = document.createElement('div');
panel.id = 'domainListPanel';
// 添加头部
panel.innerHTML = `
<div class="panel-header">
<h3 class="panel-title">域名购物车</h3>
<button class="panel-close">×</button>
</div>
<div class="panel-actions">
<input type="text" class="search-box" placeholder="搜索域名..." id="domainSearch">
<button class="action-btn" id="copyAllBtn">复制所有</button>
</div>
<div class="view-toggle">
<span class="view-option active" data-view="list">列表视图</span>
<span class="view-option" data-view="tld">按后缀分组</span>
</div>
<div class="panel-content">
<ul class="domain-list" id="domainList"></ul>
</div>
<div class="panel-footer">
<div class="domain-count">共计: <span id="totalCount">0</span> 个域名</div>
</div>
`;
document.body.appendChild(panel);
// 获取元素引用
const domainList = document.getElementById('domainList');
const searchBox = document.getElementById('domainSearch');
const copyAllBtn = document.getElementById('copyAllBtn');
const totalCount = document.getElementById('totalCount');
const closeBtn = document.querySelector('.panel-close');
const viewOptions = document.querySelectorAll('.view-option');
let allDomains = [];
let currentView = 'list';
// 关闭面板
closeBtn.addEventListener('click', () => {
panel.style.display = 'none';
});
// 视图切换
viewOptions.forEach(option => {
option.addEventListener('click', () => {
viewOptions.forEach(o => o.classList.remove('active'));
option.classList.add('active');
currentView = option.dataset.view;
updateDomainListView(allDomains);
});
});
// 搜索功能
searchBox.addEventListener('input', (e) => {
const keyword = e.target.value.trim().toLowerCase();
if (!keyword) {
updateDomainListView(allDomains);
return;
}
const filteredDomains = allDomains.filter(domain =>
domain.toLowerCase().includes(keyword)
);
updateDomainListView(filteredDomains);
});
// 复制所有域名
copyAllBtn.addEventListener('click', () => {
if (allDomains.length === 0) return;
const domainsText = allDomains.join('\n');
navigator.clipboard.writeText(domainsText).then(() => {
showNotification(`已复制 ${allDomains.length} 个域名到剪贴板`);
});
});
// 拖拽功能
initPanelDrag(panel);
// 初始加载域名
setTimeout(loadDomainList, 1500);
// 加载域名列表
function loadDomainList() {
const domains = [];
// 使用XPath查找所有域名文本节点
const xpath = "//div[@data-e2e-id='sc-product-container']//p[@data-e2e-id='sc-product-sub-title']/text()";
const result = document.evaluate(
xpath,
document,
null,
XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
null
);
// 提取域名文本
for (let i = 0; i < result.snapshotLength; i++) {
const node = result.snapshotItem(i);
const domain = node.nodeValue.trim();
if (domain) domains.push(domain);
}
if (domains.length === 0) {
showNotification("未找到有效域名");
return;
}
allDomains = domains;
totalCount.textContent = domains.length;
updateDomainListView(domains);
}
// 更新域名列表视图
function updateDomainListView(domains) {
domainList.innerHTML = '';
if (domains.length === 0) {
domainList.innerHTML = '<div class="domain-item">未找到匹配域名</div>';
return;
}
if (currentView === 'tld') {
renderTLDGroupView(domains);
} else {
renderListView(domains);
}
}
// 渲染列表视图
function renderListView(domains) {
domains.forEach((domain, index) => {
const li = document.createElement('li');
li.className = 'domain-item';
li.innerHTML = `
<span class="domain-name">${index + 1}. ${domain}</span>
`;
domainList.appendChild(li);
});
}
// 渲染TLD分组视图
function renderTLDGroupView(domains) {
// 按TLD分组
const groups = {};
domains.forEach(domain => {
const tld = domain.split('.').pop();
if (!groups[tld]) groups[tld] = [];
groups[tld].push(domain);
});
// 渲染分组视图
Object.entries(groups).forEach(([tld, domainListInGroup]) => {
const section = document.createElement('div');
section.className = 'tld-group';
section.innerHTML = `
<div class="section-title">.${tld} (${domainListInGroup.length})</div>
`;
const ul = document.createElement('ul');
ul.className = 'domain-list';
domainListInGroup.forEach((domain, index) => {
const li = document.createElement('li');
li.className = 'domain-item';
li.innerHTML = `<span class="domain-name">${index + 1}. ${domain}</span>`;
ul.appendChild(li);
});
section.appendChild(ul);
domainList.appendChild(section);
});
}
}
// 拖拽功能
function initPanelDrag(panel) {
let isDragging = false;
let offsetX, offsetY;
const header = panel.querySelector('.panel-header');
header.addEventListener('mousedown', (e) => {
isDragging = true;
offsetX = e.clientX - panel.getBoundingClientRect().left;
offsetY = e.clientY - panel.getBoundingClientRect().top;
});
document.addEventListener('mousemove', (e) => {
if (isDragging) {
panel.style.left = `${e.clientX - offsetX}px`;
panel.style.top = `${e.clientY - offsetY}px`;
}
});
document.addEventListener('mouseup', () => {
isDragging = false;
});
}
// 显示通知
function showNotification(message) {
const notification = document.createElement('div');
notification.textContent = message;
notification.style.cssText = `
position: fixed;
top: 20px;
left: 50%;
transform: translateX(-50%);
background: #4caf50;
color: white;
padding: 12px 24px;
border-radius: 4px;
z-index: 10000;
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
animation: fadeInOut 3s ease;
`;
document.body.appendChild(notification);
setTimeout(() => {
notification.remove();
}, 3000);
}
// 延迟初始化,确保页面加载完成
if (document.readyState === 'complete') {
initDomainList();
} else {
window.addEventListener('load', initDomainList);
}
})();