您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Improvement for the Village Page
// ==UserScript== // @name BvS Village Page Revamp // @namespace North // @description Improvement for the Village Page // @include http*://*animecubed.com/billy/bvs/village.html // @include http*://*animecubedgaming.com/billy/bvs/village.html // @version 1.5 // @history 1.5 New domain - animecubedgaming.com - Channel28 // @history 1.4 Now https compatible (Updated by Channel28) // @history 1.3 Moved Candyween to Areas column // @history 1.2 A couple of changes based on feedback // @history 1.1 Interchanged columns for the Village and the Admin tabs // @history 1.0 Initial release // @grant none // ==/UserScript== function addStyle(css) { var head = document.getElementsByTagName('head')[0]; var style = document.createElement('style'); style.type = 'text/css'; style.textContent = css; head.appendChild(style); } function hasClass(element, className) { return element.className.match(new RegExp('(\\s|^)' + className + '(\\s|$)')); } function addClass(element, className) { if (!hasClass(element, className)) { element.className += " " + className; } } function removeClass(element, className) { if (hasClass(element, className)) { var reg = new RegExp('(\\s|^)' + className + '(\\s|$)'); element.className = element.className.replace(reg, ' '); } } function tabsPanel(tabsData) { var tabChangeCallbacks = []; var tabsContainer = document.createElement('div'); var contentContainer = document.createElement('div'); var container = document.createElement('div'); container.appendChild(tabsContainer); container.appendChild(contentContainer); addClass(tabsContainer, 'tab-panel'); var selectTab = function(tabData) { for (var i = 0; i < tabsData.length; i++) { removeClass(tabsData[i].tab, 'active-tab'); removeClass(tabsData[i].tab, 'hover-inactive-tab'); tabsData[i].content.style.display = 'none'; tabsData[i].hide && tabsData[i].hide(); } addClass(tabData.tab, 'active-tab'); tabData.content.style.display = ''; tabData.show && tabData.show(); for (i = 0; i < tabChangeCallbacks.length; i++) { tabChangeCallbacks[i](tabData.index); } }; var createTab = function(tabData) { var tab = document.createElement('div'); addClass(tab, 'inactive-tab'); tab.appendChild(document.createTextNode(tabData.text)); tabsContainer.appendChild(tab); tabData.tab = tab; if (tabData.content) { contentContainer.appendChild(tabData.content); tabData.content.style.display = 'none'; } tabData.hide && tabData.hide(); var onTabClick = function() { selectTab(tabData); }; var onTabMouseEnter = function() { if (!hasClass(tab, 'active-tab')) { addClass(tab, 'hover-inactive-tab'); } }; var onTabMouseLeave = function() { removeClass(tab, 'hover-inactive-tab'); }; tab.addEventListener('click', onTabClick, false); tab.addEventListener('mouseenter', onTabMouseEnter, false); tab.addEventListener('mouseleave', onTabMouseLeave, false); }; for (var i = 0; i < tabsData.length; i++) { tabsData[i].index = i; createTab(tabsData[i]); } addStyle([ ".tab-panel {height: 24px; border-bottom: 4px solid #2647a0;}", ".tab-panel div {float: left; font-size: 12px; cursor: pointer; position: relative; padding-left: 5px; padding-right: 5px; line-height: 22px; border-top-left-radius: 1ex; border-top-right-radius: 1ex; border: 1px solid #a3a3a3; border-bottom-width: 0; bottom: 1px;}", ".tab-panel .inactive-tab {background: #d8d8d8 url() top left repeat-x; margin-top: 2px;}", ".tab-panel .active-tab {background: #2647a0 url() top left repeat-x; border: 1px solid #243356; border-bottom-width: 0; line-height: 24px; color: white; cursor: default;}", ".tab-panel .hover-inactive-tab {background: #bfdaff url() top left repeat-x; margin-top: 2px;}" ].join("\n")); return { container: container, selectTab: function(tabIndex) { if (tabsData.length > tabIndex) { selectTab(tabsData[tabIndex]); } }, onTabChange: function(callback) { tabChangeCallbacks.push(callback); } }; } function getBvsDate() { var bvsOffset = -5; var d = new Date(); var utc = d.getTime() + (d.getTimezoneOffset() * 60000); return new Date(utc + (3600000 * bvsOffset)); } function getLastBvsDayrollDate() { var bvsDate = getBvsDate(); var hours = bvsDate.getHours(); var hoursOffsetToDayroll = hours < 5 ? (19 + hours) : (hours - 5); bvsDate.setMinutes(10); return new Date(bvsDate.getTime() - hoursOffsetToDayroll * 3600000); } function markTodayElement(element) { var header = element.querySelector('b'); var result = /(\d{1,2})\/(\d{1,2}) \(\w{3} - (\d{1,2}):(\d{1,2})/.exec(header.textContent); if (result && result.length === 5) { var date = new Date(); var month = result[1] - 1; var day = result[2]; var hour = result[3]; var minute = result[4]; var year = date.getFullYear(); if (date.getMonth() === 0 && month === 11) { year--; } date = new Date(year, month, day, hour, minute); var dayroll = getLastBvsDayrollDate(); if (date > dayroll) { addClass(element, 'today'); } } } var cleanClonedNode = (function() { var counter = 0; return function(node) { var nodesWithId = document.evaluate(".//*[@id]", node, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0; i < nodesWithId.snapshotLength; i++) { var item = nodesWithId.snapshotItem(0); item.id = ''; } var formNodes = document.evaluate(".//form", node, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); var formNames = []; for (var i = 0; i < formNodes.snapshotLength; i++) { var form = formNodes.snapshotItem(i); formNames.push(form.name); form.name += counter; } var linkNodes = document.evaluate(".//a[contains(@href, 'submit()')]", node, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0; i < linkNodes.snapshotLength; i++) { var link = linkNodes.snapshotItem(i); for (var j = 0; j < formNames.length; j++) { link.href = link.href.replace('.' + formNames[j] + '.submit', '.' + formNames[j] + counter + '.submit'); } } counter++; }; })(); function getClonedPageComponent(header) { var container = document.createElement('div'); var title = document.createElement('b'); title.appendChild(document.createTextNode(header)); container.appendChild(title); var xpath = '//td/font/b[text()="' + header + '"]/ancestor::table[1]/following-sibling::table[1]'; var component = document.evaluate(xpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); if (component.snapshotLength) { var clonedComponent = component.snapshotItem(0).cloneNode(true); cleanClonedNode(clonedComponent); container.appendChild(clonedComponent); } return container; } function getSpyReportLink() { var container = document.createElement('div'); var header = document.createElement('b'); header.appendChild(document.createTextNode('Spy Report:')); container.appendChild(header); container.appendChild(document.forms['spyreportshow'].cloneNode(true)); cleanClonedNode(container); return container; } function getActionComponent(formName) { var element = document.forms[formName]; while (element && element.tagName.toLowerCase() !== 'table') { element = element.parentNode; } element = element.cloneNode(); cleanClonedNode(element); return element; } function createAnnouncemntTab() { var container = document.createElement('div'); var newAnnouncement = document.getElementById('annul').cloneNode(true); newAnnouncement.style.height = ''; newAnnouncement.id = ''; container.appendChild(newAnnouncement); var changeLink = document.querySelector('form[name=cann] p'); if (changeLink) { container.appendChild(changeLink.cloneNode(true)); } return container; } function createMessagesTab(selector) { var container = document.createElement('div'); var form = document.createElement('form'); form.style.marginBottom = '0'; form.name = 'emess' + selector.length; form.action = document.forms['emess'].action; form.method = document.forms['emess'].method; form.appendChild(document.getElementsByName('player')[0].cloneNode(true)); form.appendChild(document.getElementsByName('pwd')[0].cloneNode(true)); var ul = document.createElement('ul'); ul.className = 'checklist cl3'; ul.style.height = 'auto'; ul.style.margin = '0'; form.appendChild(ul); var replaceClass = function(element, classToReplace, newClass) { if (hasClass(element, classToReplace)) { removeClass(element, classToReplace); addClass(element, newClass); } }; var messages = document.querySelectorAll(selector); for (var i = 0; i < messages.length; i++) { var clonedMessage = messages[i].cloneNode(true); if (i % 2 === 0) { replaceClass(clonedMessage, 'alt', 'alt2'); replaceClass(clonedMessage, 'alt3', 'alt4'); replaceClass(clonedMessage, 'alt5', 'alt6'); } else { replaceClass(clonedMessage, 'alt2', 'alt'); replaceClass(clonedMessage, 'alt4', 'alt3'); replaceClass(clonedMessage, 'alt6', 'alt5'); } markTodayElement(clonedMessage); ul.appendChild(clonedMessage); } var cloneForm = function(name) { var newForm = document.forms[name]; var formName = name + selector.length; newForm = newForm.cloneNode(true); newForm.name = formName; var links = newForm.querySelectorAll('a'); for (var i = 0; i < links.length; i++) { links[i].href = links[i].href.replace(name, formName); } return newForm; }; var table = document.createElement('table'); var tr = document.createElement('tr'); var tdLeft = document.createElement('td'); var tdRight = document.createElement('td'); table.style.width = '100%'; tdLeft.style.textAlign = 'left'; tdLeft.style.width = '50%'; tdRight.style.textAlign = 'right'; tdRight.style.verticalAlign = 'top'; tr.appendChild(tdLeft); tr.appendChild(tdRight); table.appendChild(tr); container.appendChild(table); var eraseMessagesInput = document.getElementsByName('messageerase')[0]; if (eraseMessagesInput) { form.appendChild(eraseMessagesInput.cloneNode(true)); var eraseLink = eraseMessagesInput.nextElementSibling.querySelector('a').cloneNode(true); eraseLink.href = eraseLink.href.replace('emess', form.name); tdLeft.appendChild(eraseLink); } if (document.forms['refreshmes']) { tdRight.appendChild(cloneForm('refreshmes')); } if (document.forms['emessall']) { tdLeft.appendChild(cloneForm('emessall')); } if (document.forms['leavem']) { container.appendChild(cloneForm('leavem')); } container.appendChild(form); addStyle("li.today label > b {background-color: #9AFFF2;}"); return container; } function createActionsTab() { var categories = [{ title: 'Areas', formNames: ['kat', 'phases', 'pizzamenu', 'fieldmenu', 'robofighto', 'concenter', 'tvtime', 'eldiablo', 'zat', 'festival', 'candyween'] }, { title: 'Agression', formNames: ['spy', 'vattack', 'bbook', 'obscure', 'irreport', 'spyreport'] }, { title: 'Bonuses', formNames: ['lemonaid', 'dicetime', 'brotime', 'blackstones', 'pandtime', 'ramen', 'ninjabeach'] }, { title: 'Actions', formNames: ['snowday', 'patrol', 'rescol', 'paperwork', 'zatrs', 'repcol', 'sgrs', 'ingredienthunt'] }, { title: 'Admin', formNames: ['setperm', 'science'] }]; var other = []; var cellNumber = 0; var table = document.createElement('table'); var tr; table.className='village-actions'; var createTd = function(title) { if (cellNumber % 2 === 0) { tr = document.createElement('tr'); table.appendChild(tr); } var td = document.createElement('td'); td.style.verticalAlign = 'top'; tr.appendChild(td); cellNumber++; td.appendChild(document.createTextNode(title)); return td; }; var actions = document.evaluate("//table[@width='255']//form/ancestor::table[1]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0; i < actions.snapshotLength; i++) { var action = actions.snapshotItem(i).cloneNode(true); var form = action.querySelector('form'); var added = false; for (var j = 0; j < categories.length; j++) { var category = categories[j]; for (var k = 0; k < category.formNames.length; k++) { if (form.name === category.formNames[k]) { category.actions = category.actions || []; category.actions.push(action); added = true; } } } if (!added) { other.push(action); } cleanClonedNode(action); } for (i = 0; i < categories.length; i++) { category = categories[i]; var td = createTd(category.title); for (j = 0; category.actions && j < category.actions.length; j++) { td.appendChild(category.actions[j]); } } if (other.length) { td = createTd('Other'); for (i = 0; i < other.length; i++) { td.appendChild(other[i]); } } addStyle("table.village-actions table {margin-bottom: 3px;}"); return table; } function createVillageTab() { var container = document.createElement('div'); var table = document.createElement('table'); var tr = document.createElement('tr'); var rightTd = document.createElement('td'); var leftTd = document.createElement('td'); rightTd.style.verticalAlign = 'top'; leftTd.style.verticalAlign = 'top'; table.style.width = '100%'; leftTd.style.width = '50%'; tr.appendChild(leftTd); tr.appendChild(rightTd); table.appendChild(tr); container.appendChild(table); var members = getClonedPageComponent('Members:'); var permissions = getClonedPageComponent('Permissions:'); var resources = getClonedPageComponent('Resources:'); var spyingOn = getClonedPageComponent('Spying On:'); var villageContracts = getClonedPageComponent('Village Contracts:'); var villageBank = getClonedPageComponent('Village Bank:'); var leaveVillage = getClonedPageComponent('Leave Village:'); rightTd.appendChild(members); leftTd.appendChild(resources); leftTd.appendChild(permissions); leftTd.appendChild(spyingOn); leftTd.appendChild(villageContracts); leftTd.appendChild(villageBank); leftTd.appendChild(leaveVillage); return container; } function createAdminTab() { var container = document.createElement('div'); var table = document.createElement('table'); var tr = document.createElement('tr'); var rightTd = document.createElement('td'); var leftTd = document.createElement('td'); rightTd.style.verticalAlign = 'top'; leftTd.style.verticalAlign = 'top'; table.style.width = '100%'; leftTd.style.width = '50%'; tr.appendChild(leftTd); tr.appendChild(rightTd); table.appendChild(tr); container.appendChild(table); var villageOptions = getClonedPageComponent('Village Options:'); var resourcePoints = getClonedPageComponent('Resource Points:'); var upgrades = getClonedPageComponent('Upgrades:'); var invasionDefense = getClonedPageComponent('Invasion Defense:'); var applications = getClonedPageComponent('Applications / Members:'); var yesterdayReport = getClonedPageComponent("Yesterday's Report:"); var newLeader = getClonedPageComponent('Name new Leader:'); var spyReportLink = getSpyReportLink(); var temp = yesterdayReport.querySelectorAll('td > div'); for (var i = 0; i < temp.length; i++) { temp[i].style.height = 'auto'; } rightTd.appendChild(villageOptions); leftTd.appendChild(spyReportLink); leftTd.appendChild(resourcePoints); leftTd.appendChild(upgrades); leftTd.appendChild(invasionDefense); leftTd.appendChild(applications); leftTd.appendChild(yesterdayReport); leftTd.appendChild(newLeader); return container; } (function() { var announcement = document.getElementById('annul'); var container = document.createElement('div'); container.style.className = 'village-tabs'; announcement.parentNode.insertBefore(container, announcement.previousElementSibling); var elementsToHide = []; var element = announcement.previousElementSibling; while (element) { elementsToHide.push(element); element = element.nextElementSibling; } elementsToHide.push(document.forms['emess']); var playerName = document.getElementsByName('player')[0].value; var onTabChange = function(index) { localStorage.setItem('village-revamp-' + playerName, index); }; var tabs = tabsPanel([{ text: 'Chat', content: createMessagesTab('#messageul li.alt3, #messageul li.alt4, #messageul li.alt5, #messageul li.alt6') }, { text: 'Log', content: createMessagesTab('#messageul li.alt, #messageul li.alt2') }, { text: 'Actions', content: createActionsTab() }, { text: 'Village', content: createVillageTab() }, { text: 'Admin', content: createAdminTab() }, { text: 'Announcement', content: createAnnouncemntTab() }, { text: 'Old', content: document.createElement('div'), show: function() { for (var i = 0; i < elementsToHide.length; i++) { elementsToHide[i].style.display = ''; } }, hide: function() { for (var i = 0; i < elementsToHide.length; i++) { elementsToHide[i].style.display = 'none'; } } }]); tabs.onTabChange(onTabChange); var tabIndex = localStorage.getItem('village-revamp-' + playerName) || 0; tabIndex = parseInt(tabIndex); tabs.selectTab(tabIndex); container.appendChild(tabs.container); addStyle([ ".village-tabs {width: 100%; text-align: left;}" ].join("\n")); })();