main.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. class Home {
  2. static start() {
  3. this.itemQuery = { ImageTypes: "Backdrop", EnableImageTypes: "Logo,Backdrop", IncludeItemTypes: "Movie,Series", SortBy: "ProductionYear, PremiereDate, SortName", Recursive: true, ImageTypeLimit: 1, Limit: 10, Fields: "ProductionYear", SortOrder: "Descending", EnableUserData: false, EnableTotalRecordCount: false };
  4. this.coverOptions = { type: "Backdrop", maxWidth: 3000 };
  5. this.logoOptions = { type: "Logo", maxWidth: 3000 };
  6. if (window.location.href.indexOf("home") != -1) {
  7. const load = `
  8. <div class="misty-loading">
  9. <h1>MISTY MEDIA</h1>
  10. <div class="mdl-spinner"><div class="mdl-spinner__layer mdl-spinner__layer-1"><div class="mdl-spinner__circle-clipper mdl-spinner__left"><div class="mdl-spinner__circle mdl-spinner__circleLeft"></div></div><div class="mdl-spinner__circle-clipper mdl-spinner__right"><div class="mdl-spinner__circle mdl-spinner__circleRight"></div></div></div></div>
  11. </div>
  12. `;
  13. $("body").append(load);
  14. }
  15. CommonUtils.selectWait(".section0 .card", async () => {
  16. await this.initBanner();
  17. this.initEvent();
  18. });
  19. }
  20. static injectCode(code) {
  21. let hash = md5(code);
  22. return new Promise((resolve, reject) => {
  23. const channel = new BroadcastChannel(hash);
  24. channel.addEventListener("message", (event) => resolve(event.data));
  25. const script = `
  26. <script class="I${hash}">
  27. setTimeout(async ()=> {
  28. async function R${hash}(){${code}};
  29. const channel = new BroadcastChannel("${hash}");
  30. channel.postMessage(await R${hash}());
  31. document.querySelector("script.I${hash}").remove()
  32. }, 16)
  33. </script>
  34. `;
  35. $(document.head || document.documentElement).append(script);
  36. });
  37. }
  38. static injectCall(func, arg) {
  39. const script = `
  40. // const client = (await window.require(["ApiClient"]))[0];
  41. const client = await new Promise((resolve, reject) => {
  42. setInterval(() => {
  43. if (window.ApiClient != undefined) resolve(window.ApiClient);
  44. }, 16);
  45. });
  46. return await client.${func}(${arg})
  47. `;
  48. return this.injectCode(script);
  49. }
  50. static getItems(query) {
  51. return this.injectCall("getItems", "client.getCurrentUserId(), " + JSON.stringify(query));
  52. }
  53. static getItem(itemId) {
  54. return this.injectCall("getItem", `client.getCurrentUserId(), "${itemId}"`);
  55. }
  56. static getImageUrl(itemId, options) {
  57. return this.injectCall("getImageUrl", itemId + ", " + JSON.stringify(options));
  58. }
  59. /* 插入Banner */
  60. static async initBanner() {
  61. const banner = `
  62. <div class="misty-banner">
  63. <div class="misty-banner-body">
  64. </div>
  65. <div class="misty-banner-library">
  66. <div class="misty-banner-logos"></div>
  67. </div>
  68. </div>
  69. `;
  70. $(".homeSectionsContainer").prepend(banner);
  71. $(".section0").detach().appendTo(".misty-banner-library");
  72. // 插入数据
  73. const data = await this.getItems(this.itemQuery);
  74. console.log(data);
  75. data.Items.forEach(async (item) => {
  76. const detail = await this.getItem(item.Id),
  77. itemHtml = `
  78. <div class="misty-banner-item" id="${detail.Id}">
  79. <img draggable="false" loading="eager" decoding="async" class="misty-banner-cover" src="${await this.getImageUrl(detail.Id, this.coverOptions)}" alt="Backdrop" style="">
  80. <div class="misty-banner-info padded-left padded-right">
  81. <h1>${detail.Name}</h1>
  82. <div><p>${detail.Overview}</p></div>
  83. <div><button onclick="appRouter.showItem('${detail.Id}')">MORE</button></div>
  84. </div>
  85. </div>
  86. `,
  87. logoHtml = `
  88. <img id="${detail.Id}" draggable="false" loading="auto" decoding="lazy" class="misty-banner-logo" data-banner="img-title" alt="Logo" src="${await this.getImageUrl(detail.Id, this.logoOptions)}">
  89. `;
  90. if (detail.ImageTags && detail.ImageTags.Logo) {
  91. $(".misty-banner-logos").append(logoHtml);
  92. }
  93. $(".misty-banner-body").append(itemHtml);
  94. console.log(item.Id, detail);
  95. });
  96. let complete = 0;
  97. let loading = setInterval(async () => {
  98. // 判断图片加载完毕
  99. $(".misty-banner-cover:not(.complete)").each((i, dom) => {
  100. if (dom.complete) {
  101. dom.classList.add("complete");
  102. complete++;
  103. }
  104. });
  105. if (complete == $(".misty-banner-item").length && $(".misty-banner-item").length != 0) {
  106. clearInterval(loading);
  107. $(".misty-loading").fadeOut(500);
  108. await CommonUtils.sleep(150);
  109. // 置入场动画
  110. let delay = 80; // 动媒体库画间隔
  111. let id = $(".misty-banner-item").eq(0).addClass("active").attr("id"); // 初次信息动画
  112. $(`.misty-banner-logo[id=${id}]`).addClass("active");
  113. await CommonUtils.sleep(200); // 间隔动画
  114. $(".section0 > div").addClass("misty-banner-library-overflow"); // 关闭overflow 防止媒体库动画溢出
  115. $(".misty-banner .card").each((i, dom) => setTimeout(() => $(dom).addClass("misty-banner-library-show"), i * delay)); // 媒体库动画
  116. await CommonUtils.sleep(delay * 8 + 1000); // 等待媒体库动画完毕
  117. $(".section0 > div").removeClass("misty-banner-library-overflow"); // 开启overflow 防止无法滚动
  118. // 滚屏逻辑
  119. var index = 0;
  120. setInterval(async () => {
  121. // 背景切换
  122. index += index + 1 == $(".misty-banner-item").length ? -index : 1;
  123. $(".misty-banner-body").css("left", -(index * 100).toString() + "%");
  124. // 信息切换
  125. $(".misty-banner-item.active").removeClass("active");
  126. let id = $(".misty-banner-item").eq(index).addClass("active").attr("id");
  127. // LOGO切换
  128. $(".misty-banner-logo.active").removeClass("active");
  129. $(`.misty-banner-logo[id=${id}]`).addClass("active");
  130. }, 8000);
  131. }
  132. }, 16);
  133. }
  134. /* 初始事件 */
  135. static initEvent() {
  136. // 通过注入方式, 方可调用appRouter函数, 以解决Content-Script window对象不同步问题
  137. const script = `
  138. // 挂载appRouter
  139. if (!window.appRouter) window.appRouter = (await window.require(["appRouter"]))[0];
  140. // 获取带有 is 属性为 emby-itemscontainer 的元素
  141. const element = document.querySelector('.section0 div[is="emby-itemscontainer"]');
  142. // 创建新元素,并将旧元素的所有子节点添加到其中
  143. const newElement = document.createElement("div");
  144. while (element.firstChild) {
  145. newElement.appendChild(element.firstChild);
  146. }
  147. // 移除 is 属性以隔离原事件
  148. newElement.removeAttribute("is");
  149. // 复制原始元素的 class 属性到新元素中
  150. Array.from(element.classList).forEach(cls => {
  151. newElement.classList.add(cls);
  152. });
  153. // 将新元素插入到原始元素的位置上
  154. element.parentNode.insertBefore(newElement, element);
  155. // 重新挂载媒体库事件
  156. const librarys = document.querySelectorAll(".view:not(.hide) .section0 .card");
  157. librarys.forEach(library => {
  158. library.addEventListener("click", () => {
  159. const dataId = library.getAttribute("data-id");
  160. appRouter.showItem(dataId)
  161. });
  162. });
  163. `;
  164. this.injectCode(script);
  165. }
  166. }
  167. // 运行
  168. Home.start();