🏠 Home 

Greasy Fork is available in English.

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.6
  17. // @author Yos_sy
  18. // @match https://x.com/*
  19. // @namespace http://tampermonkey.net/
  20. // @icon ###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###XGgo3KOy7Jws/tf04C9Nzj8cDpdOoKP91n6ugjC35HRwc6Ojrg9/uzCv9ov7Z6yLkwIHA0PVgvu89oNGo6/VAzDTIJeNENPZL3mBqL8AQgv9+v8RjztuO8XFlWubnNL+cd8B50pMbSQVoKDwHyvAGTySQmPk9Rlm1QOatRDkuOVpBJRBoRbwtPdf108NAfkQGvpiThT6fTGQVmvEcEj/8Hg8ExFec/FnKSAIDsTRTkJB65nRYJglwjwMtyZbuSJlcwGBT14NQGmoSKiIA399CDTDC###hGEqQBUGKRnBTEswV5MxQ6P49lR6NRjdqbLckqlyBHVgiktXHhJ9Wf7+PHm3nweD/tzUcEzWP91NAzWycfItt8Qs4SQDbwbD5KleWJHjyKwG1rKvThFYTUt7+wsFD832KxiLRQ3qKLBI6rgXqCJqucRFByIgqgNQMoyYcIgq+CvGaAmx6cAHhH22y+i1whA7lqkx/c8cdXfTm3nxJ9iPiAo7vx8BRqEnY6aKXnNf3kcM2V6zOcGBMEIBfKcPufZ+SRfS8TAAl/Ov15yy65BJkLGzniZOGX7XJebMTHx4Vfr+05TWaZAHiHYFn4qZ8iFR5xM4UTRi6AfjcRAE+6ohAvX/lp9S8uLobX6xXCT1t506oPQNPRR87wkw8K93ECyJVrNJwY9QSgJ1wkVCRYlBhEbcc4AdDqSqsufU6vHl9PxebjksdHvQ55TQMvCqIx8MpAAtcMuBljMBgyhJ+nu/LeCNx5mUuZgrTa6+2sJG8Hxzv5UqYf3zCWawC80EsmAIrz0+pPGgBPGsuV6zOcGPUEwIWfp8qSYIVCIbhcLkEAZIeTc1AWfnLE6dXjcwLItrLyz/CCJipASaVSmtoEUnd5rQA3AeS6AYPBIJKWuA+EfAw0diIvGlMupQvzlZ8iOPTINTbZBKDdo+XNO7kJwDVDOce/ra1N09GH5/ePxRBfbzCqCUAWfirkoUlEq4jccgs4ShLpdFrTupuiACTIen4G2Y7nY+M15nI1YyQSQSqV0qS4UrEQERRPgyaB5xWERqNROEBdLpdwNPLNKnkyExf+XAD30fCcf9JeZM+/HPpzu91Z6zFk05AIoLu7W4T5urq6NE1glA9glIETAJC5ww6F4WgCEcPTCkt2PcWbufDH43FBCrIqSSsNRQLoUa/smKoWaXw0KamfoKzm8mpBq9UqSAk4Gr3gOQDULJW3F+MmAJkJuegDAKAhAJ4jIYf+5M08iACyfaceAQQCAaH6t7e3o7u7O6PQShHAKAIRAIAM+5xy6bn9SOEdutmUbUfqMsXiyVYms4DX43NnExdsPbWaOwFpfHy1SiaTGTau3FuOvlfeBo3GuWXLFqxevRrNzc2iByHZw5T/IF8nAs+rkCsjexMtoM9T+LI/1XHcBJCToeQtvbgJ4PF44HK5MlKf+XNOAHwjz87OTkEA5CvQawmWbxiVBCB7xClGT2oz3Xyy9eR9BWQ1VC5AIu8zeZsp1kxagl43GBqPPD4+NrJ59cbW074H9NlwOIzXX38dq1evxoIFCzB9+nR0d3fj9ddfx0cffSR+TzKZFCaD1+tFdXU1KisrUVpaiuLiYrEhJuVFpNNpEbHo7u5Ge3s7WltbcfDgQRw6dEjEzFOpFOrq6vCNb3wDs2fPRnd3N15++WWsXbsWwWBQ937pVVKSwFN+BD94jF+O85OmQ74UEnhez8F37OFbd/EKP/qsfCgNYJRBT8XlLcX09hUgRyBXO3kSkWyf0ypEk0fO1acDQMYk6s/45H0PSC0Oh8NYuXIl1q1bhxtuuAFnnnmmMCnOOuss/OlPf8ITTzyB8vJynHbaaZg9ezZmzpyJmpoaTXqsTHoEOeWa1OmOjg5s3LgRDzzwACwWC374wx/i5JNPFtGJL3/5y/jjH/+Iv/3tb2hvbxdm###5RwNpNRR9oetL5Op0OsVKT+E+yqEgpyb5Ushs4s+p0q+trU3k9euF+uQQaT4jJ2sBjgW9LDJ6JO8+CZBcJ8CLZujm86ShdDotJpbcjINWerkjENdEuAD0Z3y8TwF9V2dnJ55++ml88sknuPXWW3HWWWeJ0maj0Yji4mLMmzcPANDR0YFLL70UixcvxsSJE0XYjKIEvHaCV8xxQiSfCJknb731FqLRKL7//e/jrLPOEin###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###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###uhXf+973RIhtMMYkV1gmk0n4fD6sXr0aL7/8MgKBAKZMmYJ58+ahrKwMNpsN4XAYLS0t2L17N/73f/8XFRUVuPDCC3HmmWcKIlAaQf+hCGCAiEaj2LhxIx5++GFEo1FccMEFOP3001FUVASr1So2GaHuPtFoFM3Nz###pZfw4IMP4t1338XSpUsxefJkzXsHc1Xr6OhAMpnEmWeeiWuuuQYnn3yyIB0AqKysxCWXXIKTTjoJzzzzDJYvX46WlpZBLbCRhTQWi+HDDz/EI488go6ODpx//vk488wz4fF4YLVaYTabNdctFouhtbUVL730Eh555BG8/fbb+Pa3v41x48YN2hjzEQYAij77gZKSEtx///0oKirCQw89hPnz5+PGG28UcXPeUISDbzqybds23HfffUin07jlllvwhS98QRNVGCgJkNDFYjHs2bMH5eXl8Hq9ug41svvj8TgaGxths9k0G5wOxjjoeTwex+uvv44HHngAM2fOxC233ILKysqM68Zj+3Rdkskkdu/ejfvvvx8+nw9XXXUVli1bhtWrVytNoB9QBNBPFBUVYdGiRejs7MRFF12ExYsXw+12i8QaIHvKLf9bS0sLfv/732P79u244447cMopp2jeOxDhkwWvN9qFnm09mGNIpVJYuXIlHnnkEZx//vm44oorspoZetePXh85cgTLli3DO++8g/3792Pnzp2KAPoBtTNQP5FIJBAOh3HDDTfg4osvhtPp1N0RmFYueWWn5y6XC3PmzEFzczP+8pe/YNKkSaLH4GAIoPwdNMZshzzmgZ6fQL/nhRdewEMPPYQlS5bg8ssvz7hu/Pzyc/7a4XBgzpw5aGlpwauvvqp2CeonFAH0AwaDAcXFxbjhhhtw9dVXi0lM/+tJYPSEizoL+Xw+PP3005gwYQKqqqo0obiBNPSQz3mszwyG8Oup/StWrMDDDz+MpUuX4sILL4Tdbs+4btnOpzcuq9WKKVOmoK2tDVu3blUaQD+gCKAfMBgM+OpXv4rvf//7Il12ICnBBsPnvf8aGhrQ3NyMf/zjH5gwYQIqKysHfSUeDsiCGI1G8corr+Dhhx/GFVdcgUWLFsHhcGgSePr729xuNyZOnIgNGzaoPQL6AUUA/UBZWRl+/vOfY9asWaI2Hej7JOaCTbUFxx13HLZv344VK1agvr4e1dXVGe/NZcgrfywWw9q1a/Hggw/im9/8Ji6++OIBCz+/FlSWbbVa8cYbb6jWYH2EIoB+4LLLLsMVV1wxKDsBy/atzWbDCSecgMbGRixfvhw1NTWYMGFC1s/kEvQcfi+++CKWLVuGxYsX49JLL9Wo/cDg/Baz2YyCggJ8/PHH2Ldv34C/L5+gCKCPKCkpwV133YU5c+ZoHGoDgUwCdrsds2fPRktLCx577DHU1dVhwoQJg+YYHGoQEfzjH//AsmXLsGTJElxyySWDKvzydXC73WhtbcV7772nSR1W6BkqEaiPmDx5MiZNmgSj0Tio+fI85GU0GuH1enHNNdcAAO677z6YTCYsWLBAU4CUKyQg2/zxeByrV6/Gww8/jCVLlmDRokWw2WyDvvLza2axWDB37lxUVlZiz549A/7ufIEqBuojZs6cqekEPJhCyL/LaDSirKxMbM5x33334YMPPhCblQL937l3MJHN4ffYY4/pevuBwb9m9H0TJ07EuHHjcoYYRwMUAfQBFosF9fX1KC4uHrJqOa7amkwmlJaW4qabbsKUKVPwq1/9Cps2bcqoJMwFkMPvtddew4MPPojFixfjG9/4BpxO56B4+491boPBgHHjxqG6ulqVB/cBigD6AJfLhYqKCpGnPlQrjawJeDwefPe738XcuXNx//3346OPPhqS8/YVnIRSqRTWrFmDP/3pT7jkkktwySWXwGq1DtnKz0H3wu12o7S0VBFAH6AIoA+w2WxwuVzDomLKJOD1enHjjTdi5syZuO+++3Do0CGRNz+SW###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###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###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###l5kRGE4OgKBANxuNwKBgNAYeiJgHtuXcz1ojLxPA/1dDlvSWOg7eVNUmjM0JvIxkbmjpyHmqqaYcwTA1Xy+aScdtOJTJx/aBJNi0mQu0E2VQX32SNh5llg2gR+NCR5DCbomiUQCJpNJ4wG32WzCfCIh4f4Y3iyFRwooRCjnd1AEhtR00hxkIaPnPAsx2wLCG4bycJ5MBDQXeeJQLBbTpBnzFvPU3UgmFf6Ya/Mo5whA9vTzDr12u11U9VE9P7+hZEPKF58flOxDBMC9/DzUx1cKRQBHQQJH15M3y6QYutxUlK+UPOZPBwmt0WhEMpkU4UG9eLzD4dB1wMpJR1wL5K/5fOG5/TIBANqmoqS58E1VSfjpt1IkQG41Ts9zcR7lNAGQnc/ZllJ9Sd3PRgCpVEqzovNdcynBhzz9nADIHpQ7++SqCjcS4BoAF/5UKgWLxaIxr0j4SXB4NECuxSAhowpDLvxEHKQByKFYXpfPa0TkorFjaQC8qxH/Lh4+1ttWjggAOJrqTNoRXbP+7OMw1MhJAiAvLG/dTQfP7tMjAO6YIs8vHeFwWNT1kwYQDAaFyko3TlbhcpG5RwpcA5CFP5lMwmKxaFZ/p9OJQCAgNDgiCV58Qyo6hdicTqdG7eehuFAolJH/LydmyULPBZmIqCcNIFsaOvD5vJL7DPD5A3yejix3KuJkmUvISQKQwzSU7KOX5dcTAZD3Vy/TjxOAnPs9ltN8BwO00vZnXwEAokgHgKYm32aziXp8uUknAOFgzGbiySm+eoJMZmVPPoBsGgT5A8bSvgI5RwDZQn1U1kuCT4dMAASuAZDg0+rP03xlHwCFj7LZmPkOrgHQdTEa+7avAKnCeh15LBZLVuEncpH9OtxcIwIA9JvGkmbZkwZA5+MRKXovJ4CxsK9AzhGAng+A1/UXFxcLfwAdnADoBhLr8s4+PMdfNgH4TSTkQ7y/PyANgB6B/u0rQPkA/J4DyCr8pFrzMJ7cel0O08qFTRTf78kHQGPkmYt0AEcjFWNhX4ERIQC9VF86SEWTQyy8mSffs48XZZDDSI7zU4yf8v31vP981VE4NnjojaMv+woQAfCVm2uA/H8ARK1/XwhAhtlsFk5JXtJMY5N9BnIoMZVKabohO51OTeiYOyZHw74Cw04A2RJ96KD0Xl7UQ0k/vKiH559Tiid5/cnLz8t55VVfDv0pFX9wwNOC+7OvANnZcoYgCTZpeXrqf2/i7DyrjwhAJgPupORNQAFoxkVVqbLJokcAubqvwIgRAFer+MFXer6LD1f36Tt48wlSs2jnWPngXX15fj/3NCsMHHoE0Jd9Beg7+DzhGXnUX1DPEdgbEqcIE+8YRK95jwPe9EN2KHJi4puK0Nhl4c/lfQVGlAD02kpRSq/eFl6kAfALSBeUjnA4LGx9PRKgrj56XWmUBjBwyATQ130FSEXm8wQ4ShpyPwY5HHgsEInwGhH+yIWfJxFxvwCfv9xspPeMpn0FRpwAeP92h8MhWntxEuD799FF5734+OtQKKQr+HRQVx++3ZfsdFLoP2QCkD3iQM/7###nXc7CI9KQw7N6vRl6Ajc9SCD18hLkMKCeCcBNE/peg8EwqvYVyAkC4B59TgB6m3iSJ5hWbPIB0EG5/rLqT89DoZBuqqbSAAYHnABk4e/NvgJyYw4SrJ5Cs30puKGIQ09+KO4E7MkEkIWfNz0dLfsK5AwBcE+/nhOQNABK9aWYq5zsQ+G+bFoAZZLpFWsoAhg4OAEAfd9XgGfiyYLHhUTPgdab+6eXhpxN0OXX9Hk904TMGYPBMKr2FcgpAqBCn54IgFQsshU5AfDVnpMANwHC4bCuF1Zl+w0OiACA/u8roFfMI5d38/P1FXLugizg/FHvfXqmCU8jHk37CuRcIhDPK6fGCzycBEBk8PH+ffKKT7F+CvnxiaYwdOCrGb+XtMJSrQDlZ3Byp3p/nnkn1wqMdNstufhI9kXIJeW57l8adgI4VphIVhkpm6ywsBA+nw8AhPDT5px6B2X4ZevkozD00FNxeS9G2hWI1OJUKoXu7m7dOn6uAYz0b5KdjpwIAoEAjhw5giNHjqC9vV1UnVLnIbnKVPZjDDdGnACOJfy0Svh8PrjdbgAQWXz0KLf2pkMRwMiBm1lUL0D3lxMAJXVRTgf5efRq+clBN9K/S3ZA8uehUEizkxQnAL1GM3SMFEacAI618pMThXIAAIj/U7ppT4cigOGHno1Odi45cHlHYRJ+ipnLITi9cNxIIlv0IZ1Oi0Q0XnYuawB6laYjNTdHlAB6En69JCG+d1tfDkUAIwfu4CLvN2kAXPhpN2abzZa1lp+ejySyCT79T95shvxQcvNR2YzISxMAyBT+bCnCvBqLkn76cigCGF7oXWu699FoNGPlp6YhZBL0VNM/ksgWeqRH+j16B28rr6dFjAQMAIb1zMcqBjrWASCjHrwvhyKB4YFeGA04WgbMD8rG45ly9Llstf0jBT3zhv+NFjd+8ExVPgf1NIjhxrATANBzOfCxDiC7GtabQ2FkoafSy3+T36/3fKSRbS5liw5kU/VHek6OCAEoKCjkBkbfZmYKCgqDBkUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5DEUACgp5jP8PBb5EQdmudTwAAAAASUVORK5CYII=
  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. (function () {
  29. "use strict";
  30. GM_addStyle(`
  31. /* -----------------------------------------------------------------------------------
  32. 基本的なボーダーを消す
  33. ----------------------------------------------------------------------------------- */
  34. /* light */
  35. .r-jxzhtn /* basic */,
  36. .r-1igl3o0, /* tl */
  37. /* gray */
  38. .r-18bvks7 /* basic */,
  39. .r-1ila09b /* tl */,
  40. /* dark */
  41. .r-1kqtdi0 /* basic */,
  42. .r-j5o65s /* tl */ {
  43. border: none !important;
  44. }
  45. /* -----------------------------------------------------------------------------------
  46. カキコの下のボーダーを消す
  47. ----------------------------------------------------------------------------------- */
  48. .r-109y4c4 {
  49. height: 0 !important;
  50. }
  51. /* -----------------------------------------------------------------------------------
  52. TLの幅を600pxから700pxに、右サイドバーの幅を350pxから250pxに、画像拡大時のリプサイドバーを350pxから550pxに変更
  53. ----------------------------------------------------------------------------------- */
  54. /* TL */
  55. .r-1ye8kvj {
  56. max-width: 700px !important;
  57. }
  58. /* Sidebar */
  59. .r-1hycxz {
  60. width: 250px !important;
  61. }
  62. /* 画像拡大時のリプサイドバー */
  63. .css-175oi2r.r-kemksi.r-1kqtdi0.r-th6na.r-1phboty.r-1dqxon3.r-1hycxz {
  64. width: 550px !important;
  65. }
  66. /* -----------------------------------------------------------------------------------
  67. ヘッダーのスクロールバーを消す
  68. ----------------------------------------------------------------------------------- */
  69. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd {
  70. overflow-y: scroll !important;
  71. -ms-overflow-style: none !important;
  72. scrollbar-width: none !important;
  73. }
  74. .css-175oi2r.r-1pi2tsx.r-1wtj0ep.r-1rnoaur.r-o96wvk.r-is05cd::-webkit-scrollbar {
  75. display: none !important;
  76. }
  77. /* -----------------------------------------------------------------------------------
  78. サイドバーの”Subscribe to Premium”を消す
  79. ----------------------------------------------------------------------------------- */
  80. .css-175oi2r.r-1habvwh.r-eqz5dr.r-uaa2di.r-1mmae3n.r-3pj75a.r-bnwqim {
  81. display: none !important;
  82. }
  83. /* -----------------------------------------------------------------------------------
  84. サイドバーの”Who to follow”を消す
  85. ----------------------------------------------------------------------------------- */
  86. .css-175oi2r.r-1bro5k0 {
  87. display: none !important;
  88. }
  89. /* -----------------------------------------------------------------------------------
  90. TL上のUserNameを消す
  91. ----------------------------------------------------------------------------------- */
  92. div[data-testid="User-Name"] > div:nth-child(2) > div > div:nth-child(1),
  93. div[data-testid="User-Name"] > div:nth-child(2) > div > div:nth-child(2) {
  94. display: none !important;
  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. function loadConfig() {
  131. const savedConfig = localStorage.getItem("twitterKaizenConfig");
  132. if (savedConfig) {
  133. Object.assign(config, JSON.parse(savedConfig));
  134. }
  135. }
  136. // ローカルストレージに設定を保存
  137. function saveConfig() {
  138. localStorage.setItem("twitterKaizenConfig", JSON.stringify(config));
  139. }
  140. // -----------------------------------------------------------------------------------
  141. // ユーティリティ関数と定数
  142. // -----------------------------------------------------------------------------------
  143. const Utils = {
  144. debounce: (func, wait) => {
  145. let timeout;
  146. return (...args) => {
  147. clearTimeout(timeout);
  148. timeout = setTimeout(() => func(...args), wait);
  149. };
  150. },
  151. pad: (num) => num.toString().padStart(2, "0"),
  152. createElement: (tag, options = {}) => {
  153. const element = document.createElement(tag);
  154. if (options.id) element.id = options.id;
  155. options.classList?.forEach((cls) => element.classList.add(cls));
  156. Object.entries(options.attributes || {}).forEach(([attr, value]) =>
  157. element.setAttribute(attr, value)
  158. );
  159. Object.entries(options.styles || {}).forEach(([key, value]) => {
  160. element.style[key] = value;
  161. });
  162. if (options.innerHTML) element.innerHTML = options.innerHTML;
  163. if (options.textContent) element.textContent = options.textContent;
  164. return element;
  165. },
  166. observeDOM: (
  167. targetNode,
  168. callback,
  169. config = { childList: true, subtree: true }
  170. ) => {
  171. const observer = new MutationObserver(callback);
  172. observer.observe(targetNode, config);
  173. return observer;
  174. },
  175. };
  176. // 多言語定義
  177. const TRANSLATIONS = {
  178. en: {
  179. panel: {
  180. replaceIcons: "Reclaim Twitter (restore icon)",
  181. useAbsoluteTime: "Change TL time from relative to absolute time",
  182. showTimeAndDateSidebar: "Display time and date in sidebar",
  183. useDefaultVideoPlayer: "Revert video player to default",
  184. enhanceTweetEngagements: "Easy access to quoted tweets",
  185. },
  186. weeks: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
  187. },
  188. ja: {
  189. panel: {
  190. replaceIcons: "Twitterを取り戻す (アイコンを元に戻す)",
  191. useAbsoluteTime: "TLの時間を相対時間から絶対時間に変更",
  192. showTimeAndDateSidebar: "サイドバーに時間、日付を表示",
  193. useDefaultVideoPlayer: "動画プレイヤーをデフォルトに戻す",
  194. enhanceTweetEngagements: "引用ツイートへのアクセスを簡単に",
  195. },
  196. weeks: ["日", "月", "火", "水", "木", "金", "土"],
  197. },
  198. zh: {
  199. panel: {
  200. replaceIcons: "替换 Twitter 图标",
  201. useAbsoluteTime: "使用绝对时间",
  202. showTimeAndDateSidebar: "显示时间和日期侧边栏",
  203. useDefaultVideoPlayer: "使用默认视频播放器",
  204. enhanceTweetEngagements: "增强推文互动",
  205. },
  206. weeks: ["周日", "周一", "周二", "周三", "周四", "周五", "周六"],
  207. },
  208. ko: {
  209. panel: {
  210. replaceIcons: "Twitter 아이콘 교체",
  211. useAbsoluteTime: "절대 시간 사용",
  212. showTimeAndDateSidebar: "시간 및 날짜 사이드바 표시",
  213. useDefaultVideoPlayer: "기본 비디오 플레이어 사용",
  214. enhanceTweetEngagements: "트윗 참여 향상",
  215. },
  216. weeks: ["일", "월", "화", "수", "목", "금", "토"],
  217. },
  218. ru: {
  219. panel: {
  220. replaceIcons: "Заменить иконки Twitter",
  221. useAbsoluteTime: "Использовать абсолютное время",
  222. showTimeAndDateSidebar: "Показать боковую панель времени и даты",
  223. useDefaultVideoPlayer: "Использовать стандартный видеоплеер",
  224. enhanceTweetEngagements: "Улучшить взаимодействие с твитами",
  225. },
  226. weeks: ["Вс", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб"],
  227. },
  228. de: {
  229. panel: {
  230. replaceIcons: "Twitter-Icons ersetzen",
  231. useAbsoluteTime: "Absolute Zeit verwenden",
  232. showTimeAndDateSidebar: "Zeit- und Datums-Sidebar anzeigen",
  233. useDefaultVideoPlayer: "Standard-Video-Player verwenden",
  234. enhanceTweetEngagements: "Tweet-Interaktionen verbessern",
  235. },
  236. weeks: ["So", "Mo", "Di", "Mi", "Do", "Fr", "Sa"],
  237. },
  238. };
  239. const LANG = navigator.language.split("-")[0];
  240. const CURRENT_LANG = TRANSLATIONS[LANG] || TRANSLATIONS.en;
  241. const PANEL_LANG = CURRENT_LANG.panel;
  242. const WEEKS_LANG = CURRENT_LANG.weeks;
  243. // -----------------------------------------------------------------------------------
  244. // 設定パネル
  245. // -----------------------------------------------------------------------------------
  246. const config = {
  247. replaceIcons: true,
  248. useAbsoluteTime: true,
  249. showTimeAndDateSidebar: true,
  250. useDefaultVideoPlayer: true,
  251. enhanceTweetEngagements: true,
  252. };
  253. const SettingsModule = {
  254. createSettingsUI: function () {
  255. const settingsDiv = Utils.createElement("div", {
  256. id: "twitter-kaizen-panel",
  257. classList: ["twitter-kaizen-panel"],
  258. });
  259. // パネルのインラインスタイルを追加
  260. Object.assign(settingsDiv.style, {
  261. position: "fixed",
  262. top: "10px",
  263. right: "10px",
  264. zIndex: "9999",
  265. background: "#f9f9f9",
  266. padding: "15px",
  267. border: "1px solid #ccc",
  268. borderRadius: "10px",
  269. boxShadow: "0 4px 12px rgba(0, 0, 0, 0.1)",
  270. color: "#333",
  271. fontFamily: "Arial, sans-serif",
  272. width: "300px",
  273. maxWidth: "100%",
  274. display: "none",
  275. transition: "transform 0.3s ease, opacity 0.3s ease",
  276. });
  277. const title = Utils.createElement("h3", {
  278. textContent: "Twitter Kaizen Settings",
  279. });
  280. title.style.fontSize = "18px";
  281. title.style.margin = "10px";
  282. title.style.color = "#333";
  283. settingsDiv.appendChild(title);
  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. features.forEach(({ key, label }) => {
  301. const checkbox = Utils.createElement("input", {
  302. attributes: { type: "checkbox", id: key },
  303. });
  304. checkbox.checked = config[key];
  305. checkbox.addEventListener("change", () => {
  306. config[key] = checkbox.checked;
  307. saveConfig();
  308. location.reload();
  309. });
  310. const labelElement = Utils.createElement("label", {
  311. attributes: { for: key },
  312. textContent: label,
  313. });
  314. labelElement.style.marginLeft = "8px";
  315. labelElement.style.fontSize = "14px";
  316. labelElement.style.color = "#555";
  317. settingsDiv.appendChild(checkbox);
  318. settingsDiv.appendChild(labelElement);
  319. settingsDiv.appendChild(Utils.createElement("br"));
  320. });
  321. document.body.appendChild(settingsDiv);
  322. },
  323. toggleSettingsPanel: function () {
  324. const panel = document.getElementById("twitter-kaizen-panel");
  325. if (panel) {
  326. if (panel.style.display === "none") {
  327. panel.style.display = "block";
  328. panel.style.transform = "scale(1)";
  329. panel.style.opacity = "1";
  330. } else {
  331. panel.style.transform = "scale(0.9)";
  332. panel.style.opacity = "0";
  333. setTimeout(() => {
  334. panel.style.display = "none";
  335. }, 300);
  336. }
  337. }
  338. },
  339. };
  340. // ショートカットキー
  341. function setupKeyboardShortcut() {
  342. document.addEventListener("keydown", function (e) {
  343. if (e.ctrlKey && e.altKey && e.key === "o") {
  344. SettingsModule.toggleSettingsPanel();
  345. }
  346. });
  347. }
  348. // メニューコマンドの登録
  349. function setupMenuCommand() {
  350. GM_registerMenuCommand("Toggle Twitter Kaizen Settings", () => {
  351. SettingsModule.toggleSettingsPanel();
  352. });
  353. }
  354. // -----------------------------------------------------------------------------------
  355. // Twitterを取り戻す(アイコンを戻す)
  356. // -----------------------------------------------------------------------------------
  357. function replaceTwitterIcons() {
  358. if (!config.replaceIcons) return;
  359. const paths = {
  360. 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",
  361. premium:
  362. "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",
  363. defaultHomeActive:
  364. "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",
  365. twitterHome:
  366. "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",
  367. twitterHomeActive:
  368. "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",
  369. };
  370. GM_addStyle(`
  371. /* bird */
  372. .r-64el8z[href="/home"] > div > svg > g > path, /* main */
  373. .r-1blnp2b > g > path /* splash */ {
  374. d: path("${paths.bird}") !important;
  375. }
  376. /* premium */
  377. .r-eqz5dr[href="/i/premium_sign_up"] > div > div > svg > g > path {
  378. d: path("${paths.premium}") !important;
  379. }
  380. /*
  381. 以下2つは 'X to Twitter' ( https://greasyfork.org/ja/scripts/471572-x-to-twitter ) を参考
  382. これのお陰で非アクティブ時のスタイルを適応できるようになった
  383. こんな書き方思いつかない
  384. */
  385. /* home active */
  386. .r-eqz5dr[href="/home"] > div > div > svg > g > path:not(path[d="${paths.twitterHome}"]) {
  387. d:path("${paths.twitterHomeActive}");
  388. }
  389. /* home not active */
  390. .r-eqz5dr[href="/home"] > div > div > svg > g > path:not(path[d="${paths.defaultHomeActive}"]) {
  391. d:path("${paths.twitterHome}");
  392. }
  393. `);
  394. }
  395. // -----------------------------------------------------------------------------------
  396. // TLの時間を相対時間から絶対時間に変更(HH:MM:SS・mm/dd/yy, week)
  397. // -----------------------------------------------------------------------------------
  398. // タイムスタンプモジュール
  399. const TimestampModule = {
  400. toFormattedDateString: function (date) {
  401. const YEAR = date.getFullYear().toString().slice(-2);
  402. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  403. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  404. return `${TIME}・${DATE}`;
  405. },
  406. // タイムスタンプの更新
  407. updateTimestamps: function () {
  408. if (!config.useAbsoluteTime) return;
  409. /*
  410. 1. 様々な時間要素
  411. 2. 引用の時間要素
  412. */
  413. const timeSelectors =
  414. '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';
  415. document.querySelectorAll(timeSelectors).forEach((timeElement) => {
  416. const parent = timeElement.parentNode;
  417. const span = Utils.createElement("span", {
  418. textContent: this.toFormattedDateString(
  419. new Date(timeElement.getAttribute("datetime"))
  420. ),
  421. });
  422. span.style.pointerEvents = "none";
  423. parent.appendChild(span);
  424. parent.removeChild(timeElement);
  425. });
  426. },
  427. };
  428. // -----------------------------------------------------------------------------------
  429. // サイドバーに時間、日付を表示(HH:MM:SS, mm/dd/yy, week)
  430. // -----------------------------------------------------------------------------------
  431. const SidebarModule = {
  432. createInfoElement: function (type) {
  433. if (!config.showTimeAndDateSidebar) return;
  434. const nav = document.querySelector(
  435. 'div[class="css-175oi2r r-vacyoi r-ttdzmv"]'
  436. );
  437. if (!nav || document.getElementById(type)) return;
  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. const textContentFunc = () => {
  443. const date = new Date();
  444. const YEAR = date.getFullYear().toString().slice(-2);
  445. const TIME = `${Utils.pad(date.getHours())}:${Utils.pad(date.getMinutes())}:${Utils.pad(date.getSeconds())}`;
  446. const DATE = `${Utils.pad(date.getMonth() + 1)}/${Utils.pad(date.getDate())}/${YEAR}, ${WEEKS_LANG[date.getDay()]}`;
  447. return type === "time" ? `${TIME}` : `${DATE}`;
  448. };
  449. const infoElement = Utils.createElement("div", {
  450. id: type,
  451. classList: [
  452. "css-175oi2r",
  453. "r-6koalj",
  454. "r-eqz5dr",
  455. "r-16y2uox",
  456. "r-1habvwh",
  457. "r-cnw61z",
  458. "r-13qz1uu",
  459. "r-1loqt21",
  460. "r-1ny4l3l",
  461. ],
  462. });
  463. const container = Utils.createElement("div", {
  464. id: `${type}__container`,
  465. classList: [
  466. "css-175oi2r",
  467. "r-sdzlij",
  468. "r-dnmrzs",
  469. "r-1awozwy",
  470. "r-18u37iz",
  471. "r-1777fci",
  472. "r-xyw6el",
  473. "r-o7ynqc",
  474. "r-6416eg",
  475. ],
  476. });
  477. const icon = Utils.createElement("div", {
  478. id: `${type}__container__icon`,
  479. classList: ["css-175oi2r"],
  480. innerHTML: iconHTML,
  481. });
  482. const text = Utils.createElement("div", {
  483. id: `${type}__container__text`,
  484. classList: [
  485. "css-146c3p1",
  486. "r-dnmrzs",
  487. "r-1udh08x",
  488. "r-3s2u2q",
  489. "r-bcqeeo",
  490. "r-1ttztb7",
  491. "r-qvutc0",
  492. "r-1qd0xha",
  493. "r-adyw6z",
  494. "r-135wba7",
  495. "r-16dba41",
  496. "r-dlybji",
  497. "r-nazi8o",
  498. ],
  499. });
  500. const textContent = Utils.createElement("span", {
  501. id: `${type}__text__content`,
  502. classList: ["1jxf684", "r-bcqeeo", "r-1ttztb7", "r-qvutc0", "r-poiln3"],
  503. textContent: textContentFunc(),
  504. });
  505. text.appendChild(textContent);
  506. container.appendChild(icon);
  507. container.appendChild(text);
  508. infoElement.appendChild(container);
  509. nav.appendChild(infoElement);
  510. if (type === "time") {
  511. setInterval(() => {
  512. textContent.textContent = textContentFunc();
  513. }, 1000);
  514. }
  515. },
  516. init: function () {
  517. this.createInfoElement("time");
  518. this.createInfoElement("date");
  519. const observer = new MutationObserver(() => {
  520. this.createInfoElement("time");
  521. this.createInfoElement("date");
  522. });
  523. observer.observe(document.body, { childList: true, subtree: true });
  524. },
  525. };
  526. // -----------------------------------------------------------------------------------
  527. // 動画プレイヤーをデフォルトに戻す
  528. // -----------------------------------------------------------------------------------
  529. const VideoModule = {
  530. setupDefaultVideoPlayer: function (container) {
  531. if (!config.useDefaultVideoPlayer) return;
  532. const video = container.querySelector("div:first-child > div > video");
  533. if (!video) return;
  534. video.controls = true;
  535. video.removeAttribute("disablepictureinpicture");
  536. video.muted = false;
  537. const onClick = (e) => {
  538. e.preventDefault();
  539. video
  540. .play()
  541. .then(() => {
  542. video.muted = false;
  543. })
  544. .catch((error) => console.error("Video playback error:", error));
  545. const onVolumeChange = (e) => {
  546. if (e.target.muted) {
  547. e.target.muted = false;
  548. }
  549. e.target.removeEventListener("volumechange", onVolumeChange);
  550. };
  551. e.target.addEventListener("volumechange", onVolumeChange);
  552. video.removeEventListener("click", onClick);
  553. };
  554. video.addEventListener("click", onClick);
  555. container.parentElement.appendChild(video);
  556. container.remove();
  557. },
  558. observeVideos: function () {
  559. const observer = new MutationObserver(() => {
  560. const videoContainer = document.body.querySelector(
  561. 'div[data-testid="videoComponent"]:not(.enhanced-video)'
  562. );
  563. if (videoContainer) {
  564. videoContainer.classList.add("enhanced-video");
  565. setTimeout(() => this.setupDefaultVideoPlayer(videoContainer), 100);
  566. }
  567. });
  568. observer.observe(document.body, { subtree: true, childList: true });
  569. },
  570. };
  571. // -----------------------------------------------------------------------------------
  572. // Tweet Engagements をアクセスしやすく
  573. // -----------------------------------------------------------------------------------
  574. const TweetEngagementModule = {
  575. createQuoteButton: function (tweetId) {
  576. if (!config.enhanceTweetEngagements) return;
  577. // ボタンを追加
  578. const tweetEngagementButton = Utils.createElement("a", {
  579. attributes: {
  580. href: `https://x.com${tweetId}/quotes`,
  581. "data-testid": "tweetEngagements",
  582. target: "_blank",
  583. rel: "noopener",
  584. },
  585. classList: [
  586. "css-175oi2r",
  587. "r-1777fci",
  588. "r-bt1l66",
  589. "r-bztko3",
  590. "r-lrvibr",
  591. "r-1loqt21",
  592. "r-1ny4l3l",
  593. "r-1wron08",
  594. ],
  595. });
  596. // クリックしたときリンクに飛ぶ
  597. tweetEngagementButton.addEventListener("click", (event) => {
  598. const tweetEngagementHref = event.currentTarget.getAttribute("href");
  599. window.open(tweetEngagementHref, "_blank");
  600. });
  601. // アイコン要素
  602. const tweetEngagementIconDiv = Utils.createElement("div", {
  603. attributes: { dir: "ltr" },
  604. classList: [
  605. "css-146c3p1",
  606. "r-bcqeeo",
  607. "r-1ttztb7",
  608. "r-qvutc0",
  609. "r-1qd0xha",
  610. "r-a023e6",
  611. "r-rjixqe",
  612. "r-16dba41",
  613. "r-1awozwy",
  614. "r-6koalj",
  615. "r-1h0z5md",
  616. "r-o7ynqc",
  617. "r-clp7b1",
  618. "r-3s2u2q",
  619. ],
  620. });
  621. // tweetEngagementButton 要素の親要素を確認
  622. const tweetEngagementParent = () => {
  623. let parent = tweetEngagementButton.parentElement;
  624. // 以下4つのクラスがあるないといはnullを返す
  625. while (parent) {
  626. if (
  627. parent.classList.contains("css-175oi2r") &&
  628. parent.classList.contains("r-1kbdv8c") &&
  629. parent.classList.contains("r-18u37iz") &&
  630. parent.classList.contains("r-1wtj0ep")
  631. ) {
  632. return parent;
  633. }
  634. parent = parent.parentElement;
  635. }
  636. return null;
  637. };
  638. const tweetEngagementIcon = () => {
  639. const parent = tweetEngagementParent();
  640. const tweetEngagementIconBaseClass =
  641. "r-4qtqp9 r-yyyyoo r-dnmrzs r-bnwqim r-lrvibr r-m6rgpd";
  642. // 親要素が見つからなかったとき tweetEngagementIconBaseClass を返す
  643. if (!parent) return tweetEngagementIconBaseClass;
  644. // 以下7つのクラスがあるとき(メイン投稿のとき)は r-50lct3 と r-1srniu (w,h: 1.5rem) を付加
  645. if (
  646. parent.classList.contains("r-1oszu61") &&
  647. parent.classList.contains("r-3qxfft") &&
  648. parent.classList.contains("r-n7gxbd") &&
  649. parent.classList.contains("r-2sztyj") &&
  650. parent.classList.contains("r-1efd50x") &&
  651. parent.classList.contains("r-5kkj8d") &&
  652. parent.classList.contains("r-h3s6tt") &&
  653. parent.classList.contains("r-1igl3o0") &&
  654. parent.classList.contains("r-rull8r") &&
  655. parent.classList.contains("r-qklmqi")
  656. ) {
  657. return `${tweetEngagementIconBaseClass} r-50lct3 r-1srniu`;
  658. }
  659. // そうでないとき(TL, リプのとき)は r-1xvli5t と r-1hdv0qi (w,h: 1.25rem) を付加
  660. return `${tweetEngagementIconBaseClass} r-1xvli5t r-1hdv0qi`;
  661. };
  662. // svg要素
  663. const tweetEngagementIconElement = document.createElementNS(
  664. "http://www.w3.org/2000/svg",
  665. "svg"
  666. );
  667. tweetEngagementIconElement.setAttribute("viewBox", "0 0 24 24");
  668. tweetEngagementIconElement.setAttribute("aria-hidden", "true");
  669. setTimeout(() => {
  670. tweetEngagementIconElement.setAttribute("class", tweetEngagementIcon());
  671. }, 0);
  672. tweetEngagementIconElement.innerHTML = `
  673. <g>
  674. <path d="M8.75 21V3h2v18h-2zM18 21V8.5h2V21h-2zM4 21l.004-10h2L6 21H4zm9.248 0v-7h2v7h-2z" transform="scale(0.75) translate(4, 0)" />
  675. <path d="M1.751 10c0-4.42 3.584-8 8.005-8h4.366c4.49 0 8.129 3.64 8.129 8.13 0 2.96-1.607 5.68-4.196 7.11l-8.054 4.46v-3.69h-.067c-4.49.1-8.183-3.51-8.183-8.01zm8.005-6c-3.317 0-6.005 2.69-6.005 6 0 3.37 2.77 6.08 6.138 6.01l.351-.01h1.761v2.3l5.087-2.81c1.951-1.08 3.163-3.13 3.163-5.36 0-3.39-2.744-6.13-6.129-6.13H9.756z" />
  676. </g>
  677. `;
  678. // アイコン背景要素
  679. const tweetEngagementBgDiv = document.createElement("div");
  680. tweetEngagementBgDiv.className =
  681. "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";
  682. tweetEngagementIconDiv.appendChild(tweetEngagementBgDiv);
  683. tweetEngagementIconDiv.appendChild(tweetEngagementIconElement);
  684. // ホバーイベント
  685. tweetEngagementIconDiv.style.textOverflow = "unset";
  686. tweetEngagementIconDiv.style.color = "rgb(113, 118, 123)";
  687. const tweetEngagementIconBgDiv =
  688. tweetEngagementIconDiv.querySelector("div");
  689. // ホバー時に色を変える
  690. tweetEngagementButton.addEventListener("mouseenter", () => {
  691. tweetEngagementIconDiv.style.color = "rgb(238 201 104)";
  692. if (tweetEngagementIconBgDiv) {
  693. tweetEngagementIconBgDiv.style.backgroundColor =
  694. "rgba(238, 201, 104, 0.1)";
  695. }
  696. });
  697. tweetEngagementButton.addEventListener("mouseleave", () => {
  698. tweetEngagementIconDiv.style.color = "rgb(113, 118, 123)";
  699. if (tweetEngagementIconBgDiv) {
  700. tweetEngagementIconBgDiv.style.backgroundColor = "";
  701. }
  702. });
  703. tweetEngagementButton.appendChild(tweetEngagementIconDiv);
  704. return tweetEngagementButton;
  705. },
  706. addQuoteElement: function () {
  707. const tweetEngagementTargetDivs = document.querySelectorAll(
  708. 'div[role="group"][id^="id__"]'
  709. );
  710. tweetEngagementTargetDivs.forEach((targetDiv) => {
  711. if (targetDiv.querySelector('[data-testid="tweetEngagements"]')) {
  712. return;
  713. }
  714. // ツイートIDの取得
  715. let article = targetDiv.closest("article");
  716. let tweetId = null;
  717. if (article) {
  718. const tweetEngagementStatusLink = article.querySelector(
  719. 'a[href*="/status/"]'
  720. );
  721. if (tweetEngagementStatusLink) {
  722. let tweetEngagementHref =
  723. tweetEngagementStatusLink.getAttribute("href");
  724. if (tweetEngagementHref) {
  725. // /photo/1 を除外
  726. tweetEngagementHref = tweetEngagementHref.replace("/photo/1", "");
  727. tweetId = tweetEngagementHref;
  728. }
  729. }
  730. }
  731. const tweetEngagementQuoteButton = this.createQuoteButton(tweetId);
  732. targetDiv.insertBefore(
  733. tweetEngagementQuoteButton,
  734. targetDiv.children[4]
  735. );
  736. });
  737. },
  738. init: function () {
  739. const debouncedAddQuoteElement = Utils.debounce(
  740. () => this.addQuoteElement(),
  741. 250
  742. );
  743. // URL変更の監視
  744. let lastUrl = location.href;
  745. const urlObserver = new MutationObserver(() => {
  746. const url = location.href;
  747. if (url !== lastUrl) {
  748. lastUrl = url;
  749. debouncedAddQuoteElement();
  750. }
  751. });
  752. urlObserver.observe(document, { subtree: true, childList: true });
  753. const timeline = document.querySelector(
  754. "div[data-testid='primaryColumn']"
  755. );
  756. if (timeline) {
  757. new MutationObserver(() => {
  758. debouncedAddQuoteElement();
  759. }).observe(timeline, { childList: true, subtree: true });
  760. }
  761. const retryInterval = setInterval(debouncedAddQuoteElement, 1000);
  762. window.addEventListener("popstate", debouncedAddQuoteElement);
  763. history.pushState = ((origPushState) => {
  764. return function (state, title, url) {
  765. origPushState.apply(this, arguments);
  766. debouncedAddQuoteElement();
  767. };
  768. })(history.pushState);
  769. history.replaceState = ((origReplaceState) => {
  770. return function (state, title, url) {
  771. origReplaceState.apply(this, arguments);
  772. debouncedAddQuoteElement();
  773. };
  774. })(history.replaceState);
  775. // DOMContentLoaded で初期化
  776. document.addEventListener("DOMContentLoaded", () => {
  777. debouncedAddQuoteElement();
  778. clearInterval(retryInterval);
  779. });
  780. debouncedAddQuoteElement();
  781. },
  782. };
  783. // -----------------------------------------------------------------------------------
  784. // メイン処理
  785. // -----------------------------------------------------------------------------------
  786. function main() {
  787. loadConfig();
  788. SettingsModule.createSettingsUI();
  789. setupKeyboardShortcut();
  790. setupMenuCommand();
  791. // タイムスタンプの更新を定期的に実行
  792. setInterval(() => TimestampModule.updateTimestamps(), 1000);
  793. // アイコン情報表示を初期化
  794. replaceTwitterIcons();
  795. // サイドバーの情報表示を初期化
  796. SidebarModule.init();
  797. // 動画プレイヤーの設定を監視
  798. VideoModule.observeVideos();
  799. // 引用ツイートボタンの追加を初期化
  800. TweetEngagementModule.init();
  801. }
  802. // ページ読み込み時にメイン処理を実行
  803. window.addEventListener("load", main);
  804. })();