您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Multiple UI Tweaks to make the website look more accurate to how it was back in 2016!
// ==UserScript== // @name Noise's 2016 Website UI Tweaks // @namespace http://tampermonkey.net/ // @version 2.3 // @description Multiple UI Tweaks to make the website look more accurate to how it was back in 2016! // @author The Noise! // @match https://*.roblox.com/* // @icon  // @license MIT // ==/UserScript== (function () { 'use strict'; // Define replaceAvatarImages function at the top level const replaceAvatarImages = () => { // Only run on specified pages if (!window.location.href.includes('roblox.com/home') && !window.location.href.includes('roblox.com/communities/') && !window.location.href.includes('roblox.com/users/') && !window.location.href.includes('roblox.com/search/users')) { return; } // First, ensure header avatars are always headshots const headerAvatars = [ '#home-header img.avatar-card-image', '.container-header img[src*="AvatarHeadshot"]' ]; headerAvatars.forEach(selector => { const headerAvatar = document.querySelector(selector); if (headerAvatar && headerAvatar.src.includes('30DAY-Avatar-')) { headerAvatar.src = headerAvatar.src.replace('30DAY-Avatar-', '30DAY-AvatarHeadshot-'); console.log('Restored header avatar to headshot'); } }); // Handle user search page if (window.location.href.includes('roblox.com/search/users')) { const searchResultImages = document.querySelectorAll('.avatar-card-container img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]'); searchResultImages.forEach(image => { if (image.closest('.container-header')) return; // Skip container-header images const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-"); image.src = newSrc; console.log(`Updated search result image source to: ${newSrc}`); }); return; } // Handle friends/followers/following pages if (window.location.href.includes('roblox.com/users/') && window.location.href.includes('#!/')) { const userListImages = document.querySelectorAll('.list-item img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]'); userListImages.forEach(image => { if (image.closest('.container-header')) return; // Skip container-header images const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-"); image.src = newSrc; console.log(`Updated user list image source to: ${newSrc}`); }); return; } // Handle avatars in friends carousel on user profile pages if (window.location.href.includes('roblox.com/users/')) { const friendsCarouselImages = document.querySelectorAll('.friends-carousel-container img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]'); friendsCarouselImages.forEach(image => { if (image.closest('.container-header')) return; // Skip container-header images const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-"); image.src = newSrc; console.log(`Updated friends carousel image source to: ${newSrc}`); }); return; } // Handle other pages (home and communities) const avatarImages = document.querySelectorAll('img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]'); avatarImages.forEach(image => { if (image.closest('#home-header') || image.closest('.container-header')) { return; } if (image.id === "home-avatar-thumb") { console.log('Skipped updating home avatar image.'); return; } const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-"); image.src = newSrc; console.log(`Updated image source to: ${newSrc}`); }); }; const modifyUI = () => { // Remove "Money" navigation element const moneyNav = document.querySelector('a#nav-money'); if (moneyNav) { moneyNav.remove(); console.log('Removed "Money" navigation element.'); } // Remove "Premium" navigation element const premiumNav = document.querySelector('a#nav-premium'); if (premiumNav) { premiumNav.remove(); console.log('Removed "Premium" navigation element.'); } // Add Message button to user profile pages if (window.location.href.includes('roblox.com/users/')) { const addFriendButton = document.querySelector('.details-actions.desktop-action .btn-friends'); if (addFriendButton && !document.querySelector('.btn-message')) { const messageButtonLi = document.createElement('li'); messageButtonLi.className = 'btn-message'; messageButtonLi.innerHTML = ` <button class="btn-control-md" ng-disabled="!profileHeaderLayout.canMessage || profileHeaderLayout.userId == 0" ng-click="sendMessage()" disabled="disabled"> Message </button> `; addFriendButton.insertAdjacentElement('afterend', messageButtonLi); console.log('Added Message button next to Add Friend button'); } } // Replace "Communities" with "Groups" document.querySelectorAll('span.font-header-2.dynamic-ellipsis-item[title="Communities"]').forEach(element => { if (element.textContent.trim() === "Communities") { element.textContent = "Groups"; console.log('Replaced "Communities" with "Groups".'); } }); // Replace "Communities" in visible text nodes document.querySelectorAll('*:not(script):not(style)').forEach(node => { if (node.childNodes.length) { node.childNodes.forEach(child => { if (child.nodeType === Node.TEXT_NODE && child.nodeValue.includes("Communities")) { child.nodeValue = child.nodeValue.replace(/Communities/g, "Groups"); } }); } }); // Replace group-related text const elements = { 'button#group-join-button': 'Join Community|Join Group', 'button.ng-binding': 'Leave Community|Leave Group', 'a.ng-binding': 'Configure Community|Configure Group' }; for (const [selector, replacement] of Object.entries(elements)) { const [oldText, newText] = replacement.split('|'); document.querySelectorAll(selector).forEach(element => { if (element.textContent.trim() === oldText) { element.textContent = newText; console.log(`Replaced "${oldText}" with "${newText}"`); } }); } }; // Comments section functionality - only runs on game pages if (window.location.href.includes('roblox.com/games/')) { const addCommentsSection = (forceFallback = false) => { if (document.getElementById('AjaxCommentsContainer')) return true; const commentsHTML = ` <div id="AjaxCommentsContainer" class="comments-container" data-asset-id="32990482" data-total-collection-size="" data-is-user-authenticated="False" data-signin-url="https://www.roblox.com/newlogin?returnUrl=%2Fgames%2F32990482%2FFlood-Escape"> <h3>Comments</h3> <div class="section-content AddAComment"> <div class="comment-form"> <form class="form-horizontal ng-pristine ng-valid" role="form"> <div class="form-group"> <textarea class="form-control input-field rbx-comment-input blur" placeholder="Write a comment!" rows="1"></textarea> <div class="rbx-comment-msgs"> <span class="rbx-comment-error text-error" style="display: none;"> </span> <span class="rbx-comment-count small"></span> </div> </div> <button type="button" class="btn-secondary-md rbx-post-comment">Post Comment</button> </form> </div> <div class="comments vlist"> <div class="empty">No comments found.</div> </div> </div> </div> `; if (!forceFallback) { const badgesList = document.querySelector('.stack.badge-container.game-badges-list'); if (badgesList) { badgesList.insertAdjacentHTML('afterend', commentsHTML); setupCommentButton(); return true; } return false; } const recommendedSection = document.querySelector('.container-list.games-detail'); if (recommendedSection) { recommendedSection.insertAdjacentHTML('beforebegin', commentsHTML); setupCommentButton(); return true; } return false; }; const setupCommentButton = () => { const postCommentButton = document.querySelector('.rbx-post-comment'); if (postCommentButton) { postCommentButton.addEventListener('click', (event) => { event.preventDefault(); }); } }; const init = () => { let badgeAttempts = 0; const maxBadgeAttempts = 15; const badgeInterval = setInterval(() => { badgeAttempts++; if (addCommentsSection(false)) { clearInterval(badgeInterval); } else if (badgeAttempts >= maxBadgeAttempts) { clearInterval(badgeInterval); addCommentsSection(true); } }, 1000); addCommentsSection(false); }; init(); } // Run initial modifications modifyUI(); replaceAvatarImages(); // Observe the DOM for dynamic changes with both functions const observer = new MutationObserver(() => { modifyUI(); replaceAvatarImages(); }); observer.observe(document.body, { childList: true, subtree: true }); })();