From 710cafefa72b0150b3a6e24df1ef7be551844424 Mon Sep 17 00:00:00 2001 From: Sophia Atkinson Date: Sat, 21 Jun 2025 18:01:38 -0700 Subject: [PATCH] initial commit --- LICENSE | 373 ++++++++++++++++++++++++++++++++++++++++++++++ README.MD | 44 ++++++ background.js | 1 + content.js | 187 +++++++++++++++++++++++ icons/icon128.png | Bin 0 -> 5500 bytes icons/icon16.png | Bin 0 -> 621 bytes icons/icon32.png | Bin 0 -> 1234 bytes icons/icon48.png | Bin 0 -> 1734 bytes manifest.json | 31 ++++ options.html | 99 ++++++++++++ options.js | 14 ++ 11 files changed, 749 insertions(+) create mode 100644 LICENSE create mode 100644 README.MD create mode 100644 background.js create mode 100644 content.js create mode 100644 icons/icon128.png create mode 100644 icons/icon16.png create mode 100644 icons/icon32.png create mode 100644 icons/icon48.png create mode 100644 manifest.json create mode 100644 options.html create mode 100644 options.js diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..d0a1fa1 --- /dev/null +++ b/LICENSE @@ -0,0 +1,373 @@ +Mozilla Public License Version 2.0 +================================== + +1. Definitions +-------------- + +1.1. "Contributor" + means each individual or legal entity that creates, contributes to + the creation of, or owns Covered Software. + +1.2. "Contributor Version" + means the combination of the Contributions of others (if any) used + by a Contributor and that particular Contributor's Contribution. + +1.3. "Contribution" + means Covered Software of a particular Contributor. + +1.4. "Covered Software" + means Source Code Form to which the initial Contributor has attached + the notice in Exhibit A, the Executable Form of such Source Code + Form, and Modifications of such Source Code Form, in each case + including portions thereof. + +1.5. "Incompatible With Secondary Licenses" + means + + (a) that the initial Contributor has attached the notice described + in Exhibit B to the Covered Software; or + + (b) that the Covered Software was made available under the terms of + version 1.1 or earlier of the License, but not also under the + terms of a Secondary License. + +1.6. "Executable Form" + means any form of the work other than Source Code Form. + +1.7. "Larger Work" + means a work that combines Covered Software with other material, in + a separate file or files, that is not Covered Software. + +1.8. "License" + means this document. + +1.9. "Licensable" + means having the right to grant, to the maximum extent possible, + whether at the time of the initial grant or subsequently, any and + all of the rights conveyed by this License. + +1.10. "Modifications" + means any of the following: + + (a) any file in Source Code Form that results from an addition to, + deletion from, or modification of the contents of Covered + Software; or + + (b) any new file in Source Code Form that contains any Covered + Software. + +1.11. "Patent Claims" of a Contributor + means any patent claim(s), including without limitation, method, + process, and apparatus claims, in any patent Licensable by such + Contributor that would be infringed, but for the grant of the + License, by the making, using, selling, offering for sale, having + made, import, or transfer of either its Contributions or its + Contributor Version. + +1.12. "Secondary License" + means either the GNU General Public License, Version 2.0, the GNU + Lesser General Public License, Version 2.1, the GNU Affero General + Public License, Version 3.0, or any later versions of those + licenses. + +1.13. "Source Code Form" + means the form of the work preferred for making modifications. + +1.14. "You" (or "Your") + means an individual or a legal entity exercising rights under this + License. For legal entities, "You" includes any entity that + controls, is controlled by, or is under common control with You. For + purposes of this definition, "control" means (a) the power, direct + or indirect, to cause the direction or management of such entity, + whether by contract or otherwise, or (b) ownership of more than + fifty percent (50%) of the outstanding shares or beneficial + ownership of such entity. + +2. License Grants and Conditions +-------------------------------- + +2.1. Grants + +Each Contributor hereby grants You a world-wide, royalty-free, +non-exclusive license: + +(a) under intellectual property rights (other than patent or trademark) + Licensable by such Contributor to use, reproduce, make available, + modify, display, perform, distribute, and otherwise exploit its + Contributions, either on an unmodified basis, with Modifications, or + as part of a Larger Work; and + +(b) under Patent Claims of such Contributor to make, use, sell, offer + for sale, have made, import, and otherwise transfer either its + Contributions or its Contributor Version. + +2.2. Effective Date + +The licenses granted in Section 2.1 with respect to any Contribution +become effective for each Contribution on the date the Contributor first +distributes such Contribution. + +2.3. Limitations on Grant Scope + +The licenses granted in this Section 2 are the only rights granted under +this License. No additional rights or licenses will be implied from the +distribution or licensing of Covered Software under this License. +Notwithstanding Section 2.1(b) above, no patent license is granted by a +Contributor: + +(a) for any code that a Contributor has removed from Covered Software; + or + +(b) for infringements caused by: (i) Your and any other third party's + modifications of Covered Software, or (ii) the combination of its + Contributions with other software (except as part of its Contributor + Version); or + +(c) under Patent Claims infringed by Covered Software in the absence of + its Contributions. + +This License does not grant any rights in the trademarks, service marks, +or logos of any Contributor (except as may be necessary to comply with +the notice requirements in Section 3.4). + +2.4. Subsequent Licenses + +No Contributor makes additional grants as a result of Your choice to +distribute the Covered Software under a subsequent version of this +License (see Section 10.2) or under the terms of a Secondary License (if +permitted under the terms of Section 3.3). + +2.5. Representation + +Each Contributor represents that the Contributor believes its +Contributions are its original creation(s) or it has sufficient rights +to grant the rights to its Contributions conveyed by this License. + +2.6. Fair Use + +This License is not intended to limit any rights You have under +applicable copyright doctrines of fair use, fair dealing, or other +equivalents. + +2.7. Conditions + +Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted +in Section 2.1. + +3. Responsibilities +------------------- + +3.1. Distribution of Source Form + +All distribution of Covered Software in Source Code Form, including any +Modifications that You create or to which You contribute, must be under +the terms of this License. You must inform recipients that the Source +Code Form of the Covered Software is governed by the terms of this +License, and how they can obtain a copy of this License. You may not +attempt to alter or restrict the recipients' rights in the Source Code +Form. + +3.2. Distribution of Executable Form + +If You distribute Covered Software in Executable Form then: + +(a) such Covered Software must also be made available in Source Code + Form, as described in Section 3.1, and You must inform recipients of + the Executable Form how they can obtain a copy of such Source Code + Form by reasonable means in a timely manner, at a charge no more + than the cost of distribution to the recipient; and + +(b) You may distribute such Executable Form under the terms of this + License, or sublicense it under different terms, provided that the + license for the Executable Form does not attempt to limit or alter + the recipients' rights in the Source Code Form under this License. + +3.3. Distribution of a Larger Work + +You may create and distribute a Larger Work under terms of Your choice, +provided that You also comply with the requirements of this License for +the Covered Software. If the Larger Work is a combination of Covered +Software with a work governed by one or more Secondary Licenses, and the +Covered Software is not Incompatible With Secondary Licenses, this +License permits You to additionally distribute such Covered Software +under the terms of such Secondary License(s), so that the recipient of +the Larger Work may, at their option, further distribute the Covered +Software under the terms of either this License or such Secondary +License(s). + +3.4. Notices + +You may not remove or alter the substance of any license notices +(including copyright notices, patent notices, disclaimers of warranty, +or limitations of liability) contained within the Source Code Form of +the Covered Software, except that You may alter any license notices to +the extent required to remedy known factual inaccuracies. + +3.5. Application of Additional Terms + +You may choose to offer, and to charge a fee for, warranty, support, +indemnity or liability obligations to one or more recipients of Covered +Software. However, You may do so only on Your own behalf, and not on +behalf of any Contributor. You must make it absolutely clear that any +such warranty, support, indemnity, or liability obligation is offered by +You alone, and You hereby agree to indemnify every Contributor for any +liability incurred by such Contributor as a result of warranty, support, +indemnity or liability terms You offer. You may include additional +disclaimers of warranty and limitations of liability specific to any +jurisdiction. + +4. Inability to Comply Due to Statute or Regulation +--------------------------------------------------- + +If it is impossible for You to comply with any of the terms of this +License with respect to some or all of the Covered Software due to +statute, judicial order, or regulation then You must: (a) comply with +the terms of this License to the maximum extent possible; and (b) +describe the limitations and the code they affect. Such description must +be placed in a text file included with all distributions of the Covered +Software under this License. Except to the extent prohibited by statute +or regulation, such description must be sufficiently detailed for a +recipient of ordinary skill to be able to understand it. + +5. Termination +-------------- + +5.1. The rights granted under this License will terminate automatically +if You fail to comply with any of its terms. However, if You become +compliant, then the rights granted under this License from a particular +Contributor are reinstated (a) provisionally, unless and until such +Contributor explicitly and finally terminates Your grants, and (b) on an +ongoing basis, if such Contributor fails to notify You of the +non-compliance by some reasonable means prior to 60 days after You have +come back into compliance. Moreover, Your grants from a particular +Contributor are reinstated on an ongoing basis if such Contributor +notifies You of the non-compliance by some reasonable means, this is the +first time You have received notice of non-compliance with this License +from such Contributor, and You become compliant prior to 30 days after +Your receipt of the notice. + +5.2. If You initiate litigation against any entity by asserting a patent +infringement claim (excluding declaratory judgment actions, +counter-claims, and cross-claims) alleging that a Contributor Version +directly or indirectly infringes any patent, then the rights granted to +You by any and all Contributors for the Covered Software under Section +2.1 of this License shall terminate. + +5.3. In the event of termination under Sections 5.1 or 5.2 above, all +end user license agreements (excluding distributors and resellers) which +have been validly granted by You or Your distributors under this License +prior to termination shall survive termination. + +************************************************************************ +* * +* 6. Disclaimer of Warranty * +* ------------------------- * +* * +* Covered Software is provided under this License on an "as is" * +* basis, without warranty of any kind, either expressed, implied, or * +* statutory, including, without limitation, warranties that the * +* Covered Software is free of defects, merchantable, fit for a * +* particular purpose or non-infringing. The entire risk as to the * +* quality and performance of the Covered Software is with You. * +* Should any Covered Software prove defective in any respect, You * +* (not any Contributor) assume the cost of any necessary servicing, * +* repair, or correction. This disclaimer of warranty constitutes an * +* essential part of this License. No use of any Covered Software is * +* authorized under this License except under this disclaimer. * +* * +************************************************************************ + +************************************************************************ +* * +* 7. Limitation of Liability * +* -------------------------- * +* * +* Under no circumstances and under no legal theory, whether tort * +* (including negligence), contract, or otherwise, shall any * +* Contributor, or anyone who distributes Covered Software as * +* permitted above, be liable to You for any direct, indirect, * +* special, incidental, or consequential damages of any character * +* including, without limitation, damages for lost profits, loss of * +* goodwill, work stoppage, computer failure or malfunction, or any * +* and all other commercial damages or losses, even if such party * +* shall have been informed of the possibility of such damages. This * +* limitation of liability shall not apply to liability for death or * +* personal injury resulting from such party's negligence to the * +* extent applicable law prohibits such limitation. Some * +* jurisdictions do not allow the exclusion or limitation of * +* incidental or consequential damages, so this exclusion and * +* limitation may not apply to You. * +* * +************************************************************************ + +8. Litigation +------------- + +Any litigation relating to this License may be brought only in the +courts of a jurisdiction where the defendant maintains its principal +place of business and such litigation shall be governed by laws of that +jurisdiction, without reference to its conflict-of-law provisions. +Nothing in this Section shall prevent a party's ability to bring +cross-claims or counter-claims. + +9. Miscellaneous +---------------- + +This License represents the complete agreement concerning the subject +matter hereof. If any provision of this License is held to be +unenforceable, such provision shall be reformed only to the extent +necessary to make it enforceable. Any law or regulation which provides +that the language of a contract shall be construed against the drafter +shall not be used to construe this License against a Contributor. + +10. Versions of the License +--------------------------- + +10.1. New Versions + +Mozilla Foundation is the license steward. Except as provided in Section +10.3, no one other than the license steward has the right to modify or +publish new versions of this License. Each version will be given a +distinguishing version number. + +10.2. Effect of New Versions + +You may distribute the Covered Software under the terms of the version +of the License under which You originally received the Covered Software, +or under the terms of any subsequent version published by the license +steward. + +10.3. Modified Versions + +If you create software not governed by this License, and you want to +create a new license for such software, you may create and use a +modified version of this License if you rename the license and remove +any references to the name of the license steward (except to note that +such modified license differs from this License). + +10.4. Distributing Source Code Form that is Incompatible With Secondary +Licenses + +If You choose to distribute Source Code Form that is Incompatible With +Secondary Licenses under the terms of this version of the License, the +notice described in Exhibit B of this License must be attached. + +Exhibit A - Source Code Form License Notice +------------------------------------------- + + This Source Code Form is subject to the terms of the Mozilla Public + License, v. 2.0. If a copy of the MPL was not distributed with this + file, You can obtain one at https://mozilla.org/MPL/2.0/. + +If it is not possible or desirable to put the notice in a particular +file, then You may include the notice in a location (such as a LICENSE +file in a relevant directory) where a recipient would be likely to look +for such a notice. + +You may add additional accurate notices of copyright ownership. + +Exhibit B - "Incompatible With Secondary Licenses" Notice +--------------------------------------------------------- + + This Source Code Form is "Incompatible With Secondary Licenses", as + defined by the Mozilla Public License, v. 2.0. diff --git a/README.MD b/README.MD new file mode 100644 index 0000000..150df1b --- /dev/null +++ b/README.MD @@ -0,0 +1,44 @@ +# Apple Music Now Playing Extension + +This Extension has native support for + +- Chrome +- Chromium +- Edge +- Opera/Opera GX +- Brave +- Vivaldi + +This Extension does not yet have native support for + +- Firefox +- Tor Browser +- LibreWolf +- Anything Firefox based + +# [Roadmap](https://md.shork.ch/s/G6zQvfP0u) + +# Acknowledgments + +This project was made possible thanks to the **PreMiD Apple Music integration**. Huge thanks to the amazing contributors behind the original PreMiD integration codebase: + +* [SeMiD](https://github.com/PreMiD-Automations) +* [stawkey](https://github.com/stawkey) +* [callumok2004](https://github.com/callumok2004) +* [wenewen](https://github.com/w3new3n) +* [Florian Metz](https://github.com/Timeraa) +* [AshMW](https://github.com/AshMW2724) +* [Daniel Lau](https://github.com/theusaf) +* [Vasilis](https://github.com/down-bad) +* [Slowlife](https://github.com/Slowlife01) +* [Smaltin](https://github.com/Smaltin) +* [Dark_Ville](https://github.com/DarkVillager) +* [Bas van Zanten](https://github.com/Bas950) +* [Alexx](https://github.com/puppyonline) +* [Nam Anh](https://github.com/kyrie25) +* [Frederik](https://github.com/N0chteil) +* [Rodry](https://github.com/ImRodry) +* [Jack](https://github.com/i1u5) +* [Rhys](https://github.com/EncryptedDev) + +Check out the original project here: [PreMiD Apple Music Integration](https://github.com/PreMiD/Activities/tree/main/websites/A/Apple%20Music) diff --git a/background.js b/background.js new file mode 100644 index 0000000..5ef8030 --- /dev/null +++ b/background.js @@ -0,0 +1 @@ +// Just a goob :P \ No newline at end of file diff --git a/content.js b/content.js new file mode 100644 index 0000000..13bfa33 --- /dev/null +++ b/content.js @@ -0,0 +1,187 @@ +let pauseStartTime = null; +let endpoint = null; +let token = null; +let intervalId = null; + +async function imageUrlToBase64(url) { + try { + const response = await fetch(url); + const blob = await response.blob(); + return await new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onloadend = () => resolve(reader.result); + reader.onerror = reject; + reader.readAsDataURL(blob); + }); + } catch (e) { + console.error("Failed to fetch and convert image:", e); + return null; + } +} + +async function getNowPlayingData() { + const audio = document.querySelector('audio#apple-music-player'); + + // From PreMiD Apple Music Presence + const video = document.querySelector('apple-music-video-player')?.shadowRoot + ?.querySelector('amp-window-takeover > .container > amp-video-player-internal') + ?.shadowRoot?.querySelector('amp-video-player')?.shadowRoot + ?.querySelector('div#video-container')?.querySelector('video#apple-music-video-player'); + + const media = video || audio; + + if (!media) { + pauseStartTime = null; + return { + details: "Nothing playing", + state: "", + artworkBase64: "data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%22150%22%20height%3D%22150%22%20viewBox%3D%220%200%20150%20150%22%3E%3Crect%20width%3D%22100%25%22%20height%3D%22100%25%22%20fill%3D%22%231E1F22%22%2F%3E%3Cpath%20fill%3D%22%23FFF%22%20d%3D%22M31.66%2073.75h1.61q.59%200%201.03-.15t.73-.42q.29-.28.44-.68.14-.39.14-.88%200-.46-.14-.84-.15-.38-.43-.64-.29-.26-.73-.4t-1.04-.14h-1.61zm-2.15-5.82h3.76q1.16%200%202.01.27.84.28%201.4.76.55.49.82%201.17.26.68.26%201.49%200%20.84-.28%201.54t-.84%201.2q-.56.51-1.4.79t-1.97.28h-1.61v4.07h-2.15zm9.61-.32h1.98V79.5h-1.98zm8.29%209.77V76q-.85.04-1.44.14-.58.11-.93.28-.36.17-.51.39-.15.23-.15.49%200%20.52.31.74.31.23.8.23.61%200%201.05-.22.45-.22.87-.67m-4.17-4.31-.36-.63q1.42-1.3%203.41-1.3.72%200%201.29.24.57.23.96.65t.6%201.01q.2.58.2%201.28v5.18h-.9q-.28%200-.43-.08-.15-.09-.24-.34l-.17-.6q-.32.28-.61.5-.3.21-.62.35-.32.15-.68.22-.37.08-.81.08-.52%200-.96-.14t-.76-.42-.49-.7q-.18-.41-.18-.97%200-.31.11-.62.1-.3.34-.58.23-.28.61-.53.37-.25.92-.43.55-.19%201.28-.3.73-.12%201.66-.14v-.48q0-.83-.35-1.22-.35-.4-1.02-.4-.48%200-.79.11-.32.11-.56.25t-.43.26q-.2.11-.44.11-.21%200-.35-.11-.15-.11-.23-.25m13.83-1.78h1.58l-4.47%2010.47q-.09.2-.23.31t-.43.11h-1.47l1.54-3.3-3.32-7.59H52q.24%200%20.37.11.13.12.19.26l1.75%204.26q.09.21.15.43.06.21.11.44.07-.23.15-.45.07-.22.16-.43l1.64-4.25q.07-.16.22-.26.14-.11.33-.11m2.63%200h1.98v8.21H59.7zm2.26-2.39q0%20.26-.11.48-.1.22-.27.39-.18.17-.41.27t-.49.1-.49-.1-.39-.27q-.17-.17-.27-.39t-.1-.48.1-.5q.1-.23.27-.4.16-.16.39-.26t.49-.1.49.1.41.26q.17.17.27.4.11.24.11.5m3.45%202.75.14.65q.25-.25.52-.46.28-.21.59-.36.3-.15.66-.23.35-.09.76-.09.68%200%201.2.23t.86.64q.35.41.53.99.18.57.18%201.26v5.22h-1.97v-5.22q0-.76-.35-1.17-.34-.41-1.05-.41-.51%200-.96.23-.44.23-.84.63v5.94H63.7v-8.21h1.21q.38%200%20.5.36m10.31%203.55q.37%200%20.64-.1.28-.1.46-.28t.27-.42q.09-.25.09-.55%200-.61-.36-.96-.37-.36-1.1-.36-.72%200-1.09.36-.36.35-.36.96%200%20.29.09.54t.27.43.46.28q.27.1.63.1m2.24%204.67q0-.24-.15-.39-.14-.16-.39-.24t-.58-.12-.7-.06l-.77-.04q-.4-.02-.77-.06-.33.18-.54.43-.2.25-.2.57%200%20.22.11.41t.34.32q.24.14.61.21.38.08.92.08.56%200%20.96-.08.4-.09.66-.24.26-.14.38-.35.12-.2.12-.44m-.4-8.3h2.36v.74q0%20.35-.42.43l-.74.14q.17.42.17.92%200%20.61-.24%201.1-.25.5-.68.84t-1.02.53-1.27.19q-.24%200-.46-.02-.22-.03-.44-.07-.38.23-.38.52%200%20.25.22.37.23.11.61.16.37.05.85.06t.99.05q.5.04.98.14t.86.32q.37.21.6.59.23.37.23.95%200%20.55-.27%201.06t-.77.91q-.51.4-1.25.64-.74.25-1.69.25-.92%200-1.61-.18t-1.15-.48q-.45-.3-.68-.69-.22-.4-.22-.82%200-.58.35-.96.35-.39.95-.62-.32-.17-.52-.45-.19-.28-.19-.74%200-.18.07-.38.07-.19.2-.39.13-.19.33-.36t.47-.31q-.62-.33-.98-.89-.35-.56-.35-1.32%200-.6.24-1.1.25-.49.68-.84.44-.34%201.03-.53.6-.18%201.3-.18.53%200%201%20.11.46.1.84.31m15.67-3.64h1.89V79.5h-1.1q-.26%200-.43-.08-.17-.09-.33-.29l-6.04-7.71q.05.53.05.98v7.1h-1.9V67.93h1.13q.14%200%20.24.01.1.02.17.05.08.04.15.11.07.06.16.18l6.06%207.74-.04-.55q-.01-.27-.01-.51zm7.79%203.23q.91%200%201.66.3t1.28.84.82%201.33q.29.78.29%201.75%200%20.98-.29%201.76t-.82%201.34q-.53.55-1.28.84-.75.3-1.66.3-.92%200-1.67-.3-.75-.29-1.29-.84-.53-.56-.82-1.34t-.29-1.76q0-.97.29-1.75.29-.79.82-1.33.54-.54%201.29-.84t1.67-.3m0%206.94q1.02%200%201.52-.69.49-.69.49-2.01%200-1.33-.49-2.03-.5-.69-1.52-.69-1.04%200-1.54.7t-.5%202.02.5%202.01%201.54.69m10.66-6.81h1.97v8.21h-1.21q-.39%200-.49-.36l-.14-.66q-.5.52-1.11.83-.61.32-1.43.32-.67%200-1.19-.23t-.87-.64q-.35-.42-.53-.99t-.18-1.26v-5.22h1.98v5.22q0%20.75.34%201.16.35.41%201.05.41.51%200%20.96-.22.45-.23.85-.63zm6.66%208.34q-1.07%200-1.64-.61-.58-.6-.58-1.66v-4.59h-.84q-.16%200-.27-.1-.11-.11-.11-.31v-.79l1.32-.21.42-2.24q.04-.16.15-.25t.29-.09h1.02v2.58h2.19v1.41h-2.19v4.45q0%20.38.19.6t.51.22q.19%200%20.31-.05.13-.04.22-.09l.16-.09q.07-.05.15-.05t.14.05q.06.04.12.13l.59.96q-.43.36-.99.54-.56.19-1.16.19%22%2F%3E%3C%2Fsvg%3E", + paused: true + }; + } + + // From PreMiD Apple Music Presence + const timestampInput = document.querySelector('amp-lcd.lcd.lcd__music')?.shadowRoot + ?.querySelector('input#playback-progress[aria-valuenow][aria-valuemax]'); + + const paused = media.paused || media.readyState <= 2; + const metadata = navigator.mediaSession?.metadata || {}; + const title = metadata.title || media.title || "Unknown Title"; + const album = metadata.album || "Unknown Album"; + const artist = metadata.artist || "Unknown Artist"; + const artworkSrc = metadata.artwork?.[0]?.src || null; + + const largeImageKey = artworkSrc?.replace(/\d+x\d+[a-z]*/i, '150x150bb'); // More robust for Apple Music artwork URLs + const artworkBase64 = artworkSrc ? await imageUrlToBase64(largeImageKey) : null; + + const currentTime = timestampInput ? Number(timestampInput.getAttribute('aria-valuenow')) : media.currentTime; + const duration = timestampInput ? Number(timestampInput.getAttribute('aria-valuemax')) : media.duration; + + if (paused) { + if (!pauseStartTime) pauseStartTime = Date.now(); + if (Date.now() - pauseStartTime >= 300000) { + return { + details: "Not playing", + state: "", + paused: true + }; + } else { + return { + details: title, + state: artist, + album, + smallImageKey: 'play', + smallImageText: 'Playing', + artworkBase64, + startTimestamp: Math.floor(Date.now() / 1000) - Math.floor(currentTime), + endTimestamp: Math.floor(Date.now() / 1000) + Math.floor(duration - currentTime), + paused: true, + currentTime, + duration + }; + } + } else { + pauseStartTime = null; + } + + return { + details: title, + state: artist, + album, + smallImageKey: 'play', + smallImageText: 'Playing', + artworkBase64, + startTimestamp: Math.floor(Date.now() / 1000) - Math.floor(currentTime), + endTimestamp: Math.floor(Date.now() / 1000) + Math.floor(duration - currentTime), + paused: false, + currentTime, + duration + }; +} + +async function sendNowPlaying() { + if (!endpoint || !token) { + console.warn("Endpoint or token not loaded."); + return; + } + + const data = await getNowPlayingData(); + + try { + const response = await fetch(endpoint, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + 'Authorization': `Bearer ${token}` + }, + body: JSON.stringify(data), + }); + if (response.ok) { + console.log('Now playing data sent:', data.details, '-', data.state); + } else { + console.error('Failed to send now playing data:', response.status); + } + } catch (e) { + console.error('Error sending now playing data:', e); + } +} + +function safeCall(fn) { + try { + fn().catch(e => { + if (e.message?.includes("Extension context invalidated")) { + console.warn("Extension context invalidated, clearing interval."); + clearInterval(intervalId); + } else { + console.error("Unexpected async error:", e); + } + }); + } catch (e) { + console.error("Unexpected sync error:", e); + } +} +function startNowPlayingLoop() { + if (window._nowPlayingIntervalSet) return; + + window._nowPlayingIntervalSet = true; + + intervalId = setInterval(() => safeCall(sendNowPlaying), 1000); +} +if (!window._nowPlayingExtensionLogged) { + console.log('Apple Music Now Playing Extension made by Sophia Atkinson with help from PreMiD contributors'); + window._nowPlayingExtensionLogged = true; +} + +chrome.storage.sync.get(['endpoint', 'token'], (result) => { + if (chrome.runtime.lastError) { + console.error("Failed to load settings:", chrome.runtime.lastError.message); + return; + } + + endpoint = result.endpoint; + token = result.token; + + if (!endpoint || !token) { + console.warn("No endpoint/token configured."); + return; + } + + startNowPlayingLoop(); +}); +let lastUrl = location.href; + +new MutationObserver(() => { + if (location.href !== lastUrl) { + console.log('[NowPlaying] Route change detected.'); + lastUrl = location.href; + + clearInterval(intervalId); + window._nowPlayingIntervalSet = false; + + setTimeout(() => { + if (!window._nowPlayingIntervalSet) { + startNowPlayingLoop(); + } + }, 1000); + } +}).observe(document, { subtree: true, childList: true }); diff --git a/icons/icon128.png b/icons/icon128.png new file mode 100644 index 0000000000000000000000000000000000000000..b9647d0f29b14b4585e59ec9e0e059e19d0ae79f GIT binary patch literal 5500 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 literal 0 HcmV?d00001 diff --git a/icons/icon16.png b/icons/icon16.png new file mode 100644 index 0000000000000000000000000000000000000000..4cbe46f6c28bbdf9d5d04eef35d21ada0a935f6c GIT binary patch literal 621 zcmV-z0+RiSP)Px%CrLy>R5(wCQ%z_SQ4s!ScQ3tKZ8Rd;OSLu8`U47zHrk?qJsDrsGg+Q60Lf1UoL`*UIUN59rIqycC$shm)V*3&EGdO z1OVmU5!Z3#DZt7yv#V`Odwk0tgbnN=dGy>Q5uL8N zX@Bh()vXoFfTl*^$1emwe~0f(lqY$3&?jI~+1BxHP`VkESqmYAiRV9jid|D5qTLlp z_Vk+owai*t8`j}Yb0bQn5>ly@k)G$_>)i)fbN^YiQ$g~;8J~z;Fat!;k~X$rs%snC zTQlGnZ%sxnegSpvL~wOt0S5?@`%e1Ax&j6fftg{owE~Id;OB3QP9W%a`+Pstm0O_> zB0+NRiA5Zc4xqEneyW?MHJn3m`DSR-lOVbKm>yvriL_~bHNebVT>yJz5bDx+)Fw#o zIAS<3rPf;5k};)YX0F;&G68h8gH|;`jok=wFcoY+mi9%4z_lOW3;^?SUsHvJf9l{ri(x_0q4<#v8miBg6U#xjBWd01ITKC zHP8oppbz}*JJi`>Bh7{<6XAdCMhYa;9g~$(%qnXjR<8uVn~mZAp)%*G+>&+e+A$gG zM&`J{QCv+uY;`B{|FR00000NkvXX Hu0mjf6N(jT literal 0 HcmV?d00001 diff --git a/icons/icon32.png b/icons/icon32.png new file mode 100644 index 0000000000000000000000000000000000000000..df77753f42307c57d5bc53f0d408b71a38e0c581 GIT binary patch literal 1234 zcmV;@1TFiCP)Px(j7da6R9Hv7S8YrjMHqhG-39-vE!sa^e`uN%>qkq}G=?JuQzQ79rc@AE=p_YPDs_}@422$cr;_dB@H#4`(-5$Gl zrqoF`b8|EEyw5!IzVFNsTxF>t6woxSiT}+Q3lILBMDT zg~3=vNNG!YuWMM5l1l?!lE9Ar+?-(ZKh}xT%V*Y8^0kg;Qk0Rqm7dnz58C($VApFO zfHgPMl0XrK8&bPF^G0%`{2Lt|6^6^p%f*<+-~I-hn)O50DOum{c)#$0i~^1?WW~Aq zZWAv4F)NJNE1jF0a}4OZE`G=3al~RVG~Ql>b@P^!C-U$a)5)l7ivuVs{FG1&TE~VR z0##L2j@+CS|FVmDv=59E@Gd$fUucOFQLq^AS37s(d(Fr(YGbqcJHWystjQVB!XlK; z-7t^z3I#l)i%2J9d$TJ6d8b;XvlUA1RtRJBcf{)0Ta=tp!3oH5=*o|NOQngOW z+Qy#{t8cg<0rm5KF-|Dp z>jT@~lunpA)OZxwr6xOK=uzvbk;;oFYlTp2^ z_$WBMM=H>JuCM~*pTO!p55@`jDp0kvXaeeyy;6aX&K4w~T{#;acrigDKUX-~ZL*6Wnq`g-vS9DPSBaB|RB0eW~Xw97+C?GJ-p zpZ1OdPQZBf)#9Vz*gmPiiBI!D>B)`YpVGS1=`;5o_ay+Hijz;j?4y94!z89Q^dZ<^ z3oyF}GDc)SJyjw4+$GiV0ocS%Z?!}osH_(Xc%@=$YY=;=wC|Fu|K(KUzWxqYBq|liM1iud49fW)(7G~Vko`FiGZFZODr&g=+z(b<6;(biW&QDnkakrI zQ41|?2nJMD{hb4QT^wwqtKDBZeMLU{{@D-=r+@f7Oza1P*Tdz%_oufma^H+DA0|Qt zi-keO-Hz{e8wk)#Ze6XmRTp@igqBJ*35UFgMXdbSAMzikw#}u$GXYgqf&j@YFA%W3 wJ*O7=B6E7$r($-!+@ByK@meX#GHl!X7kOB8@ydv05C8xG07*qoM6N<$f|!0$-2eap literal 0 HcmV?d00001 diff --git a/icons/icon48.png b/icons/icon48.png new file mode 100644 index 0000000000000000000000000000000000000000..d875a84ba4a70a11c6433706772bc8df6f0427b2 GIT binary patch literal 1734 zcmV;%208hOP)Px*fJsC_RA@uhT3u`uRTMtw&TNgZ3KV(ik_a)hHKf(pNVK~Jv7tf{C~W}UUCK`Z z38-MiAltt1BcKovL699tZTL|m5?>6uZ6rn$d?~*MwA&Y;1$=GTcILY0-p+Qr-P!5P zQfY6pnVq?F?mgf6&bjCQ41!JRsoUsPn8yd812R@XuXE15c5-g&5ius(H?WiX)ARCL zpVW4WXI2HRDteb~3Il+z|K~=ZM$meZi4~$07nS)J&fxepc9SPrhEY`>I zPXSDL8ld|D@O%Fbn}5N=*#naWfZm1cXsi+bG}2e}mjO7ETo$V$sw=)=kwa7K?$`>xHH!IT98W6o~usX~m#7eC()y00?34{07Rnjn1GC zew)=+0jql>YZIjSi4!NFyu2Lfe`jZ>xCR0N2!%rM&mWgSU%l0-8jaXHZRsqZVtc0fh#B8Wz#LK!NgoZs&k3X_$|`egA)BqD$iAX1*5(NDP0iY<=RNaQUI z<5l&M#4Iq#$#)?o1{~b0$9DnC9|s*B9RdJZ*es+x-J8LhT9>LV@4gRu{g#0k7)JlV zO#rzv@YPOeyM863%F{{><1?4$9s|vvNgrPJt}6hlTY2OXK+bufUjf}E0L7)e zjDd7cLvUUE*V(7&K(2p2@UF-;o4BF(7=^+(|SS_vR8adEdX&m{zZT zdss5Cs6pc%)~Fwb;N+O zZ3AFwvGCi}A}m4RG+b!D?7OY+&r^9LT zth5-EEV3egzH9-0>{Ld1tC*58P+B!C2Gk$E0-Tg@4d8Yc^v$cctj{LTZDg}D&|5Oc zc3Xu_$_M5A@$4>ey9@NW%PqG>fZ8~+V~7!q^A~`hzW~Ocm)&e_HsZtSmEiI1gdZ^h&R) z>vTR~>w^>vP(6D9#HDu7&;4X8H&Wa7x}wPQfq3CeVt~%b#!}JpuKpa72OQHX!U+Z% z$J7xM=b^l7zi0wL&vQO6LzHL3W240Ku15zB`AiThfQSXNWL$)gf_en-dv3H2K4vAo z9!`DLHYYrGg2;7l{wfb1($2KS19B(DFrq*obd08BSAZBAM#yvPqV=Z+CaNB<&&$-l z0FIK6J!sPQ#I1|N``9DhMS8qtUPWPIh(%-tU>K0t4H%JN7oXc01^{EgxzU;T_iss8 cv2xx20GIhtFc)fnod5s;07*qoM6N<$f)so=_5c6? literal 0 HcmV?d00001 diff --git a/manifest.json b/manifest.json new file mode 100644 index 0000000..ac43127 --- /dev/null +++ b/manifest.json @@ -0,0 +1,31 @@ +{ + "manifest_version": 3, + "name": "Apple Music Now Playing", + "version": "1.3.0", + "description": "Send Apple Music now playing info to a custom endpoint.", + "permissions": [ + "storage", + "scripting", + "activeTab" + ], + "host_permissions": [ + "https://music.apple.com/*", + "https://beta.music.apple.com/", + "https://classical.music.apple.com/*", + "" + ], + "options_page": "options.html", + "content_scripts": [ + { + "matches": ["https://music.apple.com/*"], + "js": ["content.js"] + } + ], + "icons": { + "16": "icons/icon16.png", + "32": "icons/icon32.png", + "48": "icons/icon48.png", + "128": "icons/icon128.png" + } + +} diff --git a/options.html b/options.html new file mode 100644 index 0000000..ce24c18 --- /dev/null +++ b/options.html @@ -0,0 +1,99 @@ + + + + + AmNP Settings (Beta) + + + +
+

AmNP Settings (Beta)

+ + + + + +
+ + + diff --git a/options.js b/options.js new file mode 100644 index 0000000..95f5ac9 --- /dev/null +++ b/options.js @@ -0,0 +1,14 @@ +document.addEventListener('DOMContentLoaded', () => { + chrome.storage.sync.get(['endpoint', 'token'], (data) => { + document.getElementById('endpoint').value = data.endpoint || ''; + document.getElementById('token').value = data.token || ''; + }); + + document.getElementById('save').addEventListener('click', () => { + const endpoint = document.getElementById('endpoint').value; + const token = document.getElementById('token').value; + chrome.storage.sync.set({ endpoint, token }, () => { + alert('Settings saved.'); + }); + }); +});