Youtube UI Fix

Moves the controls under the video and makes the UI look like it was before august 2015

当前为 2017-09-20 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Youtube UI Fix
  3. // @namespace YtUIFix
  4. // @description Moves the controls under the video and makes the UI look like it was before august 2015
  5. // @author Roy Scheerens
  6. // @homepageURL https://greasyfork.org/en/scripts/11485
  7. // @include https://www.youtube.com*
  8. // @include https://youtube.googleapis.com/embed*
  9. // @include https://www.youtube-nocookie.com/embed*
  10. // @version 2.4.1
  11. // @grant none
  12. // ==/UserScript==
  13. var YtNewUIFix = (function () {
  14. function YtNewUIFix() {
  15. var _this = this;
  16. this.prefix = "ytfix::";
  17. this.isEmbedded = window.top !== window.self;
  18. this.isSettingsPage = window.location.href.toLowerCase().match("(\.com\/embed|\.com)\/ui_fix_options") !== null;
  19. this.mouseMoveEvent = document.createEvent("Events");
  20. this.mouseMoveEvent.initEvent("mousemove", false, false);
  21. document.body.classList.add("yt-ui-fix");
  22. this.readOptions();
  23. addEventListener("storage", function (e) {
  24. if (e.key && e.key.indexOf(_this.prefix) >= 0) {
  25. _this.readOptions();
  26. _this.showOptions();
  27. _this.addCSS();
  28. _this.handleWatchLater();
  29. }
  30. });
  31. }
  32. YtNewUIFix.prototype.readOptions = function () {
  33. this.setOption("addWatchLater", true);
  34. this.setOption("showControlsFullscreen", true);
  35. this.setOption("showControlsNonFullscreen", true);
  36. this.setOption("changeColorsFullscreen", true);
  37. this.setOption("changeColorsNonFullscreen", true);
  38. this.setOption("removeAnimations", false);
  39. this.setOption("optionsReversed", false);
  40. this.setOption("progressBigger", false);
  41. this.setOption("showTitleOnHover", false);
  42. this.setOption("alwaysVolume", false);
  43. };
  44. YtNewUIFix.prototype.applyFix = function () {
  45. var _this = this;
  46. if (document.body.innerHTML.length === 0) {
  47. // empty page can be ignored (in share tab before it's active)
  48. return;
  49. }
  50. if (!this.isSettingsPage) {
  51. this.addCSS();
  52. this.checkMoviePlayer();
  53. window.setInterval(function () {
  54. _this.checkMoviePlayer();
  55. _this.handleWatchLater();
  56. }, 1000);
  57. }
  58. this.addOptions();
  59. };
  60. YtNewUIFix.prototype.setOption = function (key, defaultVal) {
  61. if (!localStorage) {
  62. this[key] = defaultVal;
  63. }
  64. var result = this.getSetting(key);
  65. if (result) {
  66. this[key] = (result === "true");
  67. }
  68. else {
  69. this[key] = defaultVal;
  70. }
  71. };
  72. YtNewUIFix.prototype.getSetting = function (key) {
  73. return localStorage.getItem(this.prefix + key);
  74. };
  75. YtNewUIFix.prototype.setSetting = function (key, value) {
  76. localStorage.setItem(this.prefix + key, String(value));
  77. };
  78. YtNewUIFix.prototype.checkMoviePlayer = function () {
  79. if (!this.moviePlayer || !this.moviePlayer.parentNode) {
  80. this.moviePlayer = document.querySelector("div.html5-video-player");
  81. }
  82. if (this.moviePlayer && this.moviePlayer.parentNode) {
  83. if (!this.moviePlayer.classList.contains("seeking-mode") &&
  84. !this.moviePlayer.classList.contains("dragging-mode") &&
  85. (this.showControlsNonFullscreen && !this.moviePlayer.classList.contains("ytp-fullscreen") || this.showControlsFullscreen && this.moviePlayer.classList.contains("ytp-fullscreen"))) {
  86. this.moviePlayer.dispatchEvent(this.mouseMoveEvent);
  87. }
  88. if (this.showTitleOnHover) {
  89. this.moviePlayer.classList.remove("ytp-hide-info-bar");
  90. }
  91. }
  92. };
  93. YtNewUIFix.prototype.handleWatchLater = function () {
  94. if (!this.watchLaterbutton || !this.settingsButton) {
  95. this.watchLaterbutton = document.querySelector(".ytp-chrome-top .ytp-watch-later-button");
  96. this.settingsButton = document.querySelector(".ytp-settings-button");
  97. if (this.watchLaterbutton && this.watchLaterbutton.parentElement) {
  98. this.oldWatchParent = this.watchLaterbutton.parentElement;
  99. }
  100. }
  101. if (this.watchLaterbutton && this.settingsButton) {
  102. if (this.addWatchLater && this.settingsButton.parentNode) {
  103. if (this.watchLaterbutton.parentNode !== this.settingsButton.parentNode) {
  104. this.settingsButton.parentNode.insertBefore(this.watchLaterbutton, this.settingsButton);
  105. }
  106. }
  107. else {
  108. this.oldWatchParent.appendChild(this.watchLaterbutton);
  109. }
  110. }
  111. };
  112. YtNewUIFix.prototype.addCSS = function () {
  113. var css = "";
  114. var StyleId = "YoutubeNewUIFix-Style";
  115. css = this.fixColors(css);
  116. css = this.fixControls(css);
  117. css = this.fixBigMode(css);
  118. css = this.addExtras(css);
  119. var style = document.getElementById(StyleId);
  120. if (style && style.parentNode) {
  121. style.parentNode.removeChild(style);
  122. delete style.textContent;
  123. }
  124. style = document.createElement("style");
  125. style.id = StyleId;
  126. style.textContent = css;
  127. document.head.appendChild(style);
  128. };
  129. YtNewUIFix.prototype.fixControls = function (css) {
  130. // options
  131. css += "h3.optionChanged::after { content: 'Refresh page to save changes'; color: red; position: relative; left: 15px; }\n\n";
  132. css += ".account-content-on-player { top: 0px; left: 0px; position: absolute; padding: 0; margin: 0; width: 100%; height: 100% }";
  133. css += ".account-content-on-player account-section { top: 50px; left: 50px; z-index: 100; background-color: white; padding: 10px 25px 30px; margin: 0; }";
  134. css += ".html5-video-player.ytp-fullscreen .html5-video-container { height: 100vh; }";
  135. css += ".html5-video-player:not(.ytp-fullscreen) .html5-video-container { height: 100%; }";
  136. css += ".ytp-chrome-bottom { left: 0 !important; }\n";
  137. css += ".ytp-chrome-controls { margin: 0 -12px; }\n";
  138. css += "body:not(.ytwp-window-player) #page:not(.watch-stage-mode) #watch7-sidebar { transform: translateY(-35px); }\n";
  139. css += "body:not(.ytwp-window-player) #page.watch-stage-mode #watch-appbar-playlist { margin-top: 35px; }\n";
  140. css += ".html5-main-video { top: 0!important; }\n";
  141. // progressbar
  142. css += ".ytp-progress-bar-container:not(.ytp-pulling) { height: 5px!important; bottom: 30px!important; }\n";
  143. css += ".ytp-progress-list { transform-origin: center top; }\n\n";
  144. css += ".ytp-big-mode .ytp-progress-list { transform: scaleY(0.6); }\n\n";
  145. // scale down
  146. css += ".ytp-chrome-bottom { height: 35px!important; padding: 0!important; }\n";
  147. css += ".ytp-chrome-controls .ytp-button { width: 33px!important; }\n";
  148. css += ".ytp-chrome-controls .ytp-watch-later-icon { width: 24px; height: 24px; }\n";
  149. css += ".ytp-chrome-controls .ytp-watch-later-button { opacity: 1; display: inline-block; transform: translateY(-4px); }\n";
  150. css += ".ytp-left-controls { margin-left: 5px }\n";
  151. css += ".ytp-time-display { height: 31px; line-height: 32px!important; font-size: 12px!important; }\n";
  152. css += ".ytp-left-controls, .ytp-right-controls { height: 32px!important; margin-top: 3px; line-height: 36px; }\n\n";
  153. // badges
  154. css += ".ytp-settings-button.ytp-hd-quality-badge::after,.ytp-settings-button.ytp-4k-quality-badge::after,.ytp-settings-button.ytp-5k-quality-badge::after,.ytp-settings-button.ytp-8k-quality-badge::after { content:''!important; top:6px!important; right:4px!important; height:9px!important; width:13px!important; padding: 0!important; }\n";
  155. css += ".ytp-settings-button.ytp-hd-quality-badge:after { background-image:url()!important; }";
  156. css += ".ytp-settings-button.ytp-4k-quality-badge:after { background-image:url()!important; }";
  157. css += ".ytp-settings-button.ytp-5k-quality-badge:after { background-image:url()!important; }";
  158. css += ".ytp-settings-button.ytp-8k-quality-badge:after { background-image:url()!important; }";
  159. css += ".ytp-settings-button.ytp-3d-badge-grey:after,.ytp-settings-button.ytp-3d-badge:after { background-image:url()!important; }";
  160. css += ".ytp-color-white .ytp-settings-button.ytp-hd-quality-badge:after { background-image:url()!important; }";
  161. css += ".ytp-color-white .ytp-settings-button.ytp-4k-quality-badge:after { background-image:url()!important; }";
  162. css += ".ytp-color-white .ytp-settings-button.ytp-5k-quality-badge:after { background-image:url()!important; }";
  163. css += ".ytp-color-white .ytp-settings-button.ytp-8k-quality-badge:after { background-image:url()!important; }";
  164. css += ".ytp-color-white .ytp-settings-button.ytp-3d-badge-grey:after,.ytp-color-white .ytp-settings-button.ytp-3d-badge:after { background-image:url()!important; }";
  165. // closed captions (line under the button, and position of captions)
  166. css += ".ytp-chrome-controls .ytp-button[aria-pressed='true']::after { width: 18px; left: 8px; }\n";
  167. css += ".caption-window.ytp-caption-window-bottom { margin-bottom:40px!important }\n";
  168. css += ".ytp-player-content { top: 48px !important; bottom: 49px !important; }\n";
  169. return css;
  170. };
  171. YtNewUIFix.prototype.fixBigMode = function (css) {
  172. /* big mode: smaller scrubber */
  173. css += ".ytp-big-mode .ytp-scrubber-button { height: 13px!important; width: 13px!important; border-radius: 6.5px!important; }\n";
  174. css += ".ytp-big-mode .ytp-scrubber-container { top: -4px; left: -6.5px; }\n\n";
  175. /* big mode: 24px edges instead of 12px */
  176. css += ".ytp-big-mode .ytp-left-controls { margin-left: -7px }\n";
  177. css += ".ytp-big-mode .ytp-fullscreen-button { margin-right: -7px }\n\n";
  178. /* big mode: smaller volume slider */
  179. css += ".ytp-big-mode .ytp-volume-slider-handle { width: 12px; height: 12px; border-radius: 6px; margin-top: -6px; }\n";
  180. css += ".ytp-big-mode .ytp-volume-slider-active .ytp-volume-panel { width: 72px; }\n\n";
  181. return css;
  182. };
  183. YtNewUIFix.prototype.fixColors = function (css) {
  184. if (this.changeColorsNonFullscreen) {
  185. css += ".html5-video-player:not(.ytp-big-mode) .ytp-chrome-bottom { background-color: #1B1B1B; border-left: 12px solid #1B1B1B; border-right: 12px solid #1B1B1B; }\n";
  186. css += ".html5-video-player:not(.ytp-big-mode) .ytp-gradient-bottom { display: none!important; }\n";
  187. css += ".html5-video-player:not(.ytp-big-mode) .ytp-chrome-controls svg path { fill: #8E8E8E }\n";
  188. }
  189. else {
  190. css += ".html5-video-player:not(.ytp-big-mode) .ytp-chrome-bottom { border-left: 12px solid transparent; border-right: 12px solid transparent; }\n";
  191. if (this.showControlsNonFullscreen) {
  192. css += ".html5-video-player:not(.ytp-fullscreen) .ytp-gradient-bottom { display: none!important; }\n";
  193. }
  194. }
  195. if (this.changeColorsFullscreen) {
  196. css += ".ytp-big-mode .ytp-chrome-bottom { background-color: #1B1B1B; border-left: 24px solid #1B1B1B; border-right: 24px solid #1B1B1B; }\n";
  197. css += ".ytp-big-mode .ytp-gradient-bottom { display: none!important; }\n";
  198. css += ".ytp-big-mode .ytp-chrome-controls svg path { fill: #8E8E8E }\n";
  199. }
  200. else {
  201. css += ".ytp-big-mode .ytp-chrome-bottom { border-left: 24px solid transparent; border-right: 24px solid transparent; }\n";
  202. if (this.showControlsFullscreen) {
  203. css += ".html5-video-player.ytp-fullscreen .ytp-gradient-bottom { display: none!important; }\n";
  204. }
  205. }
  206. css += ".ytp-gradient-top { display: none!important; }\n";
  207. css += "\n";
  208. return css;
  209. };
  210. YtNewUIFix.prototype.addExtras = function (css) {
  211. if (this.showControlsFullscreen) {
  212. css += "html:not(.floater) .html5-video-player.ytp-fullscreen:not(.ytp-hide-controls) .html5-main-video { height: calc(100% - 35px)!important; min-height: calc(100% - 35px) !important; max-height: calc(100% - 35px) !important; }\n";
  213. css += ".html5-video-player.ytp-fullscreen:not(.ytp-hide-controls) .ytp-chrome-bottom { opacity: 1!important; display: block!important; }\n";
  214. }
  215. if (this.showControlsNonFullscreen) {
  216. css += "html:not(.floater) .html5-video-player:not(.ytp-fullscreen):not(.ytp-hide-controls) .html5-main-video { height: calc(100% - 35px)!important; min-height: calc(100% - 35px) !important; max-height: calc(100% - 35px) !important; }\n";
  217. css += ".html5-video-player:not(.ytp-fullscreen):not(.ytp-hide-controls) .ytp-chrome-bottom { opacity: 1!important; display: block!important; }\n";
  218. css += "html:not(.floater):not(.part_fullbrowser) #movie_player:not(.ytp-fullscreen):not(.ytp-hide-controls) { height: calc(100% + 35px)!important; }\n";
  219. css += "html.floater .html5-video-player { padding-bottom: 35px; }\n";
  220. css += "#theater-background { padding-bottom: 35px; }\n\n";
  221. css += "#placeholder-player { padding-bottom: 35px; }\n";
  222. }
  223. if (!this.showControlsFullscreen && !this.showControlsNonFullscreen) {
  224. css += ".html5-video-player .html5-main-video { height: 100%!important; }\n";
  225. }
  226. if (!this.showTitleOnHover) {
  227. // hide always
  228. css += ".ytp-chrome-top { display: none!important; }\n";
  229. }
  230. if (this.removeAnimations) {
  231. css += ".ytp-bezel { display: none!important; }\n";
  232. css += ".html5-endscreen *, .html5-video-player div { transition-property: none !important; animation: none !important; }\n";
  233. }
  234. if (this.optionsReversed) {
  235. css += ".ytp-panel { display: -webkit-flex; -webkit-flex-direction: column; display: flex; flex-direction: column; }\n";
  236. css += ".ytp-panel-header { order: 2; border-top: 1px solid #444; border-bottom: none; }\n";
  237. css += ".ytp-panel-content { order: 1; }\n";
  238. }
  239. if (this.alwaysVolume) {
  240. /* Have the volume slider always be visible */
  241. css += ".ytp-volume-panel { width: 52px; margin-right: 3px; } .ytp-big-mode .ytp-volume-panel { width: 78px; }";
  242. }
  243. if (this.progressBigger) {
  244. /* Make the progressbar fill up the entire space when not hovering over (thanks to Takato) */
  245. css += ".ytp-progress-bar-container:not(:hover):not(.ytp-pulling) .ytp-progress-list { width: calc(100% + 24px); left: -12px; }";
  246. css += ".ytp-big-mode .ytp-progress-bar-container:not(:hover):not(.ytp-pulling) .ytp-progress-list { width: calc(100% + 48px); left: -24px; }";
  247. }
  248. return css;
  249. };
  250. YtNewUIFix.prototype.showOptions = function () {
  251. var options = document.querySelectorAll("#YoutubeNewUIFix-Options input");
  252. if (options.length > 0) {
  253. for (var i = 0; i < options.length; i++) {
  254. options[i].checked = (this.getSetting(options[i].name) === "true");
  255. }
  256. }
  257. };
  258. YtNewUIFix.prototype.addOptions = function () {
  259. if (localStorage) {
  260. var accSection = document.createElement("div");
  261. accSection.id = "YoutubeNewUIFix-Options";
  262. accSection.classList.add("account-section");
  263. var header = document.createElement("h3");
  264. header.classList.add("account-section-header");
  265. header.textContent = "Youtube UI Fix Options";
  266. accSection.appendChild(header);
  267. {
  268. accSection.appendChild(this.createOption("addWatchLater", "Add the watch later button to the controls"));
  269. accSection.appendChild(this.createOption("changeColorsNonFullscreen", "Change the colors back to their original gray (in non-full-screen mode)"));
  270. accSection.appendChild(this.createOption("changeColorsFullscreen", "Change the colors back to their original gray in full-screen mode"));
  271. accSection.appendChild(this.createOption("showControlsNonFullscreen", "Always show the controls (in non-full-screen mode)"));
  272. accSection.appendChild(this.createOption("showControlsFullscreen", "Always show the controls in full-screen mode"));
  273. accSection.appendChild(this.createOption("removeAnimations", "Remove all animations"));
  274. accSection.appendChild(this.createOption("optionsReversed", "Move the 'go back' button in the settings menus to the bottom"));
  275. accSection.appendChild(this.createOption("progressBigger", "Make the progressbar take up the whole width (but not when hovering over)"));
  276. accSection.appendChild(this.createOption("showTitleOnHover", "Have the title show when hovering over the video"));
  277. accSection.appendChild(this.createOption("alwaysVolume", "Have the volume slider be always visible"));
  278. }
  279. var content = document.querySelector("div.account-content");
  280. var footer = document.querySelector("div.account-footer");
  281. var selectedItem = document.querySelector(".creator-sidebar-item.selected");
  282. if (this.isSettingsPage) {
  283. document.head.innerHTML = document.body.innerHTML = "";
  284. document.body.appendChild(accSection);
  285. }
  286. else if (content && footer && selectedItem.innerHTML.indexOf("Playback") >= 0) {
  287. content.insertBefore(accSection, footer);
  288. }
  289. }
  290. };
  291. YtNewUIFix.prototype.createOption = function (name, description) {
  292. var _this = this;
  293. var accDiv = document.createElement("div");
  294. accDiv.classList.add("account-section-setting");
  295. accDiv.innerHTML = "\n\t\t <label>\n\t\t\t <span class='yt-uix-form-input-checkbox-container " + (this[name] ? "checked" : "") + "'>\n <input class='yt-uix-form-input-checkbox' name='" + name + "' " + (this[name] ? "checked='checked'" : "") + " type='checkbox'>\n <span class='yt-uix-form-input-checkbox-element'></span>\n </span>\n\t\t\t " + description + "\n\t\t </label>";
  296. var accInput = accDiv.querySelector("input[name='" + name + "']");
  297. accInput.onclick = function () {
  298. _this.setSetting(name, accInput.checked);
  299. };
  300. return accDiv;
  301. };
  302. return YtNewUIFix;
  303. }());
  304. new YtNewUIFix().applyFix();
  305. //# sourceMappingURL=Youtube_UI_Fix.user.js.map