Greasy Fork镜像 支持简体中文。

Twitter kaizen

Twitterの表示を改善するスクリプト

目前為 2024-11-13 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name Twitter kaizen
  3. // @name:ja Twitter kaizen
  4. // @name:en Twitter kaizen
  5. // @name:zh-CN Twitter kaizen
  6. // @name:ko Twitter kaizen
  7. // @name:ru Twitter kaizen
  8. // @name:de Twitter kaizen
  9. // @description Twitterの表示を改善するスクリプト
  10. // @description:ja Twitterの表示を改善するスクリプト
  11. // @description:en Script to improve Twitter display
  12. // @description:zh-CN 改善Twitter显示的脚本
  13. // @description:ko 트위터 표시를 개선하는 스크립트
  14. // @description:ru Скрипт для улучшения отображения Twitter
  15. // @description:de Skript zur Verbesserung der Twitter-Anzeige
  16. // @version 2.4
  17. // @author Yos_sy
  18. // @match https://x.com/*
  19. // @namespace http://tampermonkey.net/
  20. // @icon 
  21. // @license MIT
  22. // @run-at document-start
  23. // @require https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.5.2/js/all.min.js
  24. // @grant GM_addStyle
  25. // @grant GM_getResourceText
  26. // @grant GM_registerMenuCommand
  27. // ==/UserScript==
  28.  
  29. (function () {
  30. "use strict";
  31.  
  32. GM_addStyle(`
  33. /* -----------------------------------------------------------------------------------
  34. 基本的なボーダーを消す
  35. ----------------------------------------------------------------------------------- */
  36. .r-1kqtdi0,
  37. .r-1igl3o0 {
  38. border: none !important;
  39. }
  40. /* -----------------------------------------------------------------------------------
  41. カキコの下のボーダーを消す
  42. ----------------------------------------------------------------------------------- */
  43. .r-109y4c4 {
  44. height: 0 !important;
  45. }
  46. /* -----------------------------------------------------------------------------------
  47. TLの幅を600pxから700pxに、右サイドバーの幅を350pxから250pxに変更
  48. ----------------------------------------------------------------------------------- */
  49. .r-1ye8kvj {
  50. max-width: 700px !important;
  51. }
  52. .r-1hycxz {
  53. width: 250px !important;
  54. }
  55. .css-175oi2r.r-kemksi.r-1kqtdi0.r-th6na.r-1phboty.r-1dqxon3.r-1hycxz {
  56. width: 350px !important;
  57. }
  58. /* -----------------------------------------------------------------------------------
  59. ヘッダーのスクロールバーを消す
  60. ----------------------------------------------------------------------------------- */
  61. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd {
  62. overflow-y: scroll !important;
  63. -ms-overflow-style: none !important;
  64. scrollbar-width: none !important;
  65. }
  66. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd::-webkit-scrollbar {
  67. display: none !important;
  68. }
  69. /* -----------------------------------------------------------------------------------
  70. サイドバーの”Subscribe to Premium”を消す
  71. ----------------------------------------------------------------------------------- */
  72. .css-175oi2r.r-1habvwh.r-eqz5dr.r-uaa2di.r-1mmae3n.r-3pj75a.r-bnwqim {
  73. display: none;
  74. }
  75. /* -----------------------------------------------------------------------------------
  76. サイドバーの”Who to follow”を消す
  77. ----------------------------------------------------------------------------------- */
  78. .css-175oi2r.r-1bro5k0 {
  79. display: none;
  80. }
  81. /* -----------------------------------------------------------------------------------
  82. TL上のUserNameを消す
  83. ----------------------------------------------------------------------------------- */
  84. a
  85. > .css-146c3p1.r-dnmrzs.r-1udh08x.r-3s2u2q.r-bcqeeo.r-1ttztb7.r-qvutc0.r-1qd0xha.r-a023e6.r-rjixqe.r-16dba41.r-18u37iz.r-1wvb978,
  86. .css-175oi2r:nth-child(2)
  87. > .css-175oi2r
  88. > .css-175oi2r:nth-child(2)
  89. > .css-175oi2r
  90. > .css-175oi2r:nth-child(1)
  91. > .css-175oi2r
  92. > .css-146c3p1:nth-child(1)
  93. > .css-1jxf684,
  94. .css-146c3p1.r-bcqeeo.r-1ttztb7.r-qvutc0.r-1qd0xha.r-a023e6.r-rjixqe.r-16dba41.r-1q142lx.r-n7gxbd {
  95. display: none;
  96. }
  97. /* -----------------------------------------------------------------------------------
  98. サイドバーのWhat’s happeningのステータスを見やすく
  99. ----------------------------------------------------------------------------------- */
  100. .css-175oi2r.r-1mmae3n.r-3pj75a.r-o7ynqc.r-6416eg.r-1ny4l3l.r-1loqt21
  101. > div
  102. > div
  103. > .css-175oi2r.r-1wbh5a2.r-1awozwy.r-18u37iz {
  104. display: flex;
  105. flex-flow: column;
  106. }
  107. .r-r2y082 {
  108. max-width: 100%;
  109. }
  110. /* -----------------------------------------------------------------------------------
  111. 時計、日付のフォントカラーを変更
  112. ----------------------------------------------------------------------------------- */
  113. #date__container__text,
  114. #time__container__text {
  115. color: #e7e9ea;
  116. }
  117. `);
  118.  
  119. // ローカルストレージから設定を読み込む
  120. function loadConfig() {
  121. const savedConfig = localStorage.getItem("twitterKaizenConfig");
  122. if (savedConfig) {
  123. Object.assign(config, JSON.parse(savedConfig));
  124. }
  125. }
  126.  
  127. // ローカルストレージに設定を保存
  128. function saveConfig() {
  129. localStorage.setItem("twitterKaizenConfig", JSON.stringify(config));
  130. }
  131.  
  132. // -----------------------------------------------------------------------------------
  133. // ユーティリティ関数と定数
  134. // -----------------------------------------------------------------------------------
  135. const Utils = {
  136. debounce: (func, wait) => {
  137. let timeout;
  138. return (...args) => {
  139. clearTimeout(timeout);
  140. timeout = setTimeout(() => func(...args), wait);
  141. };
  142. },
  143.  
  144. pad: (num) => num.toString().padStart(2, "0"),
  145.  
  146. createElement: (tag, options = {}) => {
  147. const element = document.createElement(tag);
  148. if (options.id) element.id = options.id;
  149. options.classList?.forEach((cls) => element.classList.add(cls));
  150. Object.entries(options.attributes || {}).forEach(([attr, value]) =>
  151. element.setAttribute(attr, value)
  152. );
  153. if (options.innerHTML) element.innerHTML = options.innerHTML;
  154. if (options.textContent) element.textContent = options.textContent;
  155. return element;
  156. },
  157.  
  158. observeDOM: (
  159. targetNode,
  160. callback,
  161. config = { childList: true, subtree: true }
  162. ) => {
  163. const observer = new MutationObserver(callback);
  164. observer.observe(targetNode, config);
  165. return observer;
  166. },
  167. };
  168.  
  169. // 多言語定義
  170. const TRANSLATIONS = {
  171. en: {
  172. panel: {
  173. replaceIcons: "Reclaim Twitter (restore icon)",
  174. useAbsoluteTime: "Change TL time from relative to absolute time",
  175. showTimeAndDateSidebar: "Display time and date in sidebar",
  176. useDefaultVideoPlayer: "Revert video player to default",
  177. enhanceTweetEngagements: "Easy access to quoted tweets",
  178. },
  179. weeks: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
  180. },
  181. ja: {
  182. panel: {
  183. replaceIcons: "Twitterを取り戻す (アイコンを元に戻す)",
  184. useAbsoluteTime: "TLの時間を相対時間から絶対時間に変更",
  185. showTimeAndDateSidebar: "サイドバーに時間、日付を表示",
  186. useDefaultVideoPlayer: "動画プレイヤーをデフォルトに戻す",
  187. enhanceTweetEngagements: "引用ツイートへのアクセスを簡単に",
  188. },
  189. weeks: ["日", "月", "火", "水", "木", "金", "土"],
  190. },
  191. zh: {
  192. panel: {
  193. replaceIcons: "替换 Twitter 图标",
  194. useAbsoluteTime: "使用绝对时间",
  195. showTimeAndDateSidebar: "显示时间和日期侧边栏",
  196. useDefaultVideoPlayer: "使用默认视频播放器",
  197. enhanceTweetEngagements: "增强推文互动",
  198. },
  199. weeks: ["周日", "周一", "周二", "周三", "周四", "周五", "周六"],
  200. },
  201. ko: {
  202. panel: {
  203. replaceIcons: "Twitter 아이콘 교체",
  204. useAbsoluteTime: "절대 시간 사용",
  205. showTimeAndDateSidebar: "시간 및 날짜 사이드바 표시",
  206. useDefaultVideoPlayer: "기본 비디오 플레이어 사용",
  207. enhanceTweetEngagements: "트윗 참여 향상",
  208. },
  209. weeks: ["일", "월", "화", "수", "목", "금", "토"],
  210. },
  211. ru: {
  212. panel: {
  213. replaceIcons: "Заменить иконки Twitter",
  214. useAbsoluteTime: "Использовать абсолютное время",
  215. showTimeAndDateSidebar: "Показать боковую панель времени и даты",
  216. useDefaultVideoPlayer: "Использовать стандартный видеоплеер",
  217. enhanceTweetEngagements: "Улучшить взаимодействие с твитами",
  218. },
  219. weeks: ["Вс", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб"],
  220. },
  221. de: {
  222. panel: {
  223. replaceIcons: "Twitter-Icons ersetzen",
  224. useAbsoluteTime: "Absolute Zeit verwenden",
  225. showTimeAndDateSidebar: "Zeit- und Datums-Sidebar anzeigen",
  226. useDefaultVideoPlayer: "Standard-Video-Player verwenden",
  227. enhanceTweetEngagements: "Tweet-Interaktionen verbessern",
  228. },
  229. weeks: ["So", "Mo", "Di", "Mi", "Do", "Fr", "Sa"],
  230. },
  231. };
  232.  
  233. const LANG = navigator.language.split("-")[0];
  234. const CURRENT_LANG = TRANSLATIONS[LANG] || TRANSLATIONS.en;
  235.  
  236. const PANEL_LANG = CURRENT_LANG.panel;
  237. const WEEKS_LANG = CURRENT_LANG.weeks;
  238.  
  239. // -----------------------------------------------------------------------------------
  240. // 設定パネル
  241. // -----------------------------------------------------------------------------------
  242. const config = {
  243. replaceIcons: true,
  244. useAbsoluteTime: true,
  245. showTimeAndDateSidebar: true,
  246. useDefaultVideoPlayer: true,
  247. enhanceTweetEngagements: true,
  248. };
  249.  
  250. const SettingsModule = {
  251. createSettingsUI: function () {
  252. const settingsDiv = Utils.createElement("div", {
  253. id: "twitter-kaizen-panel",
  254. classList: ["twitter-kaizen-panel"],
  255. });
  256.  
  257. // パネルのインラインスタイルを追加
  258. Object.assign(settingsDiv.style, {
  259. position: "fixed",
  260. top: "10px",
  261. right: "10px",
  262. zIndex: "9999",
  263. background: "#f9f9f9",
  264. padding: "15px",
  265. border: "1px solid #ccc",
  266. borderRadius: "10px",
  267. boxShadow: "0 4px 12px rgba(0, 0, 0, 0.1)",
  268. color: "#333",
  269. fontFamily: "Arial, sans-serif",
  270. width: "300px",
  271. maxWidth: "100%",
  272. display: "none",
  273. transition: "transform 0.3s ease, opacity 0.3s ease",
  274. });
  275.  
  276. const title = Utils.createElement("h3", {
  277. textContent: "Twitter Kaizen Settings",
  278. });
  279. title.style.fontSize = "18px";
  280. title.style.margin = "10px";
  281. title.style.color = "#333";
  282. settingsDiv.appendChild(title);
  283.  
  284. const features = [
  285. { key: "replaceIcons", label: PANEL_LANG.replaceIcons },
  286. { key: "useAbsoluteTime", label: PANEL_LANG.useAbsoluteTime },
  287. {
  288. key: "showTimeAndDateSidebar",
  289. label: PANEL_LANG.showTimeAndDateSidebar,
  290. },
  291. {
  292. key: "useDefaultVideoPlayer",
  293. label: PANEL_LANG.useDefaultVideoPlayer,
  294. },
  295. {
  296. key: "enhanceTweetEngagements",
  297. label: PANEL_LANG.enhanceTweetEngagements,
  298. },
  299. ];
  300.  
  301. features.forEach(({ key, label }) => {
  302. const checkbox = Utils.createElement("input", {
  303. attributes: { type: "checkbox", id: key },
  304. });
  305. checkbox.checked = config[key];
  306. checkbox.addEventListener("change", () => {
  307. config[key] = checkbox.checked;
  308. saveConfig();
  309. location.reload();
  310. });
  311.  
  312. const labelElement = Utils.createElement("label", {
  313. attributes: { for: key },
  314. textContent: label,
  315. });
  316. labelElement.style.marginLeft = "8px";
  317. labelElement.style.fontSize = "14px";
  318. labelElement.style.color = "#555";
  319.  
  320. settingsDiv.appendChild(checkbox);
  321. settingsDiv.appendChild(labelElement);
  322. settingsDiv.appendChild(Utils.createElement("br"));
  323. });
  324.  
  325. document.body.appendChild(settingsDiv);
  326. },
  327.  
  328. toggleSettingsPanel: function () {
  329. const panel = document.getElementById("twitter-kaizen-panel");
  330. if (panel) {
  331. if (panel.style.display === "none") {
  332. panel.style.display = "block";
  333. panel.style.transform = "scale(1)";
  334. panel.style.opacity = "1";
  335. } else {
  336. panel.style.transform = "scale(0.9)";
  337. panel.style.opacity = "0";
  338. setTimeout(() => {
  339. panel.style.display = "none";
  340. }, 300);
  341. }
  342. }
  343. },
  344. };
  345.  
  346. // ショートカットキー
  347. function setupKeyboardShortcut() {
  348. document.addEventListener("keydown", function (e) {
  349. if (e.ctrlKey && e.altKey && e.key === "o") {
  350. SettingsModule.toggleSettingsPanel();
  351. }
  352. });
  353. }
  354.  
  355. // メニューコマンドの登録
  356. function setupMenuCommand() {
  357. GM_registerMenuCommand("Toggle Twitter Kaizen Settings", () => {
  358. SettingsModule.toggleSettingsPanel();
  359. });
  360. }
  361.  
  362. // -----------------------------------------------------------------------------------
  363. // Twitterを取り戻す(アイコンを戻す)
  364. // -----------------------------------------------------------------------------------
  365.  
  366. function replaceTwitterIcons() {
  367. if (!config.replaceIcons) return;
  368.  
  369. GM_addStyle(`
  370. /* main */
  371. .r-64el8z[href="/home"] > div > svg > g > path,
  372. .r-1h3ijdo > .r-1pi2tsx > svg > g > path,
  373. .r-1blnp2b > g > path {
  374. d: path(
  375. "M23.643 4.937c-.835.37-1.732.62-2.675.733.962-.576 1.7-1.49 2.048-2.578-.9.534-1.897.922-2.958 1.13-.85-.904-2.06-1.47-3.4-1.47-2.572 0-4.658 2.086-4.658 4.66 0 .364.042.718.12 1.06-3.873-.195-7.304-2.05-9.602-4.868-.4.69-.63 1.49-.63 2.342 0 1.616.823 3.043 2.072 3.878-.764-.025-1.482-.234-2.11-.583v.06c0 2.257 1.605 4.14 3.737 4.568-.392.106-.803.162-1.227.162-.3 0-.593-.028-.877-.082.593 1.85 2.313 3.198 4.352 3.234-1.595 1.25-3.604 1.995-5.786 1.995-.376 0-.747-.022-1.112-.065 2.062 1.323 4.51 2.093 7.14 2.093 8.57 0 13.255-7.098 13.255-13.254 0-.2-.005-.402-.014-.602.91-.658 1.7-1.477 2.323-2.41z"
  376. ) !important;
  377. }
  378. /* premium */
  379. .r-eqz5dr[href="/i/premium_sign_up"] > div > div > svg > g > path,
  380. .r-1loqt21[href="/i/premium_sign_up"] > div > svg > g > path {
  381. d: path(
  382. "M 8.52 3.59 c 0.8 -1.1 2.04 -1.84 3.48 -1.84 s 2.68 0.74 3.49 1.84 c 1.34 -0.21 2.74 0.14 3.76 1.16 s 1.37 2.42 1.16 3.77 c 1.1 0.8 1.84 2.04 1.84 3.48 s -0.74 2.68 -1.84 3.48 c 0.21 1.34 -0.14 2.75 -1.16 3.77 s -2.42 1.37 -3.76 1.16 c -0.8 1.1 -2.05 1.84 -3.49 1.84 s -2.68 -0.74 -3.48 -1.84 c -1.34 0.21 -2.75 -0.14 -3.77 -1.16 c -1.01 -1.02 -1.37 -2.42 -1.16 -3.77 c -1.09 -0.8 -1.84 -2.04 -1.84 -3.48 s 0.75 -2.68 1.84 -3.48 c -0.21 -1.35 0.14 -2.75 1.16 -3.77 s 2.43 -1.37 3.77 -1.16 Z m 3.48 0.16 c -0.85 0 -1.66 0.53 -2.12 1.43 l -0.38 0.77 l -0.82 -0.27 c -0.96 -0.32 -1.91 -0.12 -2.51 0.49 c -0.6 0.6 -0.8 1.54 -0.49 2.51 l 0.27 0.81 l -0.77 0.39 c -0.9 0.46 -1.43 1.27 -1.43 2.12 s 0.53 1.66 1.43 2.12 l 0.77 0.39 l -0.27 0.81 c -0.31 0.97 -0.11 1.91 0.49 2.51 c 0.6 0.61 1.55 0.81 2.51 0.49 l 0.82 -0.27 l 0.38 0.77 c 0.46 0.9 1.27 1.43 2.12 1.43 s 1.66 -0.53 2.12 -1.43 l 0.39 -0.77 l 0.82 0.27 c 0.96 0.32 1.9 0.12 2.51 -0.49 c 0.6 -0.6 0.8 -1.55 0.48 -2.51 l -0.26 -0.81 l 0.76 -0.39 c 0.91 -0.46 1.43 -1.27 1.43 -2.12 s -0.52 -1.66 -1.43 -2.12 l -0.77 -0.39 l 0.27 -0.81 c 0.32 -0.97 0.12 -1.91 -0.48 -2.51 c -0.61 -0.61 -1.55 -0.81 -2.51 -0.49 l -0.82 0.27 l -0.39 -0.77 c -0.46 -0.9 -1.27 -1.43 -2.12 -1.43 Z m 4.74 5.68 l -6.2 6.77 l -3.74 -3.74 l 1.41 -1.42 l 2.26 2.26 l 4.8 -5.23 l 1.47 1.36 Z"
  383. ) !important;
  384. }
  385. /* home */
  386. .r-eqz5dr[href="/home"] > div > div > svg > g > path {
  387. d: path(
  388. "M12,1.696 L0.622,8.807l1.06,1.696L3,9.679V19.5C3,20.881 4.119,22 5.5,22h13c1.381,0 2.5,-1.119 2.5,-2.5V9.679l1.318,0.824 1.06,-1.696L12,1.696ZM12,16.5c-1.933,0 -3.5,-1.567 -3.5,-3.5s1.567,-3.5 3.5,-3.5 3.5,1.567 3.5,3.5 -1.567,3.5 -3.5,3.5Z"
  389. ) !important;
  390. }
  391. `);
  392. }
  393.  
  394. // -----------------------------------------------------------------------------------
  395. // TLの時間を相対時間から絶対時間に変更(HH:MM:SS・mm/dd/yy, week)
  396. // -----------------------------------------------------------------------------------
  397. // タイムスタンプモジュール
  398. const TimestampModule = {
  399. toFormattedDateString: function (date) {
  400. const YEAR = date.getFullYear().toString().slice(-2);
  401. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  402. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  403. return `${TIME}・${DATE}`;
  404. },
  405. // タイムスタンプの更新
  406. updateTimestamps: function () {
  407. if (!config.useAbsoluteTime) return;
  408.  
  409. const timeSelectors =
  410. 'main div[data-testid="primaryColumn"] section article a[href*="/status/"] time, div.css-175oi2r.r-18u37iz.r-1q142lx div.css-175oi2r.r-1d09ksm.r-18u37iz.r-1wbh5a2 time';
  411.  
  412. document.querySelectorAll(timeSelectors).forEach((timeElement) => {
  413. const parent = timeElement.parentNode;
  414. const span = Utils.createElement("span", {
  415. textContent: this.toFormattedDateString(
  416. new Date(timeElement.getAttribute("datetime"))
  417. ),
  418. });
  419. span.style.pointerEvents = "none";
  420. parent.appendChild(span);
  421. parent.removeChild(timeElement);
  422. });
  423. },
  424. };
  425.  
  426. // -----------------------------------------------------------------------------------
  427. // サイドバーに時間、日付を表示(HH:MM:SS, mm/dd/yy, week)
  428. // -----------------------------------------------------------------------------------
  429. const SidebarModule = {
  430. createInfoElement: function (type) {
  431. if (!config.showTimeAndDateSidebar) return;
  432.  
  433. const nav = document.querySelector(
  434. 'div[class="css-175oi2r r-vacyoi r-ttdzmv"]'
  435. );
  436. if (!nav || document.getElementById(type)) return;
  437.  
  438. const iconHTML =
  439. type === "time"
  440. ? '<i class="fa-regular fa-clock" style="width: 26.25px; height: 26.25px;"></i>'
  441. : '<i class="fa-solid fa-calendar-days" style="width: 26.25px; height: 26.25px;"></i>';
  442.  
  443. const textContentFunc = () => {
  444. const date = new Date();
  445. const YEAR = date.getFullYear().toString().slice(-2);
  446. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  447. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  448.  
  449. return type === "time" ? `${TIME}` : `${DATE}`;
  450. };
  451.  
  452. const infoElement = Utils.createElement("div", {
  453. id: type,
  454. classList: [
  455. "css-175oi2r",
  456. "r-6koalj",
  457. "r-eqz5dr",
  458. "r-16y2uox",
  459. "r-1habvwh",
  460. "r-cnw61z",
  461. "r-13qz1uu",
  462. "r-1loqt21",
  463. "r-1ny4l3l",
  464. ],
  465. });
  466.  
  467. const container = Utils.createElement("div", {
  468. id: `${type}__container`,
  469. classList: [
  470. "css-175oi2r",
  471. "r-sdzlij",
  472. "r-dnmrzs",
  473. "r-1awozwy",
  474. "r-18u37iz",
  475. "r-1777fci",
  476. "r-xyw6el",
  477. "r-o7ynqc",
  478. "r-6416eg",
  479. ],
  480. });
  481.  
  482. const icon = Utils.createElement("div", {
  483. id: `${type}__container__icon`,
  484. classList: ["css-175oi2r"],
  485. innerHTML: iconHTML,
  486. });
  487.  
  488. const text = Utils.createElement("div", {
  489. id: `${type}__container__text`,
  490. classList: [
  491. "css-146c3p1",
  492. "r-dnmrzs",
  493. "r-1udh08x",
  494. "r-3s2u2q",
  495. "r-bcqeeo",
  496. "r-1ttztb7",
  497. "r-qvutc0",
  498. "r-1qd0xha",
  499. "r-adyw6z",
  500. "r-135wba7",
  501. "r-16dba41",
  502. "r-dlybji",
  503. "r-nazi8o",
  504. ],
  505. });
  506.  
  507. const textContent = Utils.createElement("span", {
  508. id: `${type}__text__content`,
  509. classList: ["1jxf684", "r-bcqeeo", "r-1ttztb7", "r-qvutc0", "r-poiln3"],
  510. textContent: textContentFunc(),
  511. });
  512.  
  513. text.appendChild(textContent);
  514. container.appendChild(icon);
  515. container.appendChild(text);
  516. infoElement.appendChild(container);
  517. nav.appendChild(infoElement);
  518.  
  519. if (type === "time") {
  520. setInterval(() => {
  521. textContent.textContent = textContentFunc();
  522. }, 1000);
  523. }
  524. },
  525.  
  526. init: function () {
  527. this.createInfoElement("time");
  528. this.createInfoElement("date");
  529.  
  530. const observer = new MutationObserver(() => {
  531. this.createInfoElement("time");
  532. this.createInfoElement("date");
  533. });
  534.  
  535. observer.observe(document.body, { childList: true, subtree: true });
  536. },
  537. };
  538.  
  539. // -----------------------------------------------------------------------------------
  540. // 動画プレイヤーをデフォルトに戻す
  541. // -----------------------------------------------------------------------------------
  542. const VideoModule = {
  543. setupDefaultVideoPlayer: function (container) {
  544. if (!config.useDefaultVideoPlayer) return;
  545.  
  546. const video = container.querySelector("div:first-child > div > video");
  547. if (!video) return;
  548.  
  549. video.controls = true;
  550. video.removeAttribute("disablepictureinpicture");
  551. video.muted = false;
  552.  
  553. const onClick = (e) => {
  554. e.preventDefault();
  555. video
  556. .play()
  557. .then(() => {
  558. video.muted = false;
  559. })
  560. .catch((error) => console.error("Video playback error:", error));
  561.  
  562. const onVolumeChange = (e) => {
  563. if (e.target.muted) {
  564. e.target.muted = false;
  565. }
  566. e.target.removeEventListener("volumechange", onVolumeChange);
  567. };
  568.  
  569. e.target.addEventListener("volumechange", onVolumeChange);
  570. video.removeEventListener("click", onClick);
  571. };
  572.  
  573. video.addEventListener("click", onClick);
  574.  
  575. container.parentElement.appendChild(video);
  576. container.remove();
  577. },
  578.  
  579. observeVideos: function () {
  580. const observer = new MutationObserver(() => {
  581. const videoContainer = document.body.querySelector(
  582. 'div[data-testid="videoComponent"]:not(.enhanced-video)'
  583. );
  584. if (videoContainer) {
  585. videoContainer.classList.add("enhanced-video");
  586. setTimeout(() => this.setupDefaultVideoPlayer(videoContainer), 100);
  587. }
  588. });
  589.  
  590. observer.observe(document.body, { subtree: true, childList: true });
  591. },
  592. };
  593.  
  594. // -----------------------------------------------------------------------------------
  595. // Tweet Engagements をアクセスしやすく
  596. // -----------------------------------------------------------------------------------
  597. const TweetEngagementModule = {
  598. createQuoteButton: function () {
  599. const buttonWrapper = Utils.createElement("div", {
  600. classList: ["css-175oi2r", "r-18u37iz", "r-1h0z5md", "r-13awgt0"],
  601. });
  602.  
  603. const link = Utils.createElement("a", {
  604. attributes: {
  605. href: `${window.location.pathname}/quotes`,
  606. "data-testid": "tweetEngagements",
  607. target: "_blank",
  608. rel: "noopener",
  609. },
  610. classList: [
  611. "css-175oi2r",
  612. "r-1777fci",
  613. "r-bt1l66",
  614. "r-bztko3",
  615. "r-lrvibr",
  616. "r-1loqt21",
  617. "r-1ny4l3l",
  618. ],
  619. });
  620.  
  621. link.addEventListener("click", (event) => {
  622. const href = event.currentTarget.getAttribute("href");
  623. window.open(href, "_blank");
  624. });
  625.  
  626. const contentDiv = Utils.createElement("div", {
  627. attributes: { dir: "ltr" },
  628. classList: [
  629. "css-146c3p1",
  630. "r-bcqeeo",
  631. "r-1ttztb7",
  632. "r-qvutc0",
  633. "r-1qd0xha",
  634. "r-a023e6",
  635. "r-rjixqe",
  636. "r-16dba41",
  637. "r-1awozwy",
  638. "r-6koalj",
  639. "r-1h0z5md",
  640. "r-o7ynqc",
  641. "r-clp7b1",
  642. "r-3s2u2q",
  643. ],
  644. });
  645. contentDiv.style.textOverflow = "unset";
  646. contentDiv.style.color = "rgb(113, 118, 123)";
  647.  
  648. const iconDiv = Utils.createElement("div", {
  649. classList: ["css-175oi2r", "r-xoduu5"],
  650. innerHTML: `
  651. <div class="css-175oi2r r-xoduu5 r-1p0dtai r-1d2f490 r-u8s1d r-zchlnj r-ipm5af r-1niwhzg r-sdzlij r-xf4iuw r-o7ynqc r-6416eg r-1ny4l3l"></div>
  652. <svg viewBox="0 0 24 24" aria-hidden="true" class="r-4qtqp9 r-yyyyoo r-dnmrzs r-bnwqim r-lrvibr r-m6rgpd r-50lct3 r-1srniue">
  653. <g><path d="M8.75 21V3h2v18h-2zM18 21V8.5h2V21h-2zM4 21l.004-10h2L6 21H4zm9.248 0v-7h2v7h-2z"></path></g>
  654. </svg>
  655. `,
  656. });
  657.  
  658. // contentDiv にホバーイベントを追加
  659. contentDiv.addEventListener("mouseenter", () => {
  660. contentDiv.style.color = "rgb(238 201 104)";
  661. const iconBgDiv = iconDiv.querySelector("div");
  662. if (iconBgDiv) {
  663. iconBgDiv.style.backgroundColor = "rgba(238, 201, 104, 0.1)";
  664. }
  665. });
  666.  
  667. contentDiv.addEventListener("mouseleave", () => {
  668. contentDiv.style.color = "rgb(113, 118, 123)";
  669.  
  670. const iconBgDiv = iconDiv.querySelector("div");
  671. if (iconBgDiv) {
  672. iconBgDiv.style.backgroundColor = "";
  673. }
  674. });
  675.  
  676. const countDiv = Utils.createElement("div", {
  677. classList: ["css-175oi2r", "r-xoduu5", "r-1udh08x"],
  678. innerHTML: `
  679. <span data-testid="app-text-transition-container" style="transition-property: transform; transition-duration: 0.3s; transform: translate3d(0px, 0px, 0px);">
  680. <span class="css-1jxf684 r-1ttztb7 r-qvutc0 r-poiln3 r-n6v787 r-1cwl3u0 r-1k6nrdp r-n7gxbd" style="text-overflow: unset">
  681. <span class="css-1jxf684 r-bcqeeo r-1ttztb7 r-qvutc0 r-poiln3" style="text-overflow: unset">Quotes</span>
  682. </span>
  683. </span>
  684. `,
  685. });
  686.  
  687. contentDiv.appendChild(iconDiv);
  688. contentDiv.appendChild(countDiv);
  689. link.appendChild(contentDiv);
  690. buttonWrapper.appendChild(link);
  691.  
  692. return buttonWrapper;
  693. },
  694.  
  695. // 投稿ページかどうかを判定
  696. isTweetPage: function () {
  697. return (
  698. document.querySelector(
  699. "div.css-175oi2r.r-1kbdv8c.r-18u37iz.r-1oszu61.r-3qxfft.r-n7gxbd.r-2sztyj.r-1efd50x.r-5kkj8d.r-h3s6tt.r-1wtj0ep"
  700. ) !== null
  701. );
  702. },
  703.  
  704. addQuoteElement: function () {
  705. // 投稿ページの場合ボタンを表示
  706. if (!this.isTweetPage()) {
  707. return;
  708. }
  709.  
  710. const targetDiv = document.querySelector('div[role="group"][id^="id__"]');
  711. if (
  712. !targetDiv ||
  713. targetDiv.querySelector('[data-testid="tweetEngagements"]')
  714. ) {
  715. return;
  716. }
  717.  
  718. const quoteButton = this.createQuoteButton();
  719. targetDiv.insertBefore(quoteButton, targetDiv.children[4]);
  720. },
  721.  
  722. init: function () {
  723. const debouncedAddQuoteElement = Utils.debounce(
  724. () => this.addQuoteElement(),
  725. 250
  726. );
  727.  
  728. // URL変更の監視
  729. let lastUrl = location.href;
  730. const urlObserver = new MutationObserver(() => {
  731. const url = location.href;
  732. if (url !== lastUrl) {
  733. lastUrl = url;
  734. debouncedAddQuoteElement();
  735. }
  736. });
  737.  
  738. urlObserver.observe(document, { subtree: true, childList: true });
  739.  
  740. const retryInterval = setInterval(debouncedAddQuoteElement, 1000);
  741.  
  742. window.addEventListener("popstate", debouncedAddQuoteElement);
  743. history.pushState = ((origPushState) => {
  744. return function (state, title, url) {
  745. origPushState.apply(this, arguments);
  746. debouncedAddQuoteElement();
  747. };
  748. })(history.pushState);
  749.  
  750. history.replaceState = ((origReplaceState) => {
  751. return function (state, title, url) {
  752. origReplaceState.apply(this, arguments);
  753. debouncedAddQuoteElement();
  754. };
  755. })(history.replaceState);
  756.  
  757. // DOMContentLoaded で初期化
  758. document.addEventListener("DOMContentLoaded", () => {
  759. debouncedAddQuoteElement();
  760. clearInterval(retryInterval);
  761. });
  762.  
  763. debouncedAddQuoteElement();
  764. },
  765. };
  766.  
  767. // -----------------------------------------------------------------------------------
  768. // メイン処理
  769. // -----------------------------------------------------------------------------------
  770. function main() {
  771. loadConfig();
  772. SettingsModule.createSettingsUI();
  773. setupKeyboardShortcut();
  774. setupMenuCommand();
  775.  
  776. // タイムスタンプの更新を定期的に実行
  777. setInterval(() => TimestampModule.updateTimestamps(), 1000);
  778.  
  779. // アイコン情報表示を初期化
  780. replaceTwitterIcons();
  781.  
  782. // サイドバーの情報表示を初期化
  783. SidebarModule.init();
  784.  
  785. // 動画プレイヤーの設定を監視
  786. VideoModule.observeVideos();
  787.  
  788. // 引用ツイートボタンの追加を初期化
  789. TweetEngagementModule.init();
  790. }
  791.  
  792. // ページ読み込み時にメイン処理を実行
  793. window.addEventListener("load", main);
  794. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址