Greasy Fork is available in English.

Noise's 2016 Website UI Tweaks

Multiple UI Tweaks to make the website look more accurate to how it was back in 2016!

  1. // ==UserScript==
  2. // @name Noise's 2016 Website UI Tweaks
  3. // @namespace http://tampermonkey.net/
  4. // @version 2.3
  5. // @description Multiple UI Tweaks to make the website look more accurate to how it was back in 2016!
  6. // @author The Noise!
  7. // @match https://*.roblox.com/*
  8. // @icon 
  9. // @license MIT
  10. // ==/UserScript==
  11.  
  12. (function () {
  13. 'use strict';
  14.  
  15. // Define replaceAvatarImages function at the top level
  16. const replaceAvatarImages = () => {
  17. // Only run on specified pages
  18. if (!window.location.href.includes('roblox.com/home') &&
  19. !window.location.href.includes('roblox.com/communities/') &&
  20. !window.location.href.includes('roblox.com/users/') &&
  21. !window.location.href.includes('roblox.com/search/users')) {
  22. return;
  23. }
  24.  
  25. // First, ensure header avatars are always headshots
  26. const headerAvatars = [
  27. '#home-header img.avatar-card-image',
  28. '.container-header img[src*="AvatarHeadshot"]'
  29. ];
  30.  
  31. headerAvatars.forEach(selector => {
  32. const headerAvatar = document.querySelector(selector);
  33. if (headerAvatar && headerAvatar.src.includes('30DAY-Avatar-')) {
  34. headerAvatar.src = headerAvatar.src.replace('30DAY-Avatar-', '30DAY-AvatarHeadshot-');
  35. console.log('Restored header avatar to headshot');
  36. }
  37. });
  38.  
  39. // Handle user search page
  40. if (window.location.href.includes('roblox.com/search/users')) {
  41. const searchResultImages = document.querySelectorAll('.avatar-card-container img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]');
  42. searchResultImages.forEach(image => {
  43. if (image.closest('.container-header')) return; // Skip container-header images
  44. const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-");
  45. image.src = newSrc;
  46. console.log(`Updated search result image source to: ${newSrc}`);
  47. });
  48. return;
  49. }
  50.  
  51. // Handle friends/followers/following pages
  52. if (window.location.href.includes('roblox.com/users/') && window.location.href.includes('#!/')) {
  53. const userListImages = document.querySelectorAll('.list-item img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]');
  54. userListImages.forEach(image => {
  55. if (image.closest('.container-header')) return; // Skip container-header images
  56. const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-");
  57. image.src = newSrc;
  58. console.log(`Updated user list image source to: ${newSrc}`);
  59. });
  60. return;
  61. }
  62.  
  63. // Handle avatars in friends carousel on user profile pages
  64. if (window.location.href.includes('roblox.com/users/')) {
  65. const friendsCarouselImages = document.querySelectorAll('.friends-carousel-container img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]');
  66. friendsCarouselImages.forEach(image => {
  67. if (image.closest('.container-header')) return; // Skip container-header images
  68. const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-");
  69. image.src = newSrc;
  70. console.log(`Updated friends carousel image source to: ${newSrc}`);
  71. });
  72. return;
  73. }
  74.  
  75. // Handle other pages (home and communities)
  76. const avatarImages = document.querySelectorAll('img[src^="https://tr.rbxcdn.com/30DAY-AvatarHeadshot-"]');
  77. avatarImages.forEach(image => {
  78. if (image.closest('#home-header') || image.closest('.container-header')) {
  79. return;
  80. }
  81.  
  82. if (image.id === "home-avatar-thumb") {
  83. console.log('Skipped updating home avatar image.');
  84. return;
  85. }
  86.  
  87. const newSrc = image.src.replace("30DAY-AvatarHeadshot-", "30DAY-Avatar-");
  88. image.src = newSrc;
  89. console.log(`Updated image source to: ${newSrc}`);
  90. });
  91. };
  92.  
  93. const modifyUI = () => {
  94. // Remove "Money" navigation element
  95. const moneyNav = document.querySelector('a#nav-money');
  96. if (moneyNav) {
  97. moneyNav.remove();
  98. console.log('Removed "Money" navigation element.');
  99. }
  100.  
  101. // Remove "Premium" navigation element
  102. const premiumNav = document.querySelector('a#nav-premium');
  103. if (premiumNav) {
  104. premiumNav.remove();
  105. console.log('Removed "Premium" navigation element.');
  106. }
  107.  
  108. // Add Message button to user profile pages
  109. if (window.location.href.includes('roblox.com/users/')) {
  110. const addFriendButton = document.querySelector('.details-actions.desktop-action .btn-friends');
  111. if (addFriendButton && !document.querySelector('.btn-message')) {
  112. const messageButtonLi = document.createElement('li');
  113. messageButtonLi.className = 'btn-message';
  114. messageButtonLi.innerHTML = `
  115. <button class="btn-control-md" ng-disabled="!profileHeaderLayout.canMessage || profileHeaderLayout.userId == 0" ng-click="sendMessage()" disabled="disabled">
  116. Message
  117. </button>
  118. `;
  119. addFriendButton.insertAdjacentElement('afterend', messageButtonLi);
  120. console.log('Added Message button next to Add Friend button');
  121. }
  122. }
  123.  
  124. // Replace "Communities" with "Groups"
  125. document.querySelectorAll('span.font-header-2.dynamic-ellipsis-item[title="Communities"]').forEach(element => {
  126. if (element.textContent.trim() === "Communities") {
  127. element.textContent = "Groups";
  128. console.log('Replaced "Communities" with "Groups".');
  129. }
  130. });
  131.  
  132. // Replace "Communities" in visible text nodes
  133. document.querySelectorAll('*:not(script):not(style)').forEach(node => {
  134. if (node.childNodes.length) {
  135. node.childNodes.forEach(child => {
  136. if (child.nodeType === Node.TEXT_NODE && child.nodeValue.includes("Communities")) {
  137. child.nodeValue = child.nodeValue.replace(/Communities/g, "Groups");
  138. }
  139. });
  140. }
  141. });
  142.  
  143. // Replace group-related text
  144. const elements = {
  145. 'button#group-join-button': 'Join Community|Join Group',
  146. 'button.ng-binding': 'Leave Community|Leave Group',
  147. 'a.ng-binding': 'Configure Community|Configure Group'
  148. };
  149.  
  150. for (const [selector, replacement] of Object.entries(elements)) {
  151. const [oldText, newText] = replacement.split('|');
  152. document.querySelectorAll(selector).forEach(element => {
  153. if (element.textContent.trim() === oldText) {
  154. element.textContent = newText;
  155. console.log(`Replaced "${oldText}" with "${newText}"`);
  156. }
  157. });
  158. }
  159. };
  160.  
  161. // Comments section functionality - only runs on game pages
  162. if (window.location.href.includes('roblox.com/games/')) {
  163. const addCommentsSection = (forceFallback = false) => {
  164. if (document.getElementById('AjaxCommentsContainer')) return true;
  165.  
  166. const commentsHTML = `
  167. <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">
  168. <h3>Comments</h3>
  169. <div class="section-content AddAComment">
  170. <div class="comment-form">
  171. <form class="form-horizontal ng-pristine ng-valid" role="form">
  172. <div class="form-group">
  173. <textarea class="form-control input-field rbx-comment-input blur" placeholder="Write a comment!" rows="1"></textarea>
  174. <div class="rbx-comment-msgs">
  175. <span class="rbx-comment-error text-error" style="display: none;"> </span>
  176. <span class="rbx-comment-count small"></span>
  177. </div>
  178. </div>
  179. <button type="button" class="btn-secondary-md rbx-post-comment">Post Comment</button>
  180. </form>
  181. </div>
  182. <div class="comments vlist">
  183. <div class="empty">No comments found.</div>
  184. </div>
  185. </div>
  186. </div>
  187. `;
  188.  
  189. if (!forceFallback) {
  190. const badgesList = document.querySelector('.stack.badge-container.game-badges-list');
  191. if (badgesList) {
  192. badgesList.insertAdjacentHTML('afterend', commentsHTML);
  193. setupCommentButton();
  194. return true;
  195. }
  196. return false;
  197. }
  198.  
  199. const recommendedSection = document.querySelector('.container-list.games-detail');
  200. if (recommendedSection) {
  201. recommendedSection.insertAdjacentHTML('beforebegin', commentsHTML);
  202. setupCommentButton();
  203. return true;
  204. }
  205.  
  206. return false;
  207. };
  208.  
  209. const setupCommentButton = () => {
  210. const postCommentButton = document.querySelector('.rbx-post-comment');
  211. if (postCommentButton) {
  212. postCommentButton.addEventListener('click', (event) => {
  213. event.preventDefault();
  214. });
  215. }
  216. };
  217.  
  218. const init = () => {
  219. let badgeAttempts = 0;
  220. const maxBadgeAttempts = 15;
  221.  
  222. const badgeInterval = setInterval(() => {
  223. badgeAttempts++;
  224. if (addCommentsSection(false)) {
  225. clearInterval(badgeInterval);
  226. } else if (badgeAttempts >= maxBadgeAttempts) {
  227. clearInterval(badgeInterval);
  228. addCommentsSection(true);
  229. }
  230. }, 1000);
  231.  
  232. addCommentsSection(false);
  233. };
  234.  
  235. init();
  236. }
  237.  
  238. // Run initial modifications
  239. modifyUI();
  240. replaceAvatarImages();
  241.  
  242. // Observe the DOM for dynamic changes with both functions
  243. const observer = new MutationObserver(() => {
  244. modifyUI();
  245. replaceAvatarImages();
  246. });
  247.  
  248. observer.observe(document.body, { childList: true, subtree: true });
  249. })();