Twitter kaizen

Enjoy Twitter Comfortably

  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 Enjoy Twitter Comfortably
  10. // @description:ja ツイッターを快適に
  11. // @description:en Script to improve Twitter display
  12. // @description:zh-CN 舒适地使用推特
  13. // @description:ko 트위터를 편안하게
  14. // @description:ru Комфортное использование Твиттера
  15. // @description:de Twitter bequem nutzen
  16. // @version 2.6.2
  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. /* light */
  37. .r-jxzhtn /* basic */,
  38. .r-1igl3o0, /* tl */
  39. /* gray */
  40. .r-18bvks7 /* basic */,
  41. .r-1ila09b /* tl */,
  42. /* dark */
  43. .r-1kqtdi0 /* basic */,
  44. .r-j5o65s /* tl */ {
  45. border: none !important;
  46. }
  47. /* -----------------------------------------------------------------------------------
  48. カキコの下のボーダーを消す
  49. ----------------------------------------------------------------------------------- */
  50. .r-109y4c4 {
  51. height: 0 !important;
  52. }
  53. /* -----------------------------------------------------------------------------------
  54. TLの幅を600pxから700pxに、右サイドバーの幅を350pxから250pxに変更
  55. ----------------------------------------------------------------------------------- */
  56. .r-1ye8kvj {
  57. max-width: 700px !important;
  58. }
  59. .r-1hycxz {
  60. width: 250px !important;
  61. }
  62. .css-175oi2r.r-kemksi.r-1kqtdi0.r-th6na.r-1phboty.r-1dqxon3.r-1hycxz {
  63. width: 350px !important;
  64. }
  65. /* -----------------------------------------------------------------------------------
  66. ヘッダーのスクロールバーを消す
  67. ----------------------------------------------------------------------------------- */
  68. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd {
  69. overflow-y: scroll !important;
  70. -ms-overflow-style: none !important;
  71. scrollbar-width: none !important;
  72. }
  73. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd::-webkit-scrollbar {
  74. display: none !important;
  75. }
  76. /* -----------------------------------------------------------------------------------
  77. サイドバーの”Subscribe to Premium”を消す
  78. ----------------------------------------------------------------------------------- */
  79. .css-175oi2r.r-1habvwh.r-eqz5dr.r-uaa2di.r-1mmae3n.r-3pj75a.r-bnwqim {
  80. display: none !important;
  81. }
  82. /* -----------------------------------------------------------------------------------
  83. サイドバーの”Who to follow”を消す
  84. ----------------------------------------------------------------------------------- */
  85. .css-175oi2r.r-1bro5k0 {
  86. display: none !important;
  87. }
  88. /* -----------------------------------------------------------------------------------
  89. TL上のUserNameを消す
  90. ----------------------------------------------------------------------------------- */
  91. div[data-testid="User-Name"] > div:nth-child(2) > div > div:nth-child(1),
  92. div[data-testid="User-Name"] > div:nth-child(2) > div > div:nth-child(2) {
  93. display: none !important;
  94. }
  95.  
  96. /* -----------------------------------------------------------------------------------
  97. TL上のアカウント名と日付を縦並びにする
  98. ----------------------------------------------------------------------------------- */
  99. div[data-testid="User-Name"] {
  100. align-items: initial !important;
  101. flex-direction: column !important;
  102. }
  103. div[data-testid="User-Name"] > div:last-child {
  104. margin-left: 0 !important;
  105. }
  106. /* -----------------------------------------------------------------------------------
  107. 時計、日付のフォントカラーを変更 (何かしらの理由で背景色を変えてる場合を考えて 'color-scheme' の指定も追加)
  108. ----------------------------------------------------------------------------------- */
  109. /* light */
  110. html[style*="color-scheme: light;"] #date__container__text,
  111. html[style*="color-scheme: light;"] #time__container__text,
  112. body[style*="background-color: rgb(255, 255, 255);"] #date__container__text,
  113. body[style*="background-color: rgb(255, 255, 255);"] #time__container__text {
  114. color: #0f1419;
  115. }
  116. /* gray */
  117. body[style*="background-color: rgb(21, 32, 43);"] #date__container__text,
  118. body[style*="background-color: rgb(21, 32, 43);"] #time__container__text {
  119. color: #f7f9f9;
  120. }
  121. /* dark */
  122. html[style*="color-scheme: dark;"] #date__container__text,
  123. html[style*="color-scheme: dark;"] #time__container__text,
  124. body[style*="background-color: rgb(0, 0, 0);"] #date__container__text,
  125. body[style*="background-color: rgb(0, 0, 0);"] #time__container__text {
  126. color: #e7e9ea;
  127. }
  128. `);
  129.  
  130. // ローカルストレージから設定を読み込む
  131. function loadConfig() {
  132. const savedConfig = localStorage.getItem("twitterKaizenConfig");
  133. if (savedConfig) {
  134. Object.assign(config, JSON.parse(savedConfig));
  135. }
  136. }
  137.  
  138. // ローカルストレージに設定を保存
  139. function saveConfig() {
  140. localStorage.setItem("twitterKaizenConfig", JSON.stringify(config));
  141. }
  142.  
  143. // -----------------------------------------------------------------------------------
  144. // ユーティリティ関数と定数
  145. // -----------------------------------------------------------------------------------
  146. const Utils = {
  147. debounce: (func, wait) => {
  148. let timeout;
  149. return (...args) => {
  150. clearTimeout(timeout);
  151. timeout = setTimeout(() => func(...args), wait);
  152. };
  153. },
  154.  
  155. pad: (num) => num.toString().padStart(2, "0"),
  156.  
  157. createElement: (tag, options = {}) => {
  158. const element = document.createElement(tag);
  159. if (options.id) element.id = options.id;
  160. options.classList?.forEach((cls) => element.classList.add(cls));
  161. Object.entries(options.attributes || {}).forEach(([attr, value]) =>
  162. element.setAttribute(attr, value)
  163. );
  164. if (options.innerHTML) element.innerHTML = options.innerHTML;
  165. if (options.textContent) element.textContent = options.textContent;
  166. return element;
  167. },
  168.  
  169. observeDOM: (
  170. targetNode,
  171. callback,
  172. config = { childList: true, subtree: true }
  173. ) => {
  174. const observer = new MutationObserver(callback);
  175. observer.observe(targetNode, config);
  176. return observer;
  177. },
  178. };
  179.  
  180. // 多言語定義
  181. const TRANSLATIONS = {
  182. en: {
  183. panel: {
  184. replaceIcons: "Reclaim Twitter (restore icon)",
  185. useAbsoluteTime: "Change TL time from relative to absolute time",
  186. showTimeAndDateSidebar: "Display time and date in sidebar",
  187. useDefaultVideoPlayer: "Revert video player to default",
  188. enhanceTweetEngagements: "Easy access to quoted tweets",
  189. },
  190. weeks: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
  191. },
  192. ja: {
  193. panel: {
  194. replaceIcons: "Twitterを取り戻す (アイコンを元に戻す)",
  195. useAbsoluteTime: "TLの時間を相対時間から絶対時間に変更",
  196. showTimeAndDateSidebar: "サイドバーに時間、日付を表示",
  197. useDefaultVideoPlayer: "動画プレイヤーをデフォルトに戻す",
  198. enhanceTweetEngagements: "引用ツイートへのアクセスを簡単に",
  199. },
  200. weeks: ["日", "月", "火", "水", "木", "金", "土"],
  201. },
  202. zh: {
  203. panel: {
  204. replaceIcons: "替换 Twitter 图标",
  205. useAbsoluteTime: "使用绝对时间",
  206. showTimeAndDateSidebar: "显示时间和日期侧边栏",
  207. useDefaultVideoPlayer: "使用默认视频播放器",
  208. enhanceTweetEngagements: "增强推文互动",
  209. },
  210. weeks: ["周日", "周一", "周二", "周三", "周四", "周五", "周六"],
  211. },
  212. ko: {
  213. panel: {
  214. replaceIcons: "Twitter 아이콘 교체",
  215. useAbsoluteTime: "절대 시간 사용",
  216. showTimeAndDateSidebar: "시간 및 날짜 사이드바 표시",
  217. useDefaultVideoPlayer: "기본 비디오 플레이어 사용",
  218. enhanceTweetEngagements: "트윗 참여 향상",
  219. },
  220. weeks: ["일", "월", "화", "수", "목", "금", "토"],
  221. },
  222. ru: {
  223. panel: {
  224. replaceIcons: "Заменить иконки Twitter",
  225. useAbsoluteTime: "Использовать абсолютное время",
  226. showTimeAndDateSidebar: "Показать боковую панель времени и даты",
  227. useDefaultVideoPlayer: "Использовать стандартный видеоплеер",
  228. enhanceTweetEngagements: "Улучшить взаимодействие с твитами",
  229. },
  230. weeks: ["Вс", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб"],
  231. },
  232. de: {
  233. panel: {
  234. replaceIcons: "Twitter-Icons ersetzen",
  235. useAbsoluteTime: "Absolute Zeit verwenden",
  236. showTimeAndDateSidebar: "Zeit- und Datums-Sidebar anzeigen",
  237. useDefaultVideoPlayer: "Standard-Video-Player verwenden",
  238. enhanceTweetEngagements: "Tweet-Interaktionen verbessern",
  239. },
  240. weeks: ["So", "Mo", "Di", "Mi", "Do", "Fr", "Sa"],
  241. },
  242. };
  243.  
  244. const LANG = navigator.language.split("-")[0];
  245. const CURRENT_LANG = TRANSLATIONS[LANG] || TRANSLATIONS.en;
  246.  
  247. const PANEL_LANG = CURRENT_LANG.panel;
  248. const WEEKS_LANG = CURRENT_LANG.weeks;
  249.  
  250. // -----------------------------------------------------------------------------------
  251. // 設定パネル
  252. // -----------------------------------------------------------------------------------
  253. const config = {
  254. replaceIcons: true,
  255. useAbsoluteTime: true,
  256. showTimeAndDateSidebar: true,
  257. useDefaultVideoPlayer: true,
  258. enhanceTweetEngagements: true,
  259. };
  260.  
  261. const SettingsModule = {
  262. createSettingsUI: function () {
  263. const settingsDiv = Utils.createElement("div", {
  264. id: "twitter-kaizen-panel",
  265. classList: ["twitter-kaizen-panel"],
  266. });
  267.  
  268. // パネルのインラインスタイルを追加
  269. Object.assign(settingsDiv.style, {
  270. position: "fixed",
  271. top: "10px",
  272. right: "10px",
  273. zIndex: "9999",
  274. background: "#f9f9f9",
  275. padding: "15px",
  276. border: "1px solid #ccc",
  277. borderRadius: "10px",
  278. boxShadow: "0 4px 12px rgba(0, 0, 0, 0.1)",
  279. color: "#333",
  280. fontFamily: "Arial, sans-serif",
  281. width: "300px",
  282. maxWidth: "100%",
  283. display: "none",
  284. transition: "transform 0.3s ease, opacity 0.3s ease",
  285. });
  286.  
  287. const title = Utils.createElement("h3", {
  288. textContent: "Twitter Kaizen Settings",
  289. });
  290. title.style.fontSize = "18px";
  291. title.style.margin = "10px";
  292. title.style.color = "#333";
  293. settingsDiv.appendChild(title);
  294.  
  295. const features = [
  296. { key: "replaceIcons", label: PANEL_LANG.replaceIcons },
  297. { key: "useAbsoluteTime", label: PANEL_LANG.useAbsoluteTime },
  298. {
  299. key: "showTimeAndDateSidebar",
  300. label: PANEL_LANG.showTimeAndDateSidebar,
  301. },
  302. {
  303. key: "useDefaultVideoPlayer",
  304. label: PANEL_LANG.useDefaultVideoPlayer,
  305. },
  306. {
  307. key: "enhanceTweetEngagements",
  308. label: PANEL_LANG.enhanceTweetEngagements,
  309. },
  310. ];
  311.  
  312. features.forEach(({ key, label }) => {
  313. const checkbox = Utils.createElement("input", {
  314. attributes: { type: "checkbox", id: key },
  315. });
  316. checkbox.checked = config[key];
  317. checkbox.addEventListener("change", () => {
  318. config[key] = checkbox.checked;
  319. saveConfig();
  320. location.reload();
  321. });
  322.  
  323. const labelElement = Utils.createElement("label", {
  324. attributes: { for: key },
  325. textContent: label,
  326. });
  327. labelElement.style.marginLeft = "8px";
  328. labelElement.style.fontSize = "14px";
  329. labelElement.style.color = "#555";
  330.  
  331. settingsDiv.appendChild(checkbox);
  332. settingsDiv.appendChild(labelElement);
  333. settingsDiv.appendChild(Utils.createElement("br"));
  334. });
  335.  
  336. document.body.appendChild(settingsDiv);
  337. },
  338.  
  339. toggleSettingsPanel: function () {
  340. const panel = document.getElementById("twitter-kaizen-panel");
  341. if (panel) {
  342. if (panel.style.display === "none") {
  343. panel.style.display = "block";
  344. panel.style.transform = "scale(1)";
  345. panel.style.opacity = "1";
  346. } else {
  347. panel.style.transform = "scale(0.9)";
  348. panel.style.opacity = "0";
  349. setTimeout(() => {
  350. panel.style.display = "none";
  351. }, 300);
  352. }
  353. }
  354. },
  355. };
  356.  
  357. // ショートカットキー
  358. function setupKeyboardShortcut() {
  359. document.addEventListener("keydown", function (e) {
  360. if (e.ctrlKey && e.altKey && e.key === "o") {
  361. SettingsModule.toggleSettingsPanel();
  362. }
  363. });
  364. }
  365.  
  366. // メニューコマンドの登録
  367. function setupMenuCommand() {
  368. GM_registerMenuCommand("Toggle Twitter Kaizen Settings", () => {
  369. SettingsModule.toggleSettingsPanel();
  370. });
  371. }
  372.  
  373. // -----------------------------------------------------------------------------------
  374. // Twitterを取り戻す(アイコンを戻す)
  375. // -----------------------------------------------------------------------------------
  376.  
  377. function replaceTwitterIcons() {
  378. if (!config.replaceIcons) return;
  379.  
  380. const paths = {
  381. bird: "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",
  382. premium:
  383. "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",
  384. defaultHomeActive:
  385. "M21.591 7.146L12.52 1.157c-.316-.21-.724-.21-1.04 0l-9.071 5.99c-.26.173-.409.456-.409.757v13.183c0 .502.418.913.929.913H9.14c.51 0 .929-.41.929-.913v-7.075h3.909v7.075c0 .502.417.913.928.913h6.165c.511 0 .929-.41.929-.913V7.904c0-.301-.158-.584-.408-.758z",
  386. twitterHome:
  387. "M12 9c-2.209 0-4 1.791-4 4s1.791 4 4 4 4-1.791 4-4-1.791-4-4-4zm0 6c-1.105 0-2-.895-2-2s.895-2 2-2 2 .895 2 2-.895 2-2 2zm0-13.304L.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.824 1.06-1.696L12 1.696zM19 19.5c0 .276-.224.5-.5.5h-13c-.276 0-.5-.224-.5-.5V8.429l7-4.375 7 4.375V19.5z",
  388. twitterHomeActive:
  389. "M12 1.696L.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.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",
  390. };
  391.  
  392. GM_addStyle(`
  393. /* bird */
  394. .r-64el8z[href="/home"] > div > svg > g > path, /* main */
  395. .r-1blnp2b > g > path /* splash */ {
  396. d: path("${paths.bird}") !important;
  397. }
  398. /* premium */
  399. .r-eqz5dr[href="/i/premium_sign_up"] > div > div > svg > g > path {
  400. d: path("${paths.premium}") !important;
  401. }
  402.  
  403. /*
  404. 以下2つは 'X to Twitter' ( https://greasyfork.org/ja/scripts/471572-x-to-twitter ) を参考
  405. これのお陰で非アクティブ時のスタイルを適応できるようになった
  406. こんな書き方思いつかない
  407. */
  408.  
  409. /* home active */
  410. .r-eqz5dr[href="/home"] > div > div > svg > g > path:not(path[d="${paths.twitterHome}"]) {
  411. d:path("${paths.twitterHomeActive}");
  412. }
  413.  
  414. /* home not active */
  415. .r-eqz5dr[href="/home"] > div > div > svg > g > path:not(path[d="${paths.defaultHomeActive}"]) {
  416. d:path("${paths.twitterHome}");
  417. }
  418. `);
  419. }
  420.  
  421. // -----------------------------------------------------------------------------------
  422. // TLの時間を相対時間から絶対時間に変更(HH:MM:SS・mm/dd/yy, week)
  423. // -----------------------------------------------------------------------------------
  424. // タイムスタンプモジュール
  425. const TimestampModule = {
  426. toFormattedDateString: function (date) {
  427. const YEAR = date.getFullYear().toString().slice(-2);
  428. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  429. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  430. return `${TIME}・${DATE}`;
  431. },
  432. // タイムスタンプの更新
  433. updateTimestamps: function () {
  434. if (!config.useAbsoluteTime) return;
  435.  
  436. /*
  437. 1. 様々な時間要素
  438. 2. 引用の時間要素
  439. */
  440. const timeSelectors =
  441. 'a[href*="/status/"] > time, div.css-146c3p1.r-bcqeeo.r-1ttztb7.r-qvutc0.r-1qd0xha.r-a023e6.r-rjixqe.r-16dba41.r-xoduu5.r-1q142lx.r-1w6e6rj.r-9aw3ui.r-3s2u2q > time';
  442.  
  443. document.querySelectorAll(timeSelectors).forEach((timeElement) => {
  444. const parent = timeElement.parentNode;
  445. const span = Utils.createElement("span", {
  446. textContent: this.toFormattedDateString(
  447. new Date(timeElement.getAttribute("datetime"))
  448. ),
  449. });
  450. span.style.pointerEvents = "none";
  451. parent.appendChild(span);
  452. parent.removeChild(timeElement);
  453. });
  454. },
  455. };
  456.  
  457. // -----------------------------------------------------------------------------------
  458. // サイドバーに時間、日付を表示(HH:MM:SS, mm/dd/yy, week)
  459. // -----------------------------------------------------------------------------------
  460. const SidebarModule = {
  461. createInfoElement: function (type) {
  462. if (!config.showTimeAndDateSidebar) return;
  463.  
  464. const nav = document.querySelector(
  465. 'div[class="css-175oi2r r-vacyoi r-ttdzmv"]'
  466. );
  467. if (!nav || document.getElementById(type)) return;
  468.  
  469. const iconHTML =
  470. type === "time"
  471. ? '<i class="fa-regular fa-clock" style="width: 26.25px; height: 26.25px;"></i>'
  472. : '<i class="fa-solid fa-calendar-days" style="width: 26.25px; height: 26.25px;"></i>';
  473.  
  474. const textContentFunc = () => {
  475. const date = new Date();
  476. const YEAR = date.getFullYear().toString().slice(-2);
  477. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  478. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  479.  
  480. return type === "time" ? `${TIME}` : `${DATE}`;
  481. };
  482.  
  483. const infoElement = Utils.createElement("div", {
  484. id: type,
  485. classList: [
  486. "css-175oi2r",
  487. "r-6koalj",
  488. "r-eqz5dr",
  489. "r-16y2uox",
  490. "r-1habvwh",
  491. "r-cnw61z",
  492. "r-13qz1uu",
  493. "r-1loqt21",
  494. "r-1ny4l3l",
  495. ],
  496. });
  497.  
  498. const container = Utils.createElement("div", {
  499. id: `${type}__container`,
  500. classList: [
  501. "css-175oi2r",
  502. "r-sdzlij",
  503. "r-dnmrzs",
  504. "r-1awozwy",
  505. "r-18u37iz",
  506. "r-1777fci",
  507. "r-xyw6el",
  508. "r-o7ynqc",
  509. "r-6416eg",
  510. ],
  511. });
  512.  
  513. const icon = Utils.createElement("div", {
  514. id: `${type}__container__icon`,
  515. classList: ["css-175oi2r"],
  516. innerHTML: iconHTML,
  517. });
  518.  
  519. const text = Utils.createElement("div", {
  520. id: `${type}__container__text`,
  521. classList: [
  522. "css-146c3p1",
  523. "r-dnmrzs",
  524. "r-1udh08x",
  525. "r-3s2u2q",
  526. "r-bcqeeo",
  527. "r-1ttztb7",
  528. "r-qvutc0",
  529. "r-1qd0xha",
  530. "r-adyw6z",
  531. "r-135wba7",
  532. "r-16dba41",
  533. "r-dlybji",
  534. "r-nazi8o",
  535. ],
  536. });
  537.  
  538. const textContent = Utils.createElement("span", {
  539. id: `${type}__text__content`,
  540. classList: ["1jxf684", "r-bcqeeo", "r-1ttztb7", "r-qvutc0", "r-poiln3"],
  541. textContent: textContentFunc(),
  542. });
  543.  
  544. text.appendChild(textContent);
  545. container.appendChild(icon);
  546. container.appendChild(text);
  547. infoElement.appendChild(container);
  548. nav.appendChild(infoElement);
  549.  
  550. if (type === "time") {
  551. setInterval(() => {
  552. textContent.textContent = textContentFunc();
  553. }, 1000);
  554. }
  555. },
  556.  
  557. init: function () {
  558. this.createInfoElement("time");
  559. this.createInfoElement("date");
  560.  
  561. const observer = new MutationObserver(() => {
  562. this.createInfoElement("time");
  563. this.createInfoElement("date");
  564. });
  565.  
  566. observer.observe(document.body, { childList: true, subtree: true });
  567. },
  568. };
  569.  
  570. // -----------------------------------------------------------------------------------
  571. // 動画プレイヤーをデフォルトに戻す
  572. // -----------------------------------------------------------------------------------
  573. const VideoModule = {
  574. setupDefaultVideoPlayer: function (container) {
  575. if (!config.useDefaultVideoPlayer) return;
  576.  
  577. const video = container.querySelector("div:first-child > div > video");
  578. if (!video) return;
  579.  
  580. video.controls = true;
  581. video.removeAttribute("disablepictureinpicture");
  582. video.muted = false;
  583.  
  584. const onClick = (e) => {
  585. e.preventDefault();
  586. video
  587. .play()
  588. .then(() => {
  589. video.muted = false;
  590. })
  591. .catch((error) => console.error("Video playback error:", error));
  592.  
  593. const onVolumeChange = (e) => {
  594. if (e.target.muted) {
  595. e.target.muted = false;
  596. }
  597. e.target.removeEventListener("volumechange", onVolumeChange);
  598. };
  599.  
  600. e.target.addEventListener("volumechange", onVolumeChange);
  601. video.removeEventListener("click", onClick);
  602. };
  603.  
  604. video.addEventListener("click", onClick);
  605.  
  606. container.parentElement.appendChild(video);
  607. container.remove();
  608. },
  609.  
  610. observeVideos: function () {
  611. const observer = new MutationObserver(() => {
  612. const videoContainer = document.body.querySelector(
  613. 'div[data-testid="videoComponent"]:not(.enhanced-video)'
  614. );
  615. if (videoContainer) {
  616. videoContainer.classList.add("enhanced-video");
  617. setTimeout(() => this.setupDefaultVideoPlayer(videoContainer), 100);
  618. }
  619. });
  620.  
  621. observer.observe(document.body, { subtree: true, childList: true });
  622. },
  623. };
  624.  
  625. // -----------------------------------------------------------------------------------
  626. // Tweet Engagements をアクセスしやすく
  627. // -----------------------------------------------------------------------------------
  628. const TweetEngagementModule = {
  629. createQuoteButton: function () {
  630. const buttonWrapper = Utils.createElement("div", {
  631. classList: ["css-175oi2r", "r-18u37iz", "r-1h0z5md", "r-13awgt0"],
  632. });
  633.  
  634. const link = Utils.createElement("a", {
  635. attributes: {
  636. href: `${window.location.pathname}/quotes`,
  637. "data-testid": "tweetEngagements",
  638. target: "_blank",
  639. rel: "noopener",
  640. },
  641. classList: [
  642. "css-175oi2r",
  643. "r-1777fci",
  644. "r-bt1l66",
  645. "r-bztko3",
  646. "r-lrvibr",
  647. "r-1loqt21",
  648. "r-1ny4l3l",
  649. ],
  650. });
  651.  
  652. link.addEventListener("click", (event) => {
  653. const href = event.currentTarget.getAttribute("href");
  654. window.open(href, "_blank");
  655. });
  656.  
  657. const contentDiv = Utils.createElement("div", {
  658. attributes: { dir: "ltr" },
  659. classList: [
  660. "css-146c3p1",
  661. "r-bcqeeo",
  662. "r-1ttztb7",
  663. "r-qvutc0",
  664. "r-1qd0xha",
  665. "r-a023e6",
  666. "r-rjixqe",
  667. "r-16dba41",
  668. "r-1awozwy",
  669. "r-6koalj",
  670. "r-1h0z5md",
  671. "r-o7ynqc",
  672. "r-clp7b1",
  673. "r-3s2u2q",
  674. ],
  675. });
  676. contentDiv.style.textOverflow = "unset";
  677. contentDiv.style.color = "rgb(113, 118, 123)";
  678.  
  679. const iconDiv = Utils.createElement("div", {
  680. classList: ["css-175oi2r", "r-xoduu5"],
  681. innerHTML: `
  682. <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>
  683. <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">
  684. <g><path d="M8.75 21V3h2v18h-2zM18 21V8.5h2V21h-2zM4 21l.004-10h2L6 21H4zm9.248 0v-7h2v7h-2z"></path></g>
  685. </svg>
  686. `,
  687. });
  688.  
  689. // contentDiv にホバーイベントを追加
  690. contentDiv.addEventListener("mouseenter", () => {
  691. contentDiv.style.color = "rgb(238 201 104)";
  692. const iconBgDiv = iconDiv.querySelector("div");
  693. if (iconBgDiv) {
  694. iconBgDiv.style.backgroundColor = "rgba(238, 201, 104, 0.1)";
  695. }
  696. });
  697.  
  698. contentDiv.addEventListener("mouseleave", () => {
  699. contentDiv.style.color = "rgb(113, 118, 123)";
  700.  
  701. const iconBgDiv = iconDiv.querySelector("div");
  702. if (iconBgDiv) {
  703. iconBgDiv.style.backgroundColor = "";
  704. }
  705. });
  706.  
  707. const countDiv = Utils.createElement("div", {
  708. classList: ["css-175oi2r", "r-xoduu5", "r-1udh08x"],
  709. innerHTML: `
  710. <span data-testid="app-text-transition-container" style="transition-property: transform; transition-duration: 0.3s; transform: translate3d(0px, 0px, 0px);">
  711. <span class="css-1jxf684 r-1ttztb7 r-qvutc0 r-poiln3 r-n6v787 r-1cwl3u0 r-1k6nrdp r-n7gxbd" style="text-overflow: unset">
  712. <span class="css-1jxf684 r-bcqeeo r-1ttztb7 r-qvutc0 r-poiln3" style="text-overflow: unset">Quotes</span>
  713. </span>
  714. </span>
  715. `,
  716. });
  717.  
  718. contentDiv.appendChild(iconDiv);
  719. contentDiv.appendChild(countDiv);
  720. link.appendChild(contentDiv);
  721. buttonWrapper.appendChild(link);
  722.  
  723. return buttonWrapper;
  724. },
  725.  
  726. // 投稿ページかどうかを判定
  727. isTweetPage: function () {
  728. return (
  729. document.querySelector(
  730. "div.css-175oi2r.r-1kbdv8c.r-18u37iz.r-1oszu61.r-3qxfft.r-n7gxbd.r-2sztyj.r-1efd50x.r-5kkj8d.r-h3s6tt.r-1wtj0ep"
  731. ) !== null
  732. );
  733. },
  734.  
  735. addQuoteElement: function () {
  736. // 投稿ページの場合ボタンを表示
  737. if (!this.isTweetPage()) {
  738. return;
  739. }
  740.  
  741. const targetDiv = document.querySelector('div[role="group"][id^="id__"]');
  742. if (
  743. !targetDiv ||
  744. targetDiv.querySelector('[data-testid="tweetEngagements"]')
  745. ) {
  746. return;
  747. }
  748.  
  749. const quoteButton = this.createQuoteButton();
  750. targetDiv.insertBefore(quoteButton, targetDiv.children[4]);
  751. },
  752.  
  753. init: function () {
  754. const debouncedAddQuoteElement = Utils.debounce(
  755. () => this.addQuoteElement(),
  756. 250
  757. );
  758.  
  759. // URL変更の監視
  760. let lastUrl = location.href;
  761. const urlObserver = new MutationObserver(() => {
  762. const url = location.href;
  763. if (url !== lastUrl) {
  764. lastUrl = url;
  765. debouncedAddQuoteElement();
  766. }
  767. });
  768.  
  769. urlObserver.observe(document, { subtree: true, childList: true });
  770.  
  771. const retryInterval = setInterval(debouncedAddQuoteElement, 1000);
  772.  
  773. window.addEventListener("popstate", debouncedAddQuoteElement);
  774. history.pushState = ((origPushState) => {
  775. return function (state, title, url) {
  776. origPushState.apply(this, arguments);
  777. debouncedAddQuoteElement();
  778. };
  779. })(history.pushState);
  780.  
  781. history.replaceState = ((origReplaceState) => {
  782. return function (state, title, url) {
  783. origReplaceState.apply(this, arguments);
  784. debouncedAddQuoteElement();
  785. };
  786. })(history.replaceState);
  787.  
  788. // DOMContentLoaded で初期化
  789. document.addEventListener("DOMContentLoaded", () => {
  790. debouncedAddQuoteElement();
  791. clearInterval(retryInterval);
  792. });
  793.  
  794. debouncedAddQuoteElement();
  795. },
  796. };
  797.  
  798. // -----------------------------------------------------------------------------------
  799. // メイン処理
  800. // -----------------------------------------------------------------------------------
  801. function main() {
  802. loadConfig();
  803. SettingsModule.createSettingsUI();
  804. setupKeyboardShortcut();
  805. setupMenuCommand();
  806.  
  807. // タイムスタンプの更新を定期的に実行
  808. setInterval(() => TimestampModule.updateTimestamps(), 1000);
  809.  
  810. // アイコン情報表示を初期化
  811. replaceTwitterIcons();
  812.  
  813. // サイドバーの情報表示を初期化
  814. SidebarModule.init();
  815.  
  816. // 動画プレイヤーの設定を監視
  817. VideoModule.observeVideos();
  818.  
  819. // 引用ツイートボタンの追加を初期化
  820. TweetEngagementModule.init();
  821. }
  822.  
  823. // ページ読み込み時にメイン処理を実行
  824. window.addEventListener("load", main);
  825. })();