Compare commits
122 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
35d92862aa | ||
|
|
693bb35aac | ||
|
|
1ddaab325e | ||
|
|
81facbdb00 | ||
|
|
34a58ac4bd | ||
|
|
2d8444701d | ||
|
|
66f5353659 | ||
|
|
b58264998a | ||
|
|
76c0bc5b3b | ||
|
|
1428db3e1e | ||
|
|
1f5f436e44 | ||
|
|
46f5c3648d | ||
|
|
555e2ab8be | ||
|
|
26eadfc0aa | ||
|
|
142f538939 | ||
|
|
b64e80fd60 | ||
|
|
fbf5fc7edf | ||
|
|
8defba4623 | ||
|
|
7f968ee050 | ||
|
|
dec5bbe39e | ||
|
|
63f3211cc4 | ||
|
|
4270235c78 | ||
|
|
76d8a67914 | ||
|
|
1a3caf5da6 | ||
|
|
3b3ef77e61 | ||
|
|
ba580b1b52 | ||
|
|
0a6284c716 | ||
|
|
f83e863664 | ||
|
|
747e8ed6bc | ||
|
|
30845442b2 | ||
|
|
bb83201736 | ||
|
|
457ae404ba | ||
|
|
b6d679f6ef | ||
|
|
3b88a1809d | ||
|
|
4614ce4a7a | ||
|
|
57840bb149 | ||
|
|
dd90a4630a | ||
|
|
b5d5e5706e | ||
|
|
a4b5cf5b6b | ||
|
|
353bda10df | ||
|
|
0e1b91d93c | ||
|
|
9363008d07 | ||
|
|
faec7d8941 | ||
|
|
7cc70854c4 | ||
|
|
9432f7aa86 | ||
|
|
4f7243bc74 | ||
|
|
ee724fedc8 | ||
|
|
a1dbd4eb12 | ||
|
|
236d8d9e70 | ||
|
|
6d55ae7524 | ||
|
|
99a0613893 | ||
|
|
61952a0af7 | ||
|
|
eca6ba96fb | ||
|
|
c2f0f01689 | ||
|
|
30d17baff4 | ||
|
|
96bb1a3744 | ||
|
|
772a0dae40 | ||
|
|
40c4454397 | ||
|
|
e5915e715a | ||
|
|
c171fc15f5 | ||
|
|
a749b1f98e | ||
|
|
6ccf6201b4 | ||
|
|
a69c741a39 | ||
|
|
d54b4f9b07 | ||
|
|
2cd427b6e9 | ||
|
|
55c1f8b191 | ||
|
|
fc3d6efd1c | ||
|
|
5ba5940e5f | ||
|
|
621b7da344 | ||
|
|
268ce5e307 | ||
|
|
412cc2d981 | ||
|
|
949df24bdb | ||
|
|
b987969200 | ||
|
|
3306bb703d | ||
|
|
6587a4e3d0 | ||
|
|
f794b71f44 | ||
|
|
34363c502a | ||
|
|
add2f7a551 | ||
|
|
1d7e9e27ec | ||
|
|
6459653328 | ||
|
|
9d738e6061 | ||
|
|
8f5a3650e6 | ||
|
|
229f9fe5ab | ||
|
|
0686129590 | ||
|
|
cb0392eb0d | ||
|
|
ed13e05b82 | ||
|
|
310fb4d496 | ||
|
|
78d25106db | ||
|
|
a328171a8a | ||
|
|
361559cbec | ||
|
|
e08bf66a53 | ||
|
|
d6ef81138d | ||
|
|
35f21e680a | ||
|
|
705fbaed9d | ||
|
|
9e52198ef7 | ||
|
|
b1943dfe17 | ||
|
|
c55e900c0f | ||
|
|
503e9addee | ||
|
|
d630fdd217 | ||
|
|
7e4a7c2a6e | ||
|
|
1716a771f3 | ||
|
|
36347cc4b0 | ||
|
|
7f94164e55 | ||
|
|
cbab7de546 | ||
|
|
d0de5cd021 | ||
|
|
16628e9902 | ||
|
|
72bfe0a14a | ||
|
|
6498ec4216 | ||
|
|
0d350fc76b | ||
|
|
2c6e4ce610 | ||
|
|
0c552774dc | ||
|
|
9ab605bb74 | ||
|
|
3d6cba0fe4 | ||
|
|
32e5e2b690 | ||
|
|
c967c1e308 | ||
|
|
ae28d5219b | ||
|
|
e4228f889e | ||
|
|
6d721c755e | ||
|
|
6948953778 | ||
|
|
8a50cef330 | ||
|
|
a0bf5370bd | ||
|
|
c5800b431d |
@@ -51,7 +51,31 @@ jobs:
|
|||||||
echo "$CHANGELOG" >> $GITHUB_ENV
|
echo "$CHANGELOG" >> $GITHUB_ENV
|
||||||
echo "EOF" >> $GITHUB_ENV
|
echo "EOF" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Check if Release Already Exists
|
||||||
|
id: check_release
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
REPO_OWNER="${{ github.repository_owner }}"
|
||||||
|
REPO_NAME="${{ github.event.repository.name }}"
|
||||||
|
VERSION="${{ env.VERSION }}"
|
||||||
|
TAG="v$VERSION"
|
||||||
|
SERVER_URL="https://git.mahom03-spacecloud.de"
|
||||||
|
|
||||||
|
HTTP_STATUS=$(curl -s -o /dev/null -w "%{http_code}" -H "Authorization: token ${{ secrets.GITHUB_TOKEN }}" "$SERVER_URL/api/v1/repos/$REPO_OWNER/$REPO_NAME/releases/tags/$TAG")
|
||||||
|
|
||||||
|
if [ "$HTTP_STATUS" -eq 200 ]; then
|
||||||
|
echo "Release $TAG already exists. Skipping release-related steps."
|
||||||
|
echo "release_exists=true" >> $GITHUB_OUTPUT
|
||||||
|
elif [ "$HTTP_STATUS" -eq 404 ]; then
|
||||||
|
echo "No existing release for $TAG. Continuing."
|
||||||
|
echo "release_exists=false" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "Unexpected response when checking release: $HTTP_STATUS"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
- name: Build and Zip
|
- name: Build and Zip
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
# Inject version from manifest into the build
|
# Inject version from manifest into the build
|
||||||
@@ -71,6 +95,7 @@ jobs:
|
|||||||
echo "ZIP_PATH=bin/Publish/Jellyfin.Plugin.Seasonals.zip" >> $GITHUB_ENV
|
echo "ZIP_PATH=bin/Publish/Jellyfin.Plugin.Seasonals.zip" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Update manifest.json
|
- name: Update manifest.json
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
REPO_OWNER="${{ github.repository_owner }}"
|
REPO_OWNER="${{ github.repository_owner }}"
|
||||||
@@ -90,12 +115,14 @@ jobs:
|
|||||||
manifest.json > manifest.json.tmp && mv manifest.json.tmp manifest.json
|
manifest.json > manifest.json.tmp && mv manifest.json.tmp manifest.json
|
||||||
|
|
||||||
- name: Commit manifest.json
|
- name: Commit manifest.json
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
uses: stefanzweifel/git-auto-commit-action@v7
|
uses: stefanzweifel/git-auto-commit-action@v7
|
||||||
with:
|
with:
|
||||||
commit_message: "Update manifest.json for release v${{ env.VERSION }} [skip ci]"
|
commit_message: "Update manifest.json for release v${{ env.VERSION }} [skip ci]"
|
||||||
file_pattern: manifest.json
|
file_pattern: manifest.json
|
||||||
|
|
||||||
- name: Create Release
|
- name: Create Release
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
uses: akkuman/gitea-release-action@v1
|
uses: akkuman/gitea-release-action@v1
|
||||||
with:
|
with:
|
||||||
server_url: "https://git.mahom03-spacecloud.de"
|
server_url: "https://git.mahom03-spacecloud.de"
|
||||||
@@ -109,6 +136,7 @@ jobs:
|
|||||||
|
|
||||||
# Update Message in Remote Repository
|
# Update Message in Remote Repository
|
||||||
- name: Checkout Central Manifest Repo
|
- name: Checkout Central Manifest Repo
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
repository: ${{ github.repository_owner }}/jellyfin-plugin-manifest
|
repository: ${{ github.repository_owner }}/jellyfin-plugin-manifest
|
||||||
@@ -116,6 +144,7 @@ jobs:
|
|||||||
token: ${{ secrets.JELLYFIN_PLUGIN_MANIFEST_UPDATER_PAT }}
|
token: ${{ secrets.JELLYFIN_PLUGIN_MANIFEST_UPDATER_PAT }}
|
||||||
|
|
||||||
- name: Update Central Manifest
|
- name: Update Central Manifest
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
cd central-manifest
|
cd central-manifest
|
||||||
@@ -171,6 +200,7 @@ jobs:
|
|||||||
manifest.json > manifest.json.tmp && mv manifest.json.tmp manifest.json
|
manifest.json > manifest.json.tmp && mv manifest.json.tmp manifest.json
|
||||||
|
|
||||||
- name: Commit and Push Central Manifest
|
- name: Commit and Push Central Manifest
|
||||||
|
if: steps.check_release.outputs.release_exists == 'false'
|
||||||
run: |
|
run: |
|
||||||
cd central-manifest
|
cd central-manifest
|
||||||
git config user.name "CodeDevMLH"
|
git config user.name "CodeDevMLH"
|
||||||
|
|||||||
3
.gitignore
vendored
@@ -4,6 +4,5 @@ obj/
|
|||||||
.idea/
|
.idea/
|
||||||
artifacts
|
artifacts
|
||||||
|
|
||||||
test-site.html
|
test-site-old.html
|
||||||
test-site-new.html
|
|
||||||
RELEASE_GUIDE.md
|
RELEASE_GUIDE.md
|
||||||
343
CONTRIBUTING.md
Normal file
@@ -0,0 +1,343 @@
|
|||||||
|
# Contributing to Jellyfin Seasonals Plugin
|
||||||
|
|
||||||
|
Thank you for your interest in contributing seasonal themes to the Jellyfin Seasonals Plugin! This guide explains how seasonal themes are structured, how to create your own, and how to test them locally before submitting a pull request.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Table of Contents
|
||||||
|
|
||||||
|
- [Contributing to Jellyfin Seasonals Plugin](#contributing-to-jellyfin-seasonals-plugin)
|
||||||
|
- [Table of Contents](#table-of-contents)
|
||||||
|
- [Theme Architecture Overview](#theme-architecture-overview)
|
||||||
|
- [Standard Theme File Structure](#standard-theme-file-structure)
|
||||||
|
- [JavaScript File Pattern](#javascript-file-pattern)
|
||||||
|
- [Key Rules](#key-rules)
|
||||||
|
- [CSS File Pattern](#css-file-pattern)
|
||||||
|
- [Key Rules](#key-rules-1)
|
||||||
|
- [Image Assets (Optional)](#image-assets-optional)
|
||||||
|
- [Registering Your Theme](#registering-your-theme)
|
||||||
|
- [1. `seasonals.js` — Client-Side Registration](#1-seasonalsjs--client-side-registration)
|
||||||
|
- [2. `PluginConfiguration.cs` and `configPage.html` - Server-Side Registration](#2-pluginconfigurationcs-and-configpagehtml---server-side-registration)
|
||||||
|
- [Testing Your Theme Locally](#testing-your-theme-locally)
|
||||||
|
- [Steps](#steps)
|
||||||
|
- [What to Verify](#what-to-verify)
|
||||||
|
- [Submitting Your Contribution](#submitting-your-contribution)
|
||||||
|
- [Pull Request Checklist](#pull-request-checklist)
|
||||||
|
- [PR Description Template](#pr-description-template)
|
||||||
|
- [GitHub Issue Template for Theme Ideas](#github-issue-template-for-theme-ideas)
|
||||||
|
- [Questions?](#questions)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Theme Architecture Overview
|
||||||
|
|
||||||
|
Each seasonal theme consists of **2–3 components** that live in `Jellyfin.Plugin.Seasonals/Web/`:
|
||||||
|
|
||||||
|
| Component | File(s) | Purpose |
|
||||||
|
| :--- | :--- | :--- |
|
||||||
|
| **JavaScript** | `{themeName}.js` | Animation logic, DOM manipulation, element creation |
|
||||||
|
| **CSS** | `{themeName}.css` | Container styling, element appearance, keyframe animations |
|
||||||
|
| **Images** *(optional)* | `{themeName}_images/` | Image assets (PNGs, SVGs) used by the theme |
|
||||||
|
|
||||||
|
The orchestrator file `seasonals.js` manages theme loading at runtime. It reads the plugin configuration, determines which theme should be active, and dynamically injects the correct CSS and JS files.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Standard Theme File Structure
|
||||||
|
|
||||||
|
Here is a complete file layout for a theme called `mytheme`:
|
||||||
|
|
||||||
|
```
|
||||||
|
Jellyfin.Plugin.Seasonals/
|
||||||
|
└── Web/
|
||||||
|
├── mytheme.js # Animation/DOM logic
|
||||||
|
├── mytheme.css # Styles & animations
|
||||||
|
├── mytheme_images/ # (Optional) image assets
|
||||||
|
│ ├── sprite1.png
|
||||||
|
│ └── sprite2.png
|
||||||
|
└── seasonals.js # (Existing) Add your theme to ThemeConfigs
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## JavaScript File Pattern
|
||||||
|
|
||||||
|
Every theme JS file follows a **consistent skeleton**. Use this as your starting template:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// 1. Read Configuration
|
||||||
|
const config = window.SeasonalsPluginConfig?.MyTheme || {};
|
||||||
|
|
||||||
|
const enabled = config.EnableMyTheme !== undefined ? config.EnableMyTheme : true;
|
||||||
|
const elementCount = config.ElementCount || 25;
|
||||||
|
// ... add more config options as needed
|
||||||
|
|
||||||
|
let msgPrinted = false;
|
||||||
|
|
||||||
|
// 2. Toggle Function
|
||||||
|
// Hides the effect when a video player, trailer (in full width mode), dashboard, or user menu is active.
|
||||||
|
function toggleMyTheme() {
|
||||||
|
const container = document.querySelector('.mytheme-container');
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
const videoPlayer = document.querySelector('.videoPlayerContainer');
|
||||||
|
const trailerPlayer = document.querySelector('.youtubePlayerContainer');
|
||||||
|
const isDashboard = document.body.classList.contains('dashboardDocument');
|
||||||
|
const hasUserMenu = document.querySelector('#app-user-menu');
|
||||||
|
|
||||||
|
if (videoPlayer || trailerPlayer || isDashboard || hasUserMenu) {
|
||||||
|
container.style.display = 'none';
|
||||||
|
if (!msgPrinted) {
|
||||||
|
console.log('MyTheme hidden');
|
||||||
|
msgPrinted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
container.style.display = 'block';
|
||||||
|
if (msgPrinted) {
|
||||||
|
console.log('MyTheme visible');
|
||||||
|
msgPrinted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. MutationObserver
|
||||||
|
// Watches the DOM for changes so the effect can auto-hide/show.
|
||||||
|
const observer = new MutationObserver(toggleMyTheme);
|
||||||
|
observer.observe(document.body, {
|
||||||
|
childList: true,
|
||||||
|
subtree: true,
|
||||||
|
attributes: true
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. Element Creation
|
||||||
|
// Create and append your animated elements to the container.
|
||||||
|
function createElements() {
|
||||||
|
const container = document.querySelector('.mytheme-container') || document.createElement('div');
|
||||||
|
|
||||||
|
if (!document.querySelector('.mytheme-container')) {
|
||||||
|
container.className = 'mytheme-container';
|
||||||
|
container.setAttribute('aria-hidden', 'true');
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < elementCount; i++) {
|
||||||
|
const el = document.createElement('div');
|
||||||
|
el.className = 'mytheme-element';
|
||||||
|
|
||||||
|
// Set random position, delay, duration, etc.
|
||||||
|
el.style.left = `${Math.random() * 100}%`;
|
||||||
|
el.style.animationDelay = `${Math.random() * 10}s, ${Math.random() * 4}s`;
|
||||||
|
|
||||||
|
// If using images:
|
||||||
|
// const img = document.createElement('img');
|
||||||
|
// img.src = '../Seasonals/Resources/mytheme_images/sprite1.png';
|
||||||
|
// el.appendChild(img);
|
||||||
|
|
||||||
|
// If using text/emoji:
|
||||||
|
// el.textContent = '⭐';
|
||||||
|
|
||||||
|
container.appendChild(el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. Initialization
|
||||||
|
function initializeMyTheme() {
|
||||||
|
if (!enabled) return;
|
||||||
|
createElements();
|
||||||
|
toggleMyTheme();
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeMyTheme();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Key Rules
|
||||||
|
|
||||||
|
- **Always** read config from `window.SeasonalsPluginConfig?.{ThemeName}`.
|
||||||
|
- **Always** implement the toggle function with the same selectors (`.videoPlayerContainer`, `.youtubePlayerContainer`, `.dashboardDocument`, `#app-user-menu`, just use the above template).
|
||||||
|
- **Always** use `aria-hidden="true"` on the container for accessibility.
|
||||||
|
- Call your `initialize` function at the end of the file.
|
||||||
|
- For **canvas-based** themes (like `snowfall.js`), use a `<canvas>` element with `requestAnimationFrame` instead of CSS animations. Make sure to clean up with `cancelAnimationFrame` when hidden.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## CSS File Pattern
|
||||||
|
|
||||||
|
Every theme CSS file follows this structure:
|
||||||
|
|
||||||
|
```css
|
||||||
|
/* Container */
|
||||||
|
/* Full-screen overlay, transparent, non-interactive */
|
||||||
|
.mytheme-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none; /* IMPORTANT: don't block user interaction */
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Animated Element */
|
||||||
|
.mytheme-element {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 15;
|
||||||
|
user-select: none;
|
||||||
|
cursor: default;
|
||||||
|
|
||||||
|
/* Two animations: movement + secondary effect (shake, rotate, etc.) */
|
||||||
|
animation-name: mytheme-fall, mytheme-shake;
|
||||||
|
animation-duration: 10s, 3s;
|
||||||
|
animation-timing-function: linear, ease-in-out;
|
||||||
|
animation-iteration-count: infinite, infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Keyframes */
|
||||||
|
@keyframes mytheme-fall {
|
||||||
|
0% { top: -10%; }
|
||||||
|
100% { top: 100%; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes mytheme-shake {
|
||||||
|
0%, 100% { transform: translateX(0); }
|
||||||
|
50% { transform: translateX(80px); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Staggered Delays for Base Elements */
|
||||||
|
/* Spread the initial 12 elements across the screen */
|
||||||
|
.mytheme-element:nth-of-type(1) { left: 10%; animation-delay: 1s, 1s; }
|
||||||
|
.mytheme-element:nth-of-type(2) { left: 20%; animation-delay: 6s, 0.5s; }
|
||||||
|
.mytheme-element:nth-of-type(3) { left: 30%; animation-delay: 4s, 2s; }
|
||||||
|
/* ... continue for each base element */
|
||||||
|
```
|
||||||
|
|
||||||
|
### Key Rules
|
||||||
|
|
||||||
|
- **Container** must be `position: fixed`, full-screen, with `pointer-events: none` and at least `z-index: 10`.
|
||||||
|
- **Elements** should use `position: fixed` with at least `z-index: 15`.
|
||||||
|
- Use **animations** (eg. primary movement + secondary effect for natural-looking motion).
|
||||||
|
- Include **`nth-of-type` rules** for the initial set of base elements to stagger them.
|
||||||
|
- Include **webkit prefixes** (`-webkit-animation-*`, `@-webkit-keyframes`) for broader compatibility (see existing themes for examples).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Image Assets (Optional)
|
||||||
|
|
||||||
|
If your theme uses images (e.g., leaves, ghosts, eggs):
|
||||||
|
|
||||||
|
1. Create a folder: `Jellyfin.Plugin.Seasonals/Web/{themeName}_images/`
|
||||||
|
2. Place your assets inside (PNG recommended, keep files small)
|
||||||
|
3. Reference them in JS using the production path:
|
||||||
|
```javascript
|
||||||
|
img.src = '../Seasonals/Resources/mytheme_images/sprite1.png';
|
||||||
|
```
|
||||||
|
---
|
||||||
|
|
||||||
|
## Registering Your Theme
|
||||||
|
|
||||||
|
After creating your JS and CSS files, you need to register the theme in two places:
|
||||||
|
|
||||||
|
### 1. `seasonals.js` — Client-Side Registration
|
||||||
|
|
||||||
|
Add your theme to the `ThemeConfigs` object:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
const ThemeConfigs = {
|
||||||
|
// ... existing themes ...
|
||||||
|
mytheme: {
|
||||||
|
css: '../Seasonals/Resources/mytheme.css',
|
||||||
|
js: '../Seasonals/Resources/mytheme.js',
|
||||||
|
containerClass: 'mytheme-container'
|
||||||
|
},
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. `PluginConfiguration.cs` and `configPage.html` - Server-Side Registration
|
||||||
|
|
||||||
|
> [!NOTE]
|
||||||
|
> The backend registration is handled by the plugin maintainers. You do **not** need to modify C# files for your theme submission. Just focus on the JS/CSS/images.
|
||||||
|
>
|
||||||
|
> However, if you'd like to include full backend integration, add your theme to the enum/configuration in `Configuration/PluginConfiguration.cs` and the selectors in `configPage.html`.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Testing Your Theme Locally
|
||||||
|
|
||||||
|
You can test your theme without a Jellyfin server by using the included test site.
|
||||||
|
|
||||||
|
### Steps
|
||||||
|
|
||||||
|
1. Navigate to the `Jellyfin.Plugin.Seasonals/Web/` directory
|
||||||
|
2. Open `test-site.html` in your browser (just double-click the file) or vscode or what ever you use...
|
||||||
|
3. Use the **theme selector dropdown** to pick an existing theme or select **"Custom (Local Files)"** to test your own
|
||||||
|
4. When "Custom" is selected, enter your theme's JS and CSS filenames (e.g., `mytheme.js` and `mytheme.css` (must be in the same folder as `test-site.html` for this to work))
|
||||||
|
5. Click **"Load Theme"** to apply. Click **"Clear & Reload"** to reset and try again
|
||||||
|
|
||||||
|
### What to Verify
|
||||||
|
|
||||||
|
- ✅ The effect is visible on the background
|
||||||
|
- ✅ The animation runs smoothly
|
||||||
|
- ✅ Elements are spread across the full viewport
|
||||||
|
- ✅ The mock header is **not blocked** by the effect (thanks to `pointer-events: none`)
|
||||||
|
- ✅ No theme related console errors appear (check DevTools → Console)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Submitting Your Contribution
|
||||||
|
|
||||||
|
### Pull Request Checklist
|
||||||
|
|
||||||
|
- [ ] Created `{themeName}.js` following the [JS pattern](#javascript-file-pattern)
|
||||||
|
- [ ] Created `{themeName}.css` following the [CSS pattern](#css-file-pattern)
|
||||||
|
- [ ] (If applicable) Created `{themeName}_images/` with optimized assets
|
||||||
|
- [ ] Added theme to `ThemeConfigs` in `seasonals.js`
|
||||||
|
- [ ] Tested locally with `test-site.html`
|
||||||
|
- [ ] No theme related console errors
|
||||||
|
- [ ] Effect has `pointer-events: none` (doesn't block the UI)
|
||||||
|
- [ ] Effect hides during video/trailer playback (toggle function implemented)
|
||||||
|
- [ ] (Optional) Included a screenshot or short recording of the effect to the readme
|
||||||
|
|
||||||
|
### PR Description Template
|
||||||
|
|
||||||
|
```
|
||||||
|
## New Seasonal Theme: {Theme Name}
|
||||||
|
|
||||||
|
**Description:** Brief description of the theme and what occasion/season it's for.
|
||||||
|
|
||||||
|
**Screenshot / Recording:**
|
||||||
|
[Attach a screenshot or GIF showcasing the theme in action]
|
||||||
|
|
||||||
|
**Testing:**
|
||||||
|
- Tested locally with test-site-new.html ✅
|
||||||
|
- No console errors ✅
|
||||||
|
- pointer-events: none verified ✅
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## GitHub Issue Template for Theme Ideas
|
||||||
|
|
||||||
|
If you have an idea for a seasonal theme but don't want to implement it yourself, feel free to open an issue using the following template:
|
||||||
|
|
||||||
|
**Title:** `[Theme Idea] {Season/Holiday Name} Theme`
|
||||||
|
|
||||||
|
**Body:**
|
||||||
|
```
|
||||||
|
## 🎨 Theme Idea: {Season/Holiday Name}
|
||||||
|
|
||||||
|
**Occasion/Season:** What time of year is this for?
|
||||||
|
|
||||||
|
**Description:** Describe the visual effect you have in mind.
|
||||||
|
|
||||||
|
**Visual References:** Links to images, GIFs, or videos that capture the aesthetic.
|
||||||
|
|
||||||
|
**Suggested Active Period:** e.g. "March 1 – March 17" for St. Patrick's Day
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Questions?
|
||||||
|
|
||||||
|
If you have any questions about contributing, feel free to open an issue. Happy theming! 🎉
|
||||||
@@ -27,6 +27,7 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
Christmas = new ChristmasOptions();
|
Christmas = new ChristmasOptions();
|
||||||
Santa = new SantaOptions();
|
Santa = new SantaOptions();
|
||||||
Easter = new EasterOptions();
|
Easter = new EasterOptions();
|
||||||
|
Resurrection = new ResurrectionOptions();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -49,6 +50,11 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public bool EnableClientSideToggle { get; set; }
|
public bool EnableClientSideToggle { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the seasonal rules configuration as JSON.
|
||||||
|
/// </summary>
|
||||||
|
public string SeasonalRules { get; set; } = "[{\"Name\":\"New Year Fireworks\",\"StartDay\":28,\"StartMonth\":12,\"EndDay\":5,\"EndMonth\":1,\"Theme\":\"fireworks\"},{\"Name\":\"Valentine's Day\",\"StartDay\":10,\"StartMonth\":2,\"EndDay\":18,\"EndMonth\":2,\"Theme\":\"hearts\"},{\"Name\":\"Santa\",\"StartDay\":22,\"StartMonth\":12,\"EndDay\":27,\"EndMonth\":12,\"Theme\":\"santa\"},{\"Name\":\"Snowflakes (December)\",\"StartDay\":1,\"StartMonth\":12,\"EndDay\":31,\"EndMonth\":12,\"Theme\":\"snowflakes\"},{\"Name\":\"Snowfall (January)\",\"StartDay\":1,\"StartMonth\":1,\"EndDay\":31,\"EndMonth\":1,\"Theme\":\"snowfall\"},{\"Name\":\"Snowfall (February)\",\"StartDay\":1,\"StartMonth\":2,\"EndDay\":29,\"EndMonth\":2,\"Theme\":\"snowfall\"},{\"Name\":\"Easter\",\"StartDay\":25,\"StartMonth\":3,\"EndDay\":25,\"EndMonth\":4,\"Theme\":\"easter\"},{\"Name\":\"Halloween\",\"StartDay\":24,\"StartMonth\":10,\"EndDay\":5,\"EndMonth\":11,\"Theme\":\"halloween\"},{\"Name\":\"Autumn\",\"StartDay\":1,\"StartMonth\":9,\"EndDay\":30,\"EndMonth\":11,\"Theme\":\"autumn\"}]";
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the Seasonals options.
|
/// Gets or sets the Seasonals options.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -62,6 +68,7 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
public ChristmasOptions Christmas { get; set; }
|
public ChristmasOptions Christmas { get; set; }
|
||||||
public SantaOptions Santa { get; set; }
|
public SantaOptions Santa { get; set; }
|
||||||
public EasterOptions Easter { get; set; }
|
public EasterOptions Easter { get; set; }
|
||||||
|
public ResurrectionOptions Resurrection { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class AutumnOptions
|
public class AutumnOptions
|
||||||
@@ -166,3 +173,12 @@ public class EasterOptions
|
|||||||
public int MinBunnyRestTime { get; set; } = 2000;
|
public int MinBunnyRestTime { get; set; } = 2000;
|
||||||
public int MaxBunnyRestTime { get; set; } = 5000;
|
public int MaxBunnyRestTime { get; set; } = 5000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public class ResurrectionOptions
|
||||||
|
{
|
||||||
|
public int SymbolCount { get; set; } = 12;
|
||||||
|
public bool EnableResurrection { get; set; } = true;
|
||||||
|
public bool EnableRandomSymbols { get; set; } = true;
|
||||||
|
public bool EnableRandomSymbolsMobile { get; set; } = false;
|
||||||
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
@@ -6,41 +6,54 @@
|
|||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="SeasonalsConfigPage" data-role="page" class="page type-interior pluginConfigurationPage" data-require="emby-input,emby-button,emby-select,emby-checkbox">
|
<div id="SeasonalsConfigPage" data-role="page" class="page type-interior pluginConfigurationPage" data-require="emby-input,emby-button,emby-select,emby-checkbox">
|
||||||
<style>
|
|
||||||
select option:disabled {
|
|
||||||
color: #a3a3a3 !important;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
<div data-role="content">
|
<div data-role="content">
|
||||||
<div class="content-primary">
|
<div class="content-primary">
|
||||||
<div class="sectionTitleContainer">
|
<div class="sectionTitleContainer">
|
||||||
<h2 class="sectionTitle">Seasonals</h2>
|
<h2 class="sectionTitle">Seasonals Configuration</h2>
|
||||||
<a is="emby-linkbutton" class="raised raised-mini emby-button" style="margin-left: 2em;"
|
<a is="emby-linkbutton" class="raised raised-mini emby-button" style="margin-left: 2em;"
|
||||||
target="_blank" href="https://github.com/CodeDevMLH/Jellyfin-Seasonals">
|
target="_blank" href="https://github.com/CodeDevMLH/Jellyfin-Seasonals">
|
||||||
<i class="md-icon button-icon button-icon-left secondaryText"></i>
|
<i class="md-icon button-icon button-icon-left secondaryText"></i>
|
||||||
<span>Help</span>
|
<span>${Help}</span>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<hr style="max-width: 800px; margin: 1em 0;">
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
<br>
|
|
||||||
|
<div style="margin-bottom: 1.5em;">
|
||||||
|
<button class="seasonals-tab-button active" onclick="showSeasonalsTab('seasonals-basic', this)"
|
||||||
|
style="background: none; border: none; color: #fff; cursor: pointer; transition: color 0.3s, border-bottom 0.3s; padding: 0.5em 1em; border-bottom: 2px solid #00a4dc;">
|
||||||
|
<h3>General Settings</h3>
|
||||||
|
</button>
|
||||||
|
<button class="seasonals-tab-button" onclick="showSeasonalsTab('seasonals-auto-selection', this)"
|
||||||
|
style="background: none; border: none; color: #ccc; cursor: pointer; transition: color 0.3s, border-bottom 0.3s; padding: 0.5em 1em; border-bottom: 2px solid transparent;">
|
||||||
|
<h3>Auto Selection</h3>
|
||||||
|
</button>
|
||||||
|
<button class="seasonals-tab-button" onclick="showSeasonalsTab('seasonals-advanced', this)"
|
||||||
|
style="background: none; border: none; color: #ccc; cursor: pointer; transition: color 0.3s, border-bottom 0.3s; padding: 0.5em 1em; border-bottom: 2px solid transparent;">
|
||||||
|
<h3>Advanced Settings</h3>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<form id="SeasonalsConfigForm">
|
<form id="SeasonalsConfigForm">
|
||||||
|
<!-- BASIC Tab -->
|
||||||
|
<div id="seasonals-basic" class="seasonals-tab-content">
|
||||||
|
<h2 class="sectionTitle">Main Plugin Settings</h2>
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
<label class="emby-checkbox-label">
|
<label class="emby-checkbox-label">
|
||||||
<input id="IsEnabled" name="IsEnabled" type="checkbox" is="emby-checkbox" />
|
<input id="SeasonalsIsEnabled" name="SeasonalsIsEnabled" type="checkbox" is="emby-checkbox" />
|
||||||
<span>Enable Seasonals</span>
|
<span>Enable Seasonals</span>
|
||||||
</label>
|
</label>
|
||||||
<div class="fieldDescription">Enable or disable the entire plugin functionality.</div>
|
<div class="fieldDescription">Enable or disable the entire plugin functionality.</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
<label class="emby-checkbox-label">
|
<label class="emby-checkbox-label">
|
||||||
<input id="AutomateSeasonSelection" name="AutomateSeasonSelection" type="checkbox" is="emby-checkbox" />
|
<input id="SeasonalsAutomateSeasonSelection" name="SeasonalsAutomateSeasonSelection" type="checkbox" is="emby-checkbox" />
|
||||||
<span>Automate Season Selection</span>
|
<span>Automate Season Selection</span>
|
||||||
</label>
|
</label>
|
||||||
<div class="fieldDescription">Automatically select the season based on the date.</div>
|
<div class="fieldDescription">If enabled, the plugin will use the rules defined in the "Auto Selection" tab to assume the season. If no rule matches, it falls back to the "Standard Season" below.</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="selectContainer">
|
<div class="selectContainer">
|
||||||
<label class="selectLabel" for="SelectedSeason">Selected Season</label>
|
<label class="selectLabel" for="SeasonalsSelectedSeason">Standard Season</label>
|
||||||
<select id="SelectedSeason" name="SelectedSeason" class="emby-select" style="width: 100%; padding: 0.5em; background-color: #333; border: 1px solid #444; color: #fff; border-radius: 4px;">
|
<select is="emby-select" id="SeasonalsSelectedSeason" name="SeasonalsSelectedSeason" class="selectLayout emby-select-withcolor emby-select" style="width: 100%; -webkit-appearance: menulist; appearance: menulist;">
|
||||||
<option value="none">None</option>
|
<option value="none">None</option>
|
||||||
<option value="snowflakes">Snowflakes</option>
|
<option value="snowflakes">Snowflakes</option>
|
||||||
<option value="snowfall">Snowfall</option>
|
<option value="snowfall">Snowfall</option>
|
||||||
@@ -52,33 +65,52 @@
|
|||||||
<option value="santa">Santa</option>
|
<option value="santa">Santa</option>
|
||||||
<option value="autumn">Autumn</option>
|
<option value="autumn">Autumn</option>
|
||||||
<option value="easter">Easter</option>
|
<option value="easter">Easter</option>
|
||||||
<option value="summer" disabled>Summer (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="resurrection">Resurrection</option>
|
||||||
<option value="spring" disabled>Spring (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="summer" disabled>Summer (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="oktoberfest" disabled>Oktoberfest (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="spring" disabled>Spring (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="carnival" disabled>Carnival (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="oktoberfest" disabled>Oktoberfest (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="championships" disabled>European/World Championships (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="carnival" disabled>Carnival (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="patrick" disabled>St. Patrick's Day (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="championships" disabled>European/World Championships (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="thanksgiving" disabled>Thanksgiving (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="patrick" disabled>St. Patrick's Day (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="pride" disabled>Pride (not implemented yet, no idea for a theme. Please commit ideas in a issue)</option>
|
<option value="thanksgiving" disabled>Thanksgiving (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
|
<option value="pride" disabled>Pride (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
</select>
|
</select>
|
||||||
<div class="fieldDescription">The season to display if automation is disabled.</div>
|
<div class="fieldDescription">The season to display if automation is disabled or no "Auto Selection" rule matches the current date.</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
<label class="emby-checkbox-label">
|
<label class="emby-checkbox-label">
|
||||||
<input id="EnableClientSideToggle" name="EnableClientSideToggle" type="checkbox"
|
<input id="SeasonalsEnableClientSideToggle" name="SeasonalsEnableClientSideToggle" type="checkbox" is="emby-checkbox" />
|
||||||
is="emby-checkbox" />
|
|
||||||
<span>Allow Client-Side Toggle</span>
|
<span>Allow Client-Side Toggle</span>
|
||||||
</label>
|
</label>
|
||||||
<div class="fieldDescription">If enabled, users will see a settings icon in the header to toggle
|
<div class="fieldDescription">If enabled, users will see a seasonals icon in the header to toggle seasonals for their browser (device-specific).</div>
|
||||||
animations for their browser.</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<br>
|
|
||||||
|
|
||||||
<details>
|
<!-- Auto Selection Tab -->
|
||||||
<summary>Advanced Configuration</summary>
|
<div id="seasonals-auto-selection" class="seasonals-tab-content" style="display: none;">
|
||||||
<p>Configure specific settings for each seasonal theme below.</p>
|
<h2>Auto Selection Rules</h2>
|
||||||
|
|
||||||
|
<div style="background-color: rgba(255, 255, 255, 0.05); border-left: 4px solid #00a4dc; border-radius: 4px; padding: 1em 1.5em; margin: 1.5em 0; display: flex; align-items: center; gap: 1em;">
|
||||||
|
<i class="material-icons" style="color: #00a4dc; font-size: 24px;">info</i>
|
||||||
|
<div>Define rules to automatically select a season based on the date. Rules are evaluated from top to bottom. The first matching rule wins.</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="seasonalRulesList">
|
||||||
|
<!-- Rules will be injected here via JS -->
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button is="emby-button" type="button" class="raised emby-button" onclick="SeasonalsConfigPage.addRule();"
|
||||||
|
style="margin-top: 1em; display: inline-flex; align-items: center; gap: 0.4em;">
|
||||||
|
<i class="material-icons" style="font-size: 24px;">add</i>
|
||||||
|
<span>Add Rule</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Advanced Tab -->
|
||||||
|
<div id="seasonals-advanced" class="seasonals-tab-content" style="display: none;">
|
||||||
|
<h2 class="sectionTitle">Configure specific settings for each seasonal theme</h2>
|
||||||
|
<!-- <p>Configure specific settings for each seasonal theme below.</p> -->
|
||||||
<p>All symbol count settings add this number in addition to the standard 12 symbols (if additional symbols is enabled).</p>
|
<p>All symbol count settings add this number in addition to the standard 12 symbols (if additional symbols is enabled).</p>
|
||||||
|
|
||||||
<details>
|
<details>
|
||||||
<summary>Autumn</summary>
|
<summary>Autumn</summary>
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
@@ -506,7 +538,45 @@
|
|||||||
<div class="fieldDescription">Maximum time the bunny waits before appearing again.</div>
|
<div class="fieldDescription">Maximum time the bunny waits before appearing again.</div>
|
||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Resurrection</summary>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableResurrection" name="EnableResurrection" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Resurrection Seasonal</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enable the Resurrection theme in general (e.g. for automation).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomResurrection" name="EnableRandomResurrection" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Symbols</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional symbols randomly distributed across the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomResurrectionMobile" name="EnableRandomResurrectionMobile" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Symbols on Mobile</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional symbols randomly distributed across the screen on mobile devices. Warning: High values may affect performance.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="ResurrectionSymbolCount">Symbol Count</label>
|
||||||
|
<input is="emby-input" type="number" id="ResurrectionSymbolCount" name="ResurrectionSymbolCount" />
|
||||||
|
<div class="fieldDescription">Number of additional symbols (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableDifferentDurationResurrection" name="EnableDifferentDurationResurrection" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Different Duration</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Randomize the movement speed.</div>
|
||||||
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div style="background-color: rgba(255, 255, 255, 0.05); border-left: 4px solid #00a4dc; border-radius: 4px; padding: 1em 1.5em; margin: 1.5em 0; display: flex; align-items: center; gap: 1em;">
|
<div style="background-color: rgba(255, 255, 255, 0.05); border-left: 4px solid #00a4dc; border-radius: 4px; padding: 1em 1.5em; margin: 1.5em 0; display: flex; align-items: center; gap: 1em;">
|
||||||
<i class="material-icons" style="color: #00a4dc; font-size: 24px;">info</i>
|
<i class="material-icons" style="color: #00a4dc; font-size: 24px;">info</i>
|
||||||
@@ -527,19 +597,244 @@
|
|||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<script type="text/javascript">
|
<style>
|
||||||
var SeasonalsConfig = {
|
/* Styles for the Seasonal Rules List (Auto Selection Tab) */
|
||||||
pluginUniqueId: 'ef1e863f-cbb0-4e47-9f23-f0cbb1826ad4'
|
.seasonal-rule {
|
||||||
|
background: rgba(0, 0, 0, 0.2);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 15px;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 10px;
|
||||||
|
border: 1px solid rgba(255,255,255,0.05);
|
||||||
|
}
|
||||||
|
.seasonal-rule-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.seasonal-rule-content {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 2fr 1.3fr 1.3fr 2fr;
|
||||||
|
gap: 15px;
|
||||||
|
align-items: end;
|
||||||
|
}
|
||||||
|
.rule-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 5px;
|
||||||
|
}
|
||||||
|
.date-range-group {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
align-items: flex-end;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
.date-range-group > .inputContainer,
|
||||||
|
.date-range-group > .selectContainer {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 800px) {
|
||||||
|
.seasonal-rule-content {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Styles for the Tabs */
|
||||||
|
.seasonals-tab-button.active {
|
||||||
|
color: #fff !important;
|
||||||
|
border-bottom: 2px solid #00a4dc !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Disabled options in selects */
|
||||||
|
select option:disabled {
|
||||||
|
color: #a3a3a3 !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
function showSeasonalsTab(tabId, btn) {
|
||||||
|
document.querySelectorAll('.seasonals-tab-content').forEach(el => el.style.display = 'none');
|
||||||
|
document.getElementById(tabId).style.display = 'block';
|
||||||
|
|
||||||
|
document.querySelectorAll('.seasonals-tab-button').forEach(b => {
|
||||||
|
b.classList.remove('active');
|
||||||
|
b.style.color = '#ccc';
|
||||||
|
b.style.borderBottom = '2px solid transparent';
|
||||||
|
});
|
||||||
|
|
||||||
|
if (btn) {
|
||||||
|
btn.classList.add('active');
|
||||||
|
btn.style.color = '#fff';
|
||||||
|
btn.style.borderBottom = '2px solid #00a4dc';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var SeasonalsConfigPage = {
|
||||||
|
pluginUniqueId: 'ef1e863f-cbb0-4e47-9f23-f0cbb1826ad4',
|
||||||
|
|
||||||
|
addRule: function(data = null) {
|
||||||
|
var container = document.querySelector('#seasonalRulesList');
|
||||||
|
var div = document.createElement('div');
|
||||||
|
div.className = 'seasonal-rule';
|
||||||
|
|
||||||
|
var name = data ? (data.Name || data.name || 'New Rule') : 'New Rule';
|
||||||
|
var startDay = data ? (data.StartDay !== undefined ? data.StartDay : (data.startDay !== undefined ? data.startDay : 1)) : 1;
|
||||||
|
var startMonth = data ? (data.StartMonth !== undefined ? data.StartMonth : (data.startMonth !== undefined ? data.startMonth : 1)) : 1;
|
||||||
|
var endDay = data ? (data.EndDay !== undefined ? data.EndDay : (data.endDay !== undefined ? data.endDay : 1)) : 1;
|
||||||
|
var endMonth = data ? (data.EndMonth !== undefined ? data.EndMonth : (data.endMonth !== undefined ? data.endMonth : 1)) : 1;
|
||||||
|
var theme = data ? (data.Theme || data.theme || 'none') : 'none';
|
||||||
|
|
||||||
|
var days = [];
|
||||||
|
for (var i = 1; i <= 31; i++) days.push(i);
|
||||||
|
|
||||||
|
var months = [
|
||||||
|
{ v: 1, n: 'Jan' }, { v: 2, n: 'Feb' }, { v: 3, n: 'Mar' }, { v: 4, n: 'Apr' },
|
||||||
|
{ v: 5, n: 'May' }, { v: 6, n: 'Jun' }, { v: 7, n: 'Jul' }, { v: 8, n: 'Aug' },
|
||||||
|
{ v: 9, n: 'Sep' }, { v: 10, n: 'Oct' }, { v: 11, n: 'Nov' }, { v: 12, n: 'Dec' }
|
||||||
|
];
|
||||||
|
|
||||||
|
// Build select HTML via string concatenation to avoid Jellyfin's ${} localization processing
|
||||||
|
function mkSelect(val, opts, cls) {
|
||||||
|
var h = '<select class="emby-select emby-select-withcolor ' + cls + '" style="width: 100%; -webkit-appearance: menulist; appearance: menulist;">';
|
||||||
|
opts.forEach(function(o) {
|
||||||
|
var v = o.v || o;
|
||||||
|
var n = o.n || o;
|
||||||
|
h += '<option value="' + v + '" ' + (v == val ? 'selected' : '') + '>' + n + '</option>';
|
||||||
|
});
|
||||||
|
h += '</select>';
|
||||||
|
return h;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.innerHTML =
|
||||||
|
'<div class="seasonal-rule-header">' +
|
||||||
|
' <div style="font-weight: bold; font-size: 1.1em;" class="rule-title"></div>' +
|
||||||
|
' <div class="rule-actions">' +
|
||||||
|
' <button type="button" is="paper-icon-button-light" class="btn-move-up" title="Move Up"><i class="material-icons">arrow_upward</i></button>' +
|
||||||
|
' <button type="button" is="paper-icon-button-light" class="btn-move-down" title="Move Down"><i class="material-icons">arrow_downward</i></button>' +
|
||||||
|
' <button type="button" is="paper-icon-button-light" class="btn-remove" title="Remove"><i class="material-icons">delete</i></button>' +
|
||||||
|
' </div>' +
|
||||||
|
'</div>' +
|
||||||
|
'<div class="seasonal-rule-content">' +
|
||||||
|
' <div class="inputContainer" style="margin:0;">' +
|
||||||
|
' <label class="inputLabel">Name</label>' +
|
||||||
|
' <input is="emby-input" class="rule-name" onchange="SeasonalsConfigPage.updateRuleTitles();" />' +
|
||||||
|
' </div>' +
|
||||||
|
' <div class="date-range-group">' +
|
||||||
|
' <div class="selectContainer" style="margin:0; flex: 1;">' +
|
||||||
|
' <label class="selectLabel">Start Day</label>' +
|
||||||
|
mkSelect(startDay, days, 'rule-start-day') +
|
||||||
|
' </div>' +
|
||||||
|
' <div class="selectContainer" style="margin:0; flex: 1;">' +
|
||||||
|
' <label class="selectLabel">Month</label>' +
|
||||||
|
mkSelect(startMonth, months, 'rule-start-month') +
|
||||||
|
' </div>' +
|
||||||
|
' </div>' +
|
||||||
|
' <div class="date-range-group">' +
|
||||||
|
' <div class="selectContainer" style="margin:0; flex: 1;">' +
|
||||||
|
' <label class="selectLabel">End Day</label>' +
|
||||||
|
mkSelect(endDay, days, 'rule-end-day') +
|
||||||
|
' </div>' +
|
||||||
|
' <div class="selectContainer" style="margin:0; flex: 1;">' +
|
||||||
|
' <label class="selectLabel">Month</label>' +
|
||||||
|
mkSelect(endMonth, months, 'rule-end-month') +
|
||||||
|
' </div>' +
|
||||||
|
' </div>' +
|
||||||
|
' <div class="selectContainer" style="margin:0;">' +
|
||||||
|
' <label class="selectLabel">Theme</label>' +
|
||||||
|
' <select class="emby-select emby-select-withcolor rule-theme" style="width: 100%; -webkit-appearance: menulist; appearance: menulist;">' +
|
||||||
|
' <option value="none">None</option>' +
|
||||||
|
' <option value="snowflakes">Snowflakes</option>' +
|
||||||
|
' <option value="snowfall">Snowfall</option>' +
|
||||||
|
' <option value="snowstorm">Snowstorm</option>' +
|
||||||
|
' <option value="fireworks">Fireworks</option>' +
|
||||||
|
' <option value="halloween">Halloween</option>' +
|
||||||
|
' <option value="hearts">Hearts</option>' +
|
||||||
|
' <option value="christmas">Christmas</option>' +
|
||||||
|
' <option value="santa">Santa</option>' +
|
||||||
|
' <option value="autumn">Autumn</option>' +
|
||||||
|
' <option value="easter">Easter</option>' +
|
||||||
|
' <option value="resurrection">Resurrection</option>' +
|
||||||
|
' </select>' +
|
||||||
|
' </div>' +
|
||||||
|
'</div>';
|
||||||
|
|
||||||
|
container.appendChild(div);
|
||||||
|
|
||||||
|
// Set values programmatically
|
||||||
|
div.querySelector('.rule-name').value = name;
|
||||||
|
div.querySelector('.rule-theme').value = theme;
|
||||||
|
|
||||||
|
// Bind events
|
||||||
|
div.querySelector('.btn-remove').addEventListener('click', function() {
|
||||||
|
div.remove();
|
||||||
|
SeasonalsConfigPage.updateRuleTitles();
|
||||||
|
});
|
||||||
|
div.querySelector('.btn-move-up').addEventListener('click', function() {
|
||||||
|
if (div.previousElementSibling) {
|
||||||
|
container.insertBefore(div, div.previousElementSibling);
|
||||||
|
SeasonalsConfigPage.updateRuleTitles();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
div.querySelector('.btn-move-down').addEventListener('click', function() {
|
||||||
|
if (div.nextElementSibling) {
|
||||||
|
container.insertBefore(div.nextElementSibling, div);
|
||||||
|
SeasonalsConfigPage.updateRuleTitles();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this.updateRuleTitles();
|
||||||
|
},
|
||||||
|
|
||||||
|
updateRuleTitles: function() {
|
||||||
|
var rules = document.querySelectorAll('.seasonal-rule');
|
||||||
|
rules.forEach((rule, index) => {
|
||||||
|
var name = rule.querySelector('.rule-name').value;
|
||||||
|
rule.querySelector('.rule-title').innerText = '#' + (index + 1) + ' ' + name;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
renderRules: function(rules) {
|
||||||
|
var container = document.querySelector('#seasonalRulesList');
|
||||||
|
container.innerHTML = '';
|
||||||
|
if (Array.isArray(rules)) {
|
||||||
|
rules.forEach(rule => this.addRule(rule));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getRulesFromUI: function() {
|
||||||
|
var rules = [];
|
||||||
|
document.querySelectorAll('.seasonal-rule').forEach(div => {
|
||||||
|
rules.push({
|
||||||
|
Name: div.querySelector('.rule-name').value,
|
||||||
|
StartDay: parseInt(div.querySelector('.rule-start-day').value),
|
||||||
|
StartMonth: parseInt(div.querySelector('.rule-start-month').value),
|
||||||
|
EndDay: parseInt(div.querySelector('.rule-end-day').value),
|
||||||
|
EndMonth: parseInt(div.querySelector('.rule-end-month').value),
|
||||||
|
Theme: div.querySelector('.rule-theme').value
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return rules;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
document.querySelector('#SeasonalsConfigPage')
|
document.querySelector('#SeasonalsConfigPage')
|
||||||
.addEventListener('pageshow', function() {
|
.addEventListener('pageshow', function() {
|
||||||
Dashboard.showLoadingMsg();
|
Dashboard.showLoadingMsg();
|
||||||
ApiClient.getPluginConfiguration(SeasonalsConfig.pluginUniqueId).then(function (config) {
|
ApiClient.getPluginConfiguration(SeasonalsConfigPage.pluginUniqueId).then(function (config) {
|
||||||
document.querySelector('#IsEnabled').checked = config.IsEnabled;
|
document.querySelector('#SeasonalsIsEnabled').checked = config.IsEnabled;
|
||||||
document.querySelector('#SelectedSeason').value = config.SelectedSeason;
|
document.querySelector('#SeasonalsSelectedSeason').value = config.SelectedSeason;
|
||||||
document.querySelector('#AutomateSeasonSelection').checked = config.AutomateSeasonSelection;
|
document.querySelector('#SeasonalsAutomateSeasonSelection').checked = config.AutomateSeasonSelection;
|
||||||
document.querySelector('#EnableClientSideToggle').checked = config.EnableClientSideToggle !== undefined ? config.EnableClientSideToggle : true;
|
document.querySelector('#SeasonalsEnableClientSideToggle').checked = config.EnableClientSideToggle !== undefined ? config.EnableClientSideToggle : true;
|
||||||
|
|
||||||
|
// Load Rules
|
||||||
|
try {
|
||||||
|
var rules = JSON.parse(config.SeasonalRules || "[]");
|
||||||
|
SeasonalsConfigPage.renderRules(rules);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to parse SeasonalRules", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Advanced Config
|
// Advanced Config
|
||||||
// Autumn
|
// Autumn
|
||||||
@@ -625,6 +920,13 @@
|
|||||||
document.querySelector('#MinBunnyRestTime').value = config.Easter.MinBunnyRestTime;
|
document.querySelector('#MinBunnyRestTime').value = config.Easter.MinBunnyRestTime;
|
||||||
document.querySelector('#MaxBunnyRestTime').value = config.Easter.MaxBunnyRestTime;
|
document.querySelector('#MaxBunnyRestTime').value = config.Easter.MaxBunnyRestTime;
|
||||||
|
|
||||||
|
// Resurrection
|
||||||
|
document.querySelector('#EnableResurrection').checked = config.Resurrection.EnableResurrection;
|
||||||
|
document.querySelector('#ResurrectionSymbolCount').value = config.Resurrection.SymbolCount;
|
||||||
|
document.querySelector('#EnableRandomResurrection').checked = config.Resurrection.EnableRandomSymbols;
|
||||||
|
document.querySelector('#EnableRandomResurrectionMobile').checked = config.Resurrection.EnableRandomSymbolsMobile;
|
||||||
|
document.querySelector('#EnableDifferentDurationResurrection').checked = config.Resurrection.EnableDifferentDuration;
|
||||||
|
|
||||||
Dashboard.hideLoadingMsg();
|
Dashboard.hideLoadingMsg();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -632,12 +934,14 @@
|
|||||||
document.querySelector('#SeasonalsConfigForm')
|
document.querySelector('#SeasonalsConfigForm')
|
||||||
.addEventListener('submit', function(e) {
|
.addEventListener('submit', function(e) {
|
||||||
Dashboard.showLoadingMsg();
|
Dashboard.showLoadingMsg();
|
||||||
ApiClient.getPluginConfiguration(SeasonalsConfig.pluginUniqueId).then(function (config) {
|
ApiClient.getPluginConfiguration(SeasonalsConfigPage.pluginUniqueId).then(function (config) {
|
||||||
config.IsEnabled = document.querySelector('#IsEnabled').checked;
|
config.IsEnabled = document.querySelector('#SeasonalsIsEnabled').checked;
|
||||||
config.SelectedSeason = document.querySelector('#SelectedSeason').value;
|
config.SelectedSeason = document.querySelector('#SeasonalsSelectedSeason').value;
|
||||||
config.AutomateSeasonSelection = document.querySelector('#AutomateSeasonSelection').checked;
|
config.AutomateSeasonSelection = document.querySelector('#SeasonalsAutomateSeasonSelection').checked;
|
||||||
config.EnableClientSideToggle = document.querySelector('#EnableClientSideToggle').checked;
|
config.EnableClientSideToggle = document.querySelector('#SeasonalsEnableClientSideToggle').checked;
|
||||||
|
|
||||||
|
// Save Rules
|
||||||
|
config.SeasonalRules = JSON.stringify(SeasonalsConfigPage.getRulesFromUI());
|
||||||
|
|
||||||
// Advanced Config
|
// Advanced Config
|
||||||
// Autumn
|
// Autumn
|
||||||
@@ -723,7 +1027,14 @@
|
|||||||
config.Easter.MinBunnyRestTime = parseInt(document.querySelector('#MinBunnyRestTime').value);
|
config.Easter.MinBunnyRestTime = parseInt(document.querySelector('#MinBunnyRestTime').value);
|
||||||
config.Easter.MaxBunnyRestTime = parseInt(document.querySelector('#MaxBunnyRestTime').value);
|
config.Easter.MaxBunnyRestTime = parseInt(document.querySelector('#MaxBunnyRestTime').value);
|
||||||
|
|
||||||
ApiClient.updatePluginConfiguration(SeasonalsConfig.pluginUniqueId, config).then(function (result) {
|
// Resurrection
|
||||||
|
config.Resurrection.EnableResurrection = document.querySelector('#EnableResurrection').checked;
|
||||||
|
config.Resurrection.SymbolCount = parseInt(document.querySelector('#ResurrectionSymbolCount').value);
|
||||||
|
config.Resurrection.EnableRandomSymbols = document.querySelector('#EnableRandomResurrection').checked;
|
||||||
|
config.Resurrection.EnableRandomSymbolsMobile = document.querySelector('#EnableRandomResurrectionMobile').checked;
|
||||||
|
config.Resurrection.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationResurrection').checked;
|
||||||
|
|
||||||
|
ApiClient.updatePluginConfiguration(SeasonalsConfigPage.pluginUniqueId, config).then(function (result) {
|
||||||
Dashboard.processPluginConfigurationUpdateResult(result);
|
Dashboard.processPluginConfigurationUpdateResult(result);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
<!-- <TreatWarningsAsErrors>false</TreatWarningsAsErrors> -->
|
<!-- <TreatWarningsAsErrors>false</TreatWarningsAsErrors> -->
|
||||||
<Title>Jellyfin Seasonals Plugin</Title>
|
<Title>Jellyfin Seasonals Plugin</Title>
|
||||||
<Authors>CodeDevMLH</Authors>
|
<Authors>CodeDevMLH</Authors>
|
||||||
<Version>1.6.11.0</Version>
|
<Version>1.7.0.15</Version>
|
||||||
<RepositoryUrl>https://github.com/CodeDevMLH/Jellyfin-Seasonals</RepositoryUrl>
|
<RepositoryUrl>https://github.com/CodeDevMLH/Jellyfin-Seasonals</RepositoryUrl>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
@@ -26,7 +26,7 @@
|
|||||||
<None Remove="Configuration\configPage.html" />
|
<None Remove="Configuration\configPage.html" />
|
||||||
<EmbeddedResource Include="Configuration\configPage.html" />
|
<EmbeddedResource Include="Configuration\configPage.html" />
|
||||||
<None Remove="Web\**" />
|
<None Remove="Web\**" />
|
||||||
<EmbeddedResource Include="Web\**" />
|
<EmbeddedResource Include="Web\**" Exclude="Web\test-site.html" />
|
||||||
|
|
||||||
<None Include="..\README.md" />
|
<None Include="..\README.md" />
|
||||||
<None Include="..\logo.png" CopyToOutputDirectory="Always" />
|
<None Include="..\logo.png" CopyToOutputDirectory="Always" />
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ public class ScriptInjector
|
|||||||
{
|
{
|
||||||
private readonly IApplicationPaths _appPaths;
|
private readonly IApplicationPaths _appPaths;
|
||||||
private readonly ILogger<ScriptInjector> _logger;
|
private readonly ILogger<ScriptInjector> _logger;
|
||||||
public const string ScriptTag = "<script src=\"/Seasonals/Resources/seasonals.js\" defer></script>";
|
public const string ScriptTag = "<script src=\"../Seasonals/Resources/seasonals.js\" defer></script>";
|
||||||
public const string Marker = "</body>";
|
public const string Marker = "</body>";
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -48,23 +48,23 @@ observer.observe(document.body, {
|
|||||||
|
|
||||||
|
|
||||||
const images = [
|
const images = [
|
||||||
"/Seasonals/Resources/autumn_images/acorn1.png",
|
"../Seasonals/Resources/autumn_images/acorn1.png",
|
||||||
"/Seasonals/Resources/autumn_images/acorn2.png",
|
"../Seasonals/Resources/autumn_images/acorn2.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf1.png",
|
"../Seasonals/Resources/autumn_images/leaf1.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf2.png",
|
"../Seasonals/Resources/autumn_images/leaf2.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf3.png",
|
"../Seasonals/Resources/autumn_images/leaf3.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf4.png",
|
"../Seasonals/Resources/autumn_images/leaf4.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf5.png",
|
"../Seasonals/Resources/autumn_images/leaf5.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf6.png",
|
"../Seasonals/Resources/autumn_images/leaf6.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf7.png",
|
"../Seasonals/Resources/autumn_images/leaf7.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf8.png",
|
"../Seasonals/Resources/autumn_images/leaf8.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf9.png",
|
"../Seasonals/Resources/autumn_images/leaf9.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf10.png",
|
"../Seasonals/Resources/autumn_images/leaf10.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf11.png",
|
"../Seasonals/Resources/autumn_images/leaf11.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf12.png",
|
"../Seasonals/Resources/autumn_images/leaf12.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf13.png",
|
"../Seasonals/Resources/autumn_images/leaf13.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf14.png",
|
"../Seasonals/Resources/autumn_images/leaf14.png",
|
||||||
"/Seasonals/Resources/autumn_images/leaf15.png",
|
"../Seasonals/Resources/autumn_images/leaf15.png",
|
||||||
];
|
];
|
||||||
|
|
||||||
function addRandomLeaves(count) {
|
function addRandomLeaves(count) {
|
||||||
|
|||||||
@@ -61,20 +61,20 @@ observer.observe(document.body, {
|
|||||||
|
|
||||||
|
|
||||||
const images = [
|
const images = [
|
||||||
"/Seasonals/Resources/easter_images/egg_1.png",
|
"../Seasonals/Resources/easter_images/egg_1.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_2.png",
|
"../Seasonals/Resources/easter_images/egg_2.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_3.png",
|
"../Seasonals/Resources/easter_images/egg_3.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_4.png",
|
"../Seasonals/Resources/easter_images/egg_4.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_5.png",
|
"../Seasonals/Resources/easter_images/egg_5.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_6.png",
|
"../Seasonals/Resources/easter_images/egg_6.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_7.png",
|
"../Seasonals/Resources/easter_images/egg_7.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_8.png",
|
"../Seasonals/Resources/easter_images/egg_8.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_9.png",
|
"../Seasonals/Resources/easter_images/egg_9.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_10.png",
|
"../Seasonals/Resources/easter_images/egg_10.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_11.png",
|
"../Seasonals/Resources/easter_images/egg_11.png",
|
||||||
"/Seasonals/Resources/easter_images/egg_12.png",
|
"../Seasonals/Resources/easter_images/egg_12.png",
|
||||||
];
|
];
|
||||||
const rabbit = "/Seasonals/Resources/easter_images/easter-bunny.png";
|
const rabbit = "../Seasonals/Resources/easter_images/easter-bunny.png";
|
||||||
|
|
||||||
function addRandomEaster(count) {
|
function addRandomEaster(count) {
|
||||||
const easterContainer = document.querySelector('.easter-container'); // get the leave container
|
const easterContainer = document.querySelector('.easter-container'); // get the leave container
|
||||||
|
|||||||
@@ -46,9 +46,9 @@ observer.observe(document.body, {
|
|||||||
|
|
||||||
|
|
||||||
const images = [
|
const images = [
|
||||||
"/Seasonals/Resources/halloween_images/ghost_20x20.png",
|
"../Seasonals/Resources/halloween_images/ghost_20x20.png",
|
||||||
"/Seasonals/Resources/halloween_images/bat_20x20.png",
|
"../Seasonals/Resources/halloween_images/bat_20x20.png",
|
||||||
"/Seasonals/Resources/halloween_images/pumpkin_20x20.png",
|
"../Seasonals/Resources/halloween_images/pumpkin_20x20.png",
|
||||||
];
|
];
|
||||||
|
|
||||||
function addRandomSymbols(count) {
|
function addRandomSymbols(count) {
|
||||||
|
|||||||
59
Jellyfin.Plugin.Seasonals/Web/resurrection.css
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
.ressurection-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ressurection-symbol {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 15;
|
||||||
|
top: -15%;
|
||||||
|
user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
cursor: default;
|
||||||
|
animation-name: ressurection-fall, ressurection-sway;
|
||||||
|
animation-timing-function: linear, ease-in-out;
|
||||||
|
animation-iteration-count: infinite, infinite;
|
||||||
|
will-change: transform, top;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ressurection-symbol img {
|
||||||
|
z-index: 15;
|
||||||
|
height: auto;
|
||||||
|
width: 56px;
|
||||||
|
opacity: 0.95;
|
||||||
|
filter: drop-shadow(0 0 8px rgba(255, 215, 130, 0.5));
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.ressurection-symbol img {
|
||||||
|
width: 42px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes ressurection-fall {
|
||||||
|
0% {
|
||||||
|
top: -15%;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
top: 105%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes ressurection-sway {
|
||||||
|
0%,
|
||||||
|
100% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
transform: translateX(65px);
|
||||||
|
}
|
||||||
|
}
|
||||||
113
Jellyfin.Plugin.Seasonals/Web/resurrection.js
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
const config = window.SeasonalsPluginConfig?.Resurrection || {};
|
||||||
|
|
||||||
|
const enableResurrection = config.EnableResurrection !== undefined ? config.EnableResurrection : true;
|
||||||
|
const enableRandomSymbols = config.EnableRandomSymbols !== undefined ? config.EnableRandomSymbols : true;
|
||||||
|
const enableRandomSymbolsMobile = config.EnableRandomSymbolsMobile !== undefined ? config.EnableRandomSymbolsMobile : false;
|
||||||
|
const enableDifferentDuration = config.EnableDifferentDuration !== undefined ? config.EnableDifferentDuration : true;
|
||||||
|
const symbolCount = config.SymbolCount || 12;
|
||||||
|
|
||||||
|
let animationEnabled = true;
|
||||||
|
let statusLogged = false;
|
||||||
|
|
||||||
|
const images = [
|
||||||
|
'../Seasonals/Resources/resurrection_images/crosses.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/palm-branch.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/draped-cross.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/empty-tomb.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/he-is-risen.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/crown-of-thorns.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/risen-lord.png',
|
||||||
|
'../Seasonals/Resources/resurrection_images/dove.png'
|
||||||
|
];
|
||||||
|
|
||||||
|
function toggleResurrection() {
|
||||||
|
const container = document.querySelector('.resurrection-container');
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
const videoPlayer = document.querySelector('.videoPlayerContainer');
|
||||||
|
const trailerPlayer = document.querySelector('.youtubePlayerContainer');
|
||||||
|
const isDashboard = document.body.classList.contains('dashboardDocument');
|
||||||
|
const hasUserMenu = document.querySelector('#app-user-menu');
|
||||||
|
|
||||||
|
animationEnabled = !(videoPlayer || trailerPlayer || isDashboard || hasUserMenu);
|
||||||
|
container.style.display = animationEnabled ? 'block' : 'none';
|
||||||
|
|
||||||
|
if (!animationEnabled && !statusLogged) {
|
||||||
|
console.log('Resurrection hidden');
|
||||||
|
statusLogged = true;
|
||||||
|
} else if (animationEnabled && statusLogged) {
|
||||||
|
console.log('Resurrection visible');
|
||||||
|
statusLogged = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const observer = new MutationObserver(toggleResurrection);
|
||||||
|
observer.observe(document.body, {
|
||||||
|
childList: true,
|
||||||
|
subtree: true,
|
||||||
|
attributes: true
|
||||||
|
});
|
||||||
|
|
||||||
|
function createSymbol(imageSrc, leftPercent, delaySeconds) {
|
||||||
|
const symbol = document.createElement('div');
|
||||||
|
symbol.className = 'resurrection-symbol';
|
||||||
|
|
||||||
|
const img = document.createElement('img');
|
||||||
|
img.src = imageSrc;
|
||||||
|
img.alt = '';
|
||||||
|
|
||||||
|
symbol.style.left = `${leftPercent}%`;
|
||||||
|
symbol.style.animationDelay = `${delaySeconds}s, ${Math.random() * 3}s`;
|
||||||
|
|
||||||
|
if (enableDifferentDuration) {
|
||||||
|
const fallDuration = Math.random() * 7 + 7;
|
||||||
|
const swayDuration = Math.random() * 4 + 2;
|
||||||
|
symbol.style.animationDuration = `${fallDuration}s, ${swayDuration}s`;
|
||||||
|
}
|
||||||
|
|
||||||
|
symbol.appendChild(img);
|
||||||
|
return symbol;
|
||||||
|
}
|
||||||
|
|
||||||
|
function addSymbols(count) {
|
||||||
|
const container = document.querySelector('.resurrection-container');
|
||||||
|
if (!container || !enableRandomSymbols) return;
|
||||||
|
|
||||||
|
const isDesktop = window.innerWidth > 768;
|
||||||
|
if (!isDesktop && !enableRandomSymbolsMobile) return;
|
||||||
|
|
||||||
|
for (let i = 0; i < count; i++) {
|
||||||
|
const imageSrc = images[Math.floor(Math.random() * images.length)];
|
||||||
|
const left = Math.random() * 100;
|
||||||
|
const delay = Math.random() * 12;
|
||||||
|
container.appendChild(createSymbol(imageSrc, left, delay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initResurrection() {
|
||||||
|
let container = document.querySelector('.resurrection-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement('div');
|
||||||
|
container.className = 'resurrection-container';
|
||||||
|
container.setAttribute('aria-hidden', 'true');
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Place one of each of the 8 provided resurrection images first.
|
||||||
|
images.forEach((imageSrc, index) => {
|
||||||
|
const left = (index + 1) * (100 / (images.length + 1));
|
||||||
|
const delay = Math.random() * 8;
|
||||||
|
container.appendChild(createSymbol(imageSrc, left, delay));
|
||||||
|
});
|
||||||
|
|
||||||
|
const extraCount = Math.max(symbolCount - images.length, 0);
|
||||||
|
addSymbols(extraCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeResurrection() {
|
||||||
|
if (!enableResurrection) return;
|
||||||
|
initResurrection();
|
||||||
|
toggleResurrection();
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeResurrection();
|
||||||
BIN
Jellyfin.Plugin.Seasonals/Web/resurrection_images/crosses.png
Normal file
|
After Width: | Height: | Size: 412 KiB |
|
After Width: | Height: | Size: 317 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/resurrection_images/dove.png
Normal file
|
After Width: | Height: | Size: 354 KiB |
|
After Width: | Height: | Size: 382 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/resurrection_images/empty-tomb.png
Normal file
|
After Width: | Height: | Size: 472 KiB |
|
After Width: | Height: | Size: 372 KiB |
|
After Width: | Height: | Size: 384 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/resurrection_images/risen-lord.png
Normal file
|
After Width: | Height: | Size: 385 KiB |
@@ -181,18 +181,18 @@ function updateSnowflakes() {
|
|||||||
|
|
||||||
// credits: flaticon.com
|
// credits: flaticon.com
|
||||||
const presentImages = [
|
const presentImages = [
|
||||||
'/Seasonals/Resources/santa_images/gift1.png',
|
'../Seasonals/Resources/santa_images/gift1.png',
|
||||||
'/Seasonals/Resources/santa_images/gift2.png',
|
'../Seasonals/Resources/santa_images/gift2.png',
|
||||||
'/Seasonals/Resources/santa_images/gift3.png',
|
'../Seasonals/Resources/santa_images/gift3.png',
|
||||||
'/Seasonals/Resources/santa_images/gift4.png',
|
'../Seasonals/Resources/santa_images/gift4.png',
|
||||||
'/Seasonals/Resources/santa_images/gift5.png',
|
'../Seasonals/Resources/santa_images/gift5.png',
|
||||||
'/Seasonals/Resources/santa_images/gift6.png',
|
'../Seasonals/Resources/santa_images/gift6.png',
|
||||||
'/Seasonals/Resources/santa_images/gift7.png',
|
'../Seasonals/Resources/santa_images/gift7.png',
|
||||||
'/Seasonals/Resources/santa_images/gift8.png',
|
'../Seasonals/Resources/santa_images/gift8.png',
|
||||||
];
|
];
|
||||||
|
|
||||||
// credits: https://www.animatedimages.org/img-animated-santa-claus-image-0420-85884.htm
|
// credits: https://www.animatedimages.org/img-animated-santa-claus-image-0420-85884.htm
|
||||||
const santaImage = '/Seasonals/Resources/santa_images/santa.gif';
|
const santaImage = '../Seasonals/Resources/santa_images/santa.gif';
|
||||||
|
|
||||||
|
|
||||||
function createSantaElement() {
|
function createSantaElement() {
|
||||||
|
|||||||
@@ -1,65 +1,71 @@
|
|||||||
// theme-configs.js
|
/*
|
||||||
|
* Seasonals Plugin (Client Side Manager Logic)
|
||||||
|
*/
|
||||||
|
|
||||||
// theme configurations
|
const ThemeConfigs = {
|
||||||
const themeConfigs = {
|
|
||||||
snowflakes: {
|
snowflakes: {
|
||||||
css: '/Seasonals/Resources/snowflakes.css',
|
css: '../Seasonals/Resources/snowflakes.css',
|
||||||
js: '/Seasonals/Resources/snowflakes.js',
|
js: '../Seasonals/Resources/snowflakes.js',
|
||||||
containerClass: 'snowflakes'
|
containerClass: 'snowflakes'
|
||||||
},
|
},
|
||||||
snowfall: {
|
snowfall: {
|
||||||
css: '/Seasonals/Resources/snowfall.css',
|
css: '../Seasonals/Resources/snowfall.css',
|
||||||
js: '/Seasonals/Resources/snowfall.js',
|
js: '../Seasonals/Resources/snowfall.js',
|
||||||
containerClass: 'snowfall-container'
|
containerClass: 'snowfall-container'
|
||||||
},
|
},
|
||||||
snowstorm: {
|
snowstorm: {
|
||||||
css: '/Seasonals/Resources/snowstorm.css',
|
css: '../Seasonals/Resources/snowstorm.css',
|
||||||
js: '/Seasonals/Resources/snowstorm.js',
|
js: '../Seasonals/Resources/snowstorm.js',
|
||||||
containerClass: 'snowstorm-container'
|
containerClass: 'snowstorm-container'
|
||||||
},
|
},
|
||||||
fireworks: {
|
fireworks: {
|
||||||
css: '/Seasonals/Resources/fireworks.css',
|
css: '../Seasonals/Resources/fireworks.css',
|
||||||
js: '/Seasonals/Resources/fireworks.js',
|
js: '../Seasonals/Resources/fireworks.js',
|
||||||
containerClass: 'fireworks'
|
containerClass: 'fireworks'
|
||||||
},
|
},
|
||||||
halloween: {
|
halloween: {
|
||||||
css: '/Seasonals/Resources/halloween.css',
|
css: '../Seasonals/Resources/halloween.css',
|
||||||
js: '/Seasonals/Resources/halloween.js',
|
js: '../Seasonals/Resources/halloween.js',
|
||||||
containerClass: 'halloween-container'
|
containerClass: 'halloween-container'
|
||||||
},
|
},
|
||||||
hearts: {
|
hearts: {
|
||||||
css: '/Seasonals/Resources/hearts.css',
|
css: '../Seasonals/Resources/hearts.css',
|
||||||
js: '/Seasonals/Resources/hearts.js',
|
js: '../Seasonals/Resources/hearts.js',
|
||||||
containerClass: 'hearts-container'
|
containerClass: 'hearts-container'
|
||||||
},
|
},
|
||||||
christmas: {
|
christmas: {
|
||||||
css: '/Seasonals/Resources/christmas.css',
|
css: '../Seasonals/Resources/christmas.css',
|
||||||
js: '/Seasonals/Resources/christmas.js',
|
js: '../Seasonals/Resources/christmas.js',
|
||||||
containerClass: 'christmas-container'
|
containerClass: 'christmas-container'
|
||||||
},
|
},
|
||||||
santa: {
|
santa: {
|
||||||
css: '/Seasonals/Resources/santa.css',
|
css: '../Seasonals/Resources/santa.css',
|
||||||
js: '/Seasonals/Resources/santa.js',
|
js: '../Seasonals/Resources/santa.js',
|
||||||
containerClass: 'santa-container'
|
containerClass: 'santa-container'
|
||||||
},
|
},
|
||||||
autumn: {
|
autumn: {
|
||||||
css: '/Seasonals/Resources/autumn.css',
|
css: '../Seasonals/Resources/autumn.css',
|
||||||
js: '/Seasonals/Resources/autumn.js',
|
js: '../Seasonals/Resources/autumn.js',
|
||||||
containerClass: 'autumn-container'
|
containerClass: 'autumn-container'
|
||||||
},
|
},
|
||||||
easter: {
|
easter: {
|
||||||
css: '/Seasonals/Resources/easter.css',
|
css: '../Seasonals/Resources/easter.css',
|
||||||
js: '/Seasonals/Resources/easter.js',
|
js: '../Seasonals/Resources/easter.js',
|
||||||
containerClass: 'easter-container'
|
containerClass: 'easter-container'
|
||||||
},
|
},
|
||||||
|
resurrection: {
|
||||||
|
css: '../Seasonals/Resources/resurrection.css',
|
||||||
|
js: '../Seasonals/Resources/resurrection.js',
|
||||||
|
containerClass: 'resurrection-container'
|
||||||
|
},
|
||||||
summer: {
|
summer: {
|
||||||
css: '/Seasonals/Resources/summer.css',
|
css: '../Seasonals/Resources/summer.css',
|
||||||
js: '/Seasonals/Resources/summer.js',
|
js: '../Seasonals/Resources/summer.js',
|
||||||
containerClass: 'summer-container'
|
containerClass: 'summer-container'
|
||||||
},
|
},
|
||||||
spring: {
|
spring: {
|
||||||
css: '/Seasonals/Resources/spring.css',
|
css: '../Seasonals/Resources/spring.css',
|
||||||
js: '/Seasonals/Resources/spring.js',
|
js: '../Seasonals/Resources/spring.js',
|
||||||
containerClass: 'spring-container'
|
containerClass: 'spring-container'
|
||||||
},
|
},
|
||||||
none: {
|
none: {
|
||||||
@@ -67,195 +73,64 @@ const themeConfigs = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
// determine current theme based on the current month
|
const SeasonalSettingsManager = {
|
||||||
function determineCurrentTheme() {
|
initialized: false,
|
||||||
const date = new Date();
|
config: null,
|
||||||
const month = date.getMonth(); // 0-11
|
|
||||||
const day = date.getDate(); // 1-31
|
|
||||||
|
|
||||||
if ((month === 11 && day >= 28) || (month === 0 && day <= 5)) return 'fireworks'; //new year fireworks december 28 - january 5
|
init(config) {
|
||||||
|
if (this.initialized) return;
|
||||||
|
this.config = config;
|
||||||
|
|
||||||
if (month === 1 && day >= 10 && day <= 18) return 'hearts'; // valentine's day february 10 - 18
|
// Only inject settings if enabled on server by admin
|
||||||
|
if (this.config && this.config.EnableClientSideToggle !== false) {
|
||||||
if (month === 11 && day >= 22 && day <= 27) return 'santa'; // christmas december 22 - 27
|
this.injectSettingsIcon();
|
||||||
// if (month === 11 && day >= 22 && day <= 27) return 'christmas'; // christmas december 22 - 27
|
this.initialized = true;
|
||||||
|
console.log("Seasonals: Client-Side Settings Manager initialized.");
|
||||||
if (month === 11) return 'snowflakes'; // snowflakes december
|
|
||||||
if (month === 0 || month === 1) return 'snowfall'; // snow january, february
|
|
||||||
// if (month === 0 || month === 1) return 'snowstorm'; // snow january, february
|
|
||||||
|
|
||||||
if ((month === 2 && day >= 25) || (month === 3 && day <= 25)) return 'easter'; // easter march 25 - april 25
|
|
||||||
|
|
||||||
//NOT IMPLEMENTED YET
|
|
||||||
//if (month >= 2 && month <= 4) return 'spring'; // spring march, april, may
|
|
||||||
|
|
||||||
//NOT IMPLEMENTED YET
|
|
||||||
//if (month >= 5 && month <= 7) return 'summer'; // summer june, july, august
|
|
||||||
|
|
||||||
if ((month === 9 && day >= 24) || (month === 10 && day <= 5)) return 'halloween'; // halloween october 24 - november 5
|
|
||||||
|
|
||||||
if (month >= 8 && month <= 10) return 'autumn'; // autumn september, october, november
|
|
||||||
|
|
||||||
return 'none'; // Fallback (nothing)
|
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
// helper to resolve paths for local testing vs production
|
getSetting(key, defaultValue) {
|
||||||
function resolvePath(path) {
|
const value = localStorage.getItem(`seasonals-${key}`);
|
||||||
if (window.location.protocol === 'file:' || window.location.hostname === 'localhost' || window.location.hostname === '127.0.0.1') {
|
|
||||||
return path.replace('/Seasonals/Resources/', './');
|
|
||||||
}
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
|
|
||||||
// load theme csss
|
|
||||||
function loadThemeCSS(cssPath) {
|
|
||||||
if (!cssPath) return;
|
|
||||||
|
|
||||||
const link = document.createElement('link');
|
|
||||||
link.rel = 'stylesheet';
|
|
||||||
// link.href = resolvePath(cssPath);
|
|
||||||
link.href = cssPath;
|
|
||||||
|
|
||||||
link.onerror = () => {
|
|
||||||
console.error(`Failed to load CSS: ${cssPath}`);
|
|
||||||
};
|
|
||||||
|
|
||||||
document.body.appendChild(link);
|
|
||||||
console.log(`CSS file "${cssPath}" loaded.`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// load theme js
|
|
||||||
function loadThemeJS(jsPath) {
|
|
||||||
if (!jsPath) return;
|
|
||||||
|
|
||||||
const script = document.createElement('script');
|
|
||||||
script.src = jsPath;
|
|
||||||
// script.src = resolvePath(jsPath);
|
|
||||||
script.defer = true;
|
|
||||||
|
|
||||||
script.onerror = () => {
|
|
||||||
console.error(`Failed to load JS: ${jsPath}`);
|
|
||||||
};
|
|
||||||
|
|
||||||
document.body.appendChild(script);
|
|
||||||
console.log(`JS file "${jsPath}" loaded.`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// update theme container class name
|
|
||||||
function updateThemeContainer(containerClass) {
|
|
||||||
// Create container if it doesn't exist
|
|
||||||
let container = document.querySelector('.seasonals-container');
|
|
||||||
if (!container) {
|
|
||||||
container = document.createElement('div');
|
|
||||||
container.className = 'seasonals-container';
|
|
||||||
document.body.appendChild(container);
|
|
||||||
}
|
|
||||||
|
|
||||||
container.className = `seasonals-container ${containerClass}`;
|
|
||||||
console.log(`Seasonals-Container class updated to "${containerClass}".`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// function removeSelf() {
|
|
||||||
// const script = document.currentScript;
|
|
||||||
// if (script) script.parentNode.removeChild(script);
|
|
||||||
// console.log('External script removed:', script);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// initialize theme
|
|
||||||
async function initializeTheme() {
|
|
||||||
|
|
||||||
// Check local user preference
|
|
||||||
const isEnabled = getSavedSetting('seasonals-enabled', 'true') === 'true';
|
|
||||||
if (!isEnabled) {
|
|
||||||
console.log('Seasonals disabled by user preference.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const forcedTheme = getSavedSetting('seasonals-theme', 'auto');
|
|
||||||
|
|
||||||
let automateThemeSelection = true;
|
|
||||||
let defaultTheme = 'none';
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch('/Seasonals/Config');
|
|
||||||
if (response.ok) {
|
|
||||||
const config = await response.json();
|
|
||||||
automateThemeSelection = config.AutomateSeasonSelection;
|
|
||||||
defaultTheme = config.SelectedSeason;
|
|
||||||
window.SeasonalsPluginConfig = config;
|
|
||||||
console.log('Seasonals Config loaded:', config);
|
|
||||||
} else {
|
|
||||||
console.error('Failed to fetch Seasonals config');
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error fetching Seasonals config:', error);
|
|
||||||
}
|
|
||||||
|
|
||||||
let currentTheme;
|
|
||||||
|
|
||||||
if (forcedTheme !== 'auto') {
|
|
||||||
currentTheme = forcedTheme;
|
|
||||||
console.log(`User forced theme: ${currentTheme}`);
|
|
||||||
} else if (automateThemeSelection === false) {
|
|
||||||
currentTheme = defaultTheme;
|
|
||||||
} else {
|
|
||||||
currentTheme = determineCurrentTheme();
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Selected theme: ${currentTheme}`);
|
|
||||||
|
|
||||||
if (!currentTheme || currentTheme === 'none') {
|
|
||||||
console.log('No theme selected.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const theme = themeConfigs[currentTheme];
|
|
||||||
|
|
||||||
if (!theme) {
|
|
||||||
console.error(`Theme "${currentTheme}" not found.`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
updateThemeContainer(theme.containerClass);
|
|
||||||
|
|
||||||
if (theme.css) loadThemeCSS(theme.css);
|
|
||||||
if (theme.js) loadThemeJS(theme.js);
|
|
||||||
|
|
||||||
console.log(`Theme "${currentTheme}" loaded.`);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
initializeTheme();
|
|
||||||
|
|
||||||
|
|
||||||
// User UI Seasonal Settings
|
|
||||||
|
|
||||||
function getSavedSetting(key, defaultValue) {
|
|
||||||
const value = localStorage.getItem(key);
|
|
||||||
return value !== null ? value : defaultValue;
|
return value !== null ? value : defaultValue;
|
||||||
}
|
},
|
||||||
|
|
||||||
function setSavedSetting(key, value) {
|
setSetting(key, value) {
|
||||||
localStorage.setItem(key, value);
|
localStorage.setItem(`seasonals-${key}`, value);
|
||||||
}
|
},
|
||||||
|
|
||||||
function createSettingsIcon() {
|
createIcon() {
|
||||||
const button = document.createElement('button');
|
const button = document.createElement('button');
|
||||||
button.type = 'button';
|
button.type = 'button';
|
||||||
button.className = 'paper-icon-button-light headerButton seasonal-settings-button';
|
button.className = 'paper-icon-button-light headerButton seasonal-settings-button';
|
||||||
button.title = 'Seasonal Settings';
|
button.title = 'Seasonal Settings';
|
||||||
// button.innerHTML = '<span class="material-icons">ac_unit</span>';
|
// button.innerHTML = '<span class="material-icons">ac_unit</span>';
|
||||||
button.innerHTML = '<img src="/Seasonals/Resources/assets/logo_SW.svg" style="width: 24px; height: 24px; vertical-align: middle;">';
|
button.innerHTML = '<img src="../Seasonals/Resources/assets/logo_SW.svg" draggable="false" style="width: 24px; height: 24px; vertical-align: middle; pointer-events: none;">';
|
||||||
button.style.verticalAlign = 'middle';
|
button.style.verticalAlign = 'middle';
|
||||||
|
|
||||||
button.addEventListener('click', (e) => {
|
button.addEventListener('click', (e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
toggleSettingsPopup(button);
|
this.toggleSettingsPopup(button);
|
||||||
});
|
});
|
||||||
|
|
||||||
return button;
|
return button;
|
||||||
}
|
},
|
||||||
|
|
||||||
function createSettingsPopup(anchorElement) {
|
injectSettingsIcon() {
|
||||||
|
const observer = new MutationObserver((mutations, obs) => {
|
||||||
|
const headerRight = document.querySelector('.headerRight');
|
||||||
|
if (headerRight && !document.querySelector('.seasonal-settings-button')) {
|
||||||
|
const icon = this.createIcon();
|
||||||
|
headerRight.prepend(icon);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(document.body, {
|
||||||
|
childList: true,
|
||||||
|
subtree: true
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
createPopup(anchorElement) {
|
||||||
const existing = document.querySelector('.seasonal-settings-popup');
|
const existing = document.querySelector('.seasonal-settings-popup');
|
||||||
if (existing) existing.remove();
|
if (existing) existing.remove();
|
||||||
|
|
||||||
@@ -276,6 +151,7 @@ function createSettingsPopup(anchorElement) {
|
|||||||
|
|
||||||
const rect = anchorElement.getBoundingClientRect();
|
const rect = anchorElement.getBoundingClientRect();
|
||||||
|
|
||||||
|
// Positioning logic
|
||||||
let rightPos = window.innerWidth - rect.right;
|
let rightPos = window.innerWidth - rect.right;
|
||||||
if (window.innerWidth < 450 || (window.innerWidth - rightPos) < 260) {
|
if (window.innerWidth < 450 || (window.innerWidth - rightPos) < 260) {
|
||||||
popup.style.right = '1rem';
|
popup.style.right = '1rem';
|
||||||
@@ -284,10 +160,12 @@ function createSettingsPopup(anchorElement) {
|
|||||||
popup.style.right = `${rightPos}px`;
|
popup.style.right = `${rightPos}px`;
|
||||||
popup.style.left = 'auto';
|
popup.style.left = 'auto';
|
||||||
}
|
}
|
||||||
|
|
||||||
popup.style.top = `${rect.bottom + 10}px`;
|
popup.style.top = `${rect.bottom + 10}px`;
|
||||||
|
|
||||||
popup.innerHTML = `
|
// Popup HTML
|
||||||
|
let html = `
|
||||||
|
<h3 style="margin-top:0; margin-bottom:1em; border-bottom:1px solid #444; padding-bottom:0.5em;">Seasonal Settings</h3>
|
||||||
|
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription" style="margin-bottom: 0.5em;">
|
<div class="checkboxContainer checkboxContainer-withDescription" style="margin-bottom: 0.5em;">
|
||||||
<label class="emby-checkbox-label">
|
<label class="emby-checkbox-label">
|
||||||
<input id="seasonal-enable-toggle" type="checkbox" is="emby-checkbox" class="emby-checkbox" />
|
<input id="seasonal-enable-toggle" type="checkbox" is="emby-checkbox" class="emby-checkbox" />
|
||||||
@@ -298,38 +176,40 @@ function createSettingsPopup(anchorElement) {
|
|||||||
<div class="selectContainer" style="margin-bottom: 0.5em;">
|
<div class="selectContainer" style="margin-bottom: 0.5em;">
|
||||||
<label class="selectLabel" for="seasonal-theme-select" style="margin-bottom: 0.5em; display: block; color: inherit;">Force Theme</label>
|
<label class="selectLabel" for="seasonal-theme-select" style="margin-bottom: 0.5em; display: block; color: inherit;">Force Theme</label>
|
||||||
<select id="seasonal-theme-select" class="emby-select" style="width: 100%; padding: 0.5em; background-color: #333; border: 1px solid #444; color: #fff; border-radius: 4px;">
|
<select id="seasonal-theme-select" class="emby-select" style="width: 100%; padding: 0.5em; background-color: #333; border: 1px solid #444; color: #fff; border-radius: 4px;">
|
||||||
<option value="auto">Auto (Date Based)</option>
|
<option value="auto">Server-Side</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
popup.innerHTML = html;
|
||||||
|
|
||||||
// Populate Select Options
|
// Populate Select Options
|
||||||
const themeSelect = popup.querySelector('#seasonal-theme-select');
|
const themeSelect = popup.querySelector('#seasonal-theme-select');
|
||||||
Object.keys(themeConfigs).forEach(key => {
|
Object.keys(ThemeConfigs).forEach(key => {
|
||||||
if (key === 'none') return;
|
if (key === 'none') return;
|
||||||
const option = document.createElement('option');
|
const option = document.createElement('option');
|
||||||
option.value = key;
|
option.value = key;
|
||||||
// Capitalize first letter
|
|
||||||
option.textContent = key.charAt(0).toUpperCase() + key.slice(1);
|
option.textContent = key.charAt(0).toUpperCase() + key.slice(1);
|
||||||
themeSelect.appendChild(option);
|
themeSelect.appendChild(option);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Set Initial Values
|
// Set Initial Values
|
||||||
const enabledCheckbox = popup.querySelector('#seasonal-enable-toggle');
|
const enabledCheckbox = popup.querySelector('#seasonal-enable-toggle');
|
||||||
enabledCheckbox.checked = getSavedSetting('seasonals-enabled', 'true') === 'true';
|
enabledCheckbox.checked = this.getSetting('enabled', 'true') === 'true';
|
||||||
|
themeSelect.value = this.getSetting('theme', 'auto');
|
||||||
themeSelect.value = getSavedSetting('seasonals-theme', 'auto');
|
|
||||||
|
|
||||||
|
// Event Listeners
|
||||||
enabledCheckbox.addEventListener('change', (e) => {
|
enabledCheckbox.addEventListener('change', (e) => {
|
||||||
setSavedSetting('seasonals-enabled', e.target.checked);
|
this.setSetting('enabled', e.target.checked);
|
||||||
location.reload();
|
location.reload();
|
||||||
});
|
});
|
||||||
|
|
||||||
themeSelect.addEventListener('change', (e) => {
|
themeSelect.addEventListener('change', (e) => {
|
||||||
setSavedSetting('seasonals-theme', e.target.value);
|
this.setSetting('theme', e.target.value);
|
||||||
location.reload();
|
location.reload();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Close on outside click
|
||||||
const closeHandler = (e) => {
|
const closeHandler = (e) => {
|
||||||
if (!popup.contains(e.target) && e.target !== anchorElement && !anchorElement.contains(e.target)) {
|
if (!popup.contains(e.target) && e.target !== anchorElement && !anchorElement.contains(e.target)) {
|
||||||
popup.remove();
|
popup.remove();
|
||||||
@@ -339,36 +219,179 @@ function createSettingsPopup(anchorElement) {
|
|||||||
setTimeout(() => document.addEventListener('click', closeHandler), 0);
|
setTimeout(() => document.addEventListener('click', closeHandler), 0);
|
||||||
|
|
||||||
document.body.appendChild(popup);
|
document.body.appendChild(popup);
|
||||||
}
|
},
|
||||||
|
|
||||||
function toggleSettingsPopup(anchorElement) {
|
toggleSettingsPopup(anchorElement) {
|
||||||
const existing = document.querySelector('.seasonal-settings-popup');
|
const existing = document.querySelector('.seasonal-settings-popup');
|
||||||
if (existing) {
|
if (existing) {
|
||||||
existing.remove();
|
existing.remove();
|
||||||
} else {
|
} else {
|
||||||
createSettingsPopup(anchorElement);
|
this.createPopup(anchorElement);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const SeasonalsManager = {
|
||||||
|
config: null,
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
// Fetch Config
|
||||||
|
try {
|
||||||
|
const response = await fetch('../Seasonals/Config');
|
||||||
|
if (response.ok) {
|
||||||
|
this.config = await response.json();
|
||||||
|
window.SeasonalsPluginConfig = this.config;
|
||||||
|
console.log('Seasonals: Seasonals Config loaded:', this.config);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Seasonals: Error fetching Seasonals config:', error);
|
||||||
|
}
|
||||||
|
|
||||||
function injectSettingsIcon() {
|
// Initialize Settings UI
|
||||||
const observer = new MutationObserver((mutations, obs) => {
|
SeasonalSettingsManager.init(this.config);
|
||||||
// Check if admin has enabled this feature
|
|
||||||
if (window.SeasonalsPluginConfig && window.SeasonalsPluginConfig.EnableClientSideToggle === false) {
|
// User Preference Check
|
||||||
|
const isEnabled = SeasonalSettingsManager.getSetting('enabled', 'true') === 'true';
|
||||||
|
if (!isEnabled) {
|
||||||
|
console.log('Seasonals: Disabled by user preference.');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const headerRight = document.querySelector('.headerRight');
|
// Determine Theme
|
||||||
if (headerRight && !document.querySelector('.seasonal-settings-button')) {
|
const themeName = this.selectTheme();
|
||||||
const icon = createSettingsIcon();
|
console.log(`Seasonals: Selected theme: ${themeName}`);
|
||||||
headerRight.prepend(icon);
|
|
||||||
// obs.disconnect();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
observer.observe(document.body, {
|
if (!themeName || themeName === 'none') {
|
||||||
childList: true,
|
return;
|
||||||
subtree: true
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
injectSettingsIcon();
|
// Apply Theme
|
||||||
|
this.applyTheme(themeName);
|
||||||
|
},
|
||||||
|
|
||||||
|
selectTheme() {
|
||||||
|
// Check local override
|
||||||
|
const forcedTheme = SeasonalSettingsManager.getSetting('theme', 'auto');
|
||||||
|
if (forcedTheme !== 'auto') {
|
||||||
|
console.log(`Seasonals: User forced theme: ${forcedTheme}`);
|
||||||
|
return forcedTheme;
|
||||||
|
}
|
||||||
|
|
||||||
|
const automate = this.config ? this.config.AutomateSeasonSelection : true;
|
||||||
|
const defaultTheme = this.config ? this.config.SelectedSeason : 'none';
|
||||||
|
|
||||||
|
if (!automate) {
|
||||||
|
return defaultTheme;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.determineCurrentThemeDate();
|
||||||
|
},
|
||||||
|
|
||||||
|
determineCurrentThemeDate() {
|
||||||
|
var rules = [];
|
||||||
|
try {
|
||||||
|
rules = JSON.parse(this.config.SeasonalRules || "[]");
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Seasonals: Error parsing SeasonalRules", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rules.length === 0) {
|
||||||
|
// Fallback to empty/none if no rules are defined (though default should exist)
|
||||||
|
console.log("Seasonals: No auto-selection rules found.");
|
||||||
|
return 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
const date = new Date();
|
||||||
|
const month = date.getMonth() + 1; // 1-12
|
||||||
|
const day = date.getDate(); // 1-31
|
||||||
|
|
||||||
|
for (var i = 0; i < rules.length; i++) {
|
||||||
|
var rule = rules[i];
|
||||||
|
if (this.isDateInRange(day, month, rule.StartDay, rule.StartMonth, rule.EndDay, rule.EndMonth)) {
|
||||||
|
console.log(`Seasonals: Match found for rule "${rule.Name}" (${rule.Theme})`);
|
||||||
|
return rule.Theme;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'none'; // No rule matched
|
||||||
|
},
|
||||||
|
|
||||||
|
isDateInRange: function(day, month, startDay, startMonth, endDay, endMonth) {
|
||||||
|
if (startMonth > endMonth) {
|
||||||
|
// Wrapping year (e.g. Dec to Jan)
|
||||||
|
return this.isDateAfterOrEqual(day, month, startDay, startMonth) ||
|
||||||
|
this.isDateBeforeOrEqual(day, month, endDay, endMonth);
|
||||||
|
} else {
|
||||||
|
// Normal range
|
||||||
|
return this.isDateAfterOrEqual(day, month, startDay, startMonth) &&
|
||||||
|
this.isDateBeforeOrEqual(day, month, endDay, endMonth);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
isDateAfterOrEqual: function(day, month, targetDay, targetMonth) {
|
||||||
|
if (month > targetMonth) return true;
|
||||||
|
if (month === targetMonth && day >= targetDay) return true;
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
isDateBeforeOrEqual: function(day, month, targetDay, targetMonth) {
|
||||||
|
if (month < targetMonth) return true;
|
||||||
|
if (month === targetMonth && day <= targetDay) return true;
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
applyTheme(themeName) {
|
||||||
|
const theme = ThemeConfigs[themeName];
|
||||||
|
if (!theme) {
|
||||||
|
console.error(`Seasonals: Theme "${themeName}" not found.`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updateThemeContainer(theme.containerClass);
|
||||||
|
|
||||||
|
if (theme.css) this.loadResource('css', theme.css);
|
||||||
|
if (theme.js) this.loadResource('js', theme.js);
|
||||||
|
|
||||||
|
console.log(`Seasonals: Theme "${themeName}" applied.`);
|
||||||
|
},
|
||||||
|
|
||||||
|
updateThemeContainer(containerClass) {
|
||||||
|
let container = document.querySelector('.seasonals-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement('div');
|
||||||
|
container.className = 'seasonals-container';
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
container.className = `seasonals-container ${containerClass}`;
|
||||||
|
},
|
||||||
|
|
||||||
|
// helper to resolve paths for local testing vs production
|
||||||
|
resolvePath(path) {
|
||||||
|
if (window.location.protocol === 'file:' || window.location.hostname === 'localhost' || window.location.hostname === '127.0.0.1') {
|
||||||
|
return path.replace('/Seasonals/Resources/', './');
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
},
|
||||||
|
|
||||||
|
loadResource(type, path) {
|
||||||
|
if (!path) return;
|
||||||
|
|
||||||
|
if (type === 'css') {
|
||||||
|
const link = document.createElement('link');
|
||||||
|
link.rel = 'stylesheet';
|
||||||
|
link.href = path;
|
||||||
|
// link.href = resolvePath(cssPath);
|
||||||
|
link.onerror = () => console.error(`Seasonals: Failed to load CSS: ${path}`);
|
||||||
|
document.body.appendChild(link);
|
||||||
|
} else if (type === 'js') {
|
||||||
|
const script = document.createElement('script');
|
||||||
|
script.src = path;
|
||||||
|
// script.src = resolvePath(jsPath);
|
||||||
|
script.defer = true;
|
||||||
|
script.onerror = () => console.error(`Seasonals: Failed to load JS: ${path}`);
|
||||||
|
document.body.appendChild(script);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
SeasonalsManager.init();
|
||||||
|
|||||||
@@ -1,79 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
||||||
<title>Seasonals Test Display</title>
|
|
||||||
<link rel="stylesheet" href="snowfall.css">
|
|
||||||
<style>
|
|
||||||
body {
|
|
||||||
background-color: black;
|
|
||||||
color: white;
|
|
||||||
margin: 0;
|
|
||||||
font-family: sans-serif;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Mock Jellyfin Header */
|
|
||||||
.skinHeader {
|
|
||||||
background-color: #101010;
|
|
||||||
height: 60px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 0 1em;
|
|
||||||
border-bottom: 1px solid #333;
|
|
||||||
}
|
|
||||||
|
|
||||||
.headerRight {
|
|
||||||
margin-left: auto;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.paper-icon-button-light {
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
color: white;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.material-icons {
|
|
||||||
font-family: 'Material Icons';
|
|
||||||
/* Placeholder if not loaded */
|
|
||||||
font-weight: normal;
|
|
||||||
font-style: normal;
|
|
||||||
font-size: 24px;
|
|
||||||
display: inline-block;
|
|
||||||
line-height: 1;
|
|
||||||
text-transform: none;
|
|
||||||
letter-spacing: normal;
|
|
||||||
word-wrap: normal;
|
|
||||||
white-space: nowrap;
|
|
||||||
direction: ltr;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
<!-- Load Material Icons for the snowflake -->
|
|
||||||
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
|
||||||
<!-- Mock Header Structure -->
|
|
||||||
<div class="skinHeader">
|
|
||||||
<div class="skinHeader-content">
|
|
||||||
<span>Jellyfin Mock Header</span>
|
|
||||||
</div>
|
|
||||||
<div class="headerRight">
|
|
||||||
<button class="paper-icon-button-light">
|
|
||||||
<span class="material-icons">person</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="seasonals-container"></div>
|
|
||||||
|
|
||||||
<!-- Load the main script -->
|
|
||||||
<script src="seasonals.js"></script>
|
|
||||||
</body>
|
|
||||||
|
|
||||||
</html>
|
|
||||||
498
Jellyfin.Plugin.Seasonals/Web/test-site.html
Normal file
@@ -0,0 +1,498 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>Seasonals Theme Tester</title>
|
||||||
|
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600&display=swap" rel="stylesheet">
|
||||||
|
<style>
|
||||||
|
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: #101010;
|
||||||
|
color: #e0e0e0;
|
||||||
|
font-family: 'Inter', sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ── Mock Jellyfin Header ── */
|
||||||
|
.skinHeader {
|
||||||
|
background-color: #181818;
|
||||||
|
height: 60px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 1.5em;
|
||||||
|
border-bottom: 1px solid #282828;
|
||||||
|
position: relative;
|
||||||
|
z-index: 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skinHeader-content { font-size: 1.1em; font-weight: 500; }
|
||||||
|
|
||||||
|
.headerRight {
|
||||||
|
margin-left: auto;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.paper-icon-button-light {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: #ccc;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: 50%;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.paper-icon-button-light:hover { background: rgba(255,255,255,0.1); }
|
||||||
|
|
||||||
|
.material-icons {
|
||||||
|
font-family: 'Material Icons';
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
font-size: 24px;
|
||||||
|
display: inline-block;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ── Control Panel ── */
|
||||||
|
.control-panel {
|
||||||
|
position: fixed;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
background: rgba(20, 20, 20, 0.95);
|
||||||
|
backdrop-filter: blur(12px);
|
||||||
|
border-top: 1px solid #333;
|
||||||
|
padding: 1em 1.5em;
|
||||||
|
z-index: 200;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1em;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel label {
|
||||||
|
font-size: 0.85em;
|
||||||
|
color: #aaa;
|
||||||
|
font-weight: 500;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel select,
|
||||||
|
.control-panel input[type="text"] {
|
||||||
|
background: #2a2a2a;
|
||||||
|
color: #e0e0e0;
|
||||||
|
border: 1px solid #444;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 0.5em 0.75em;
|
||||||
|
font-size: 0.9em;
|
||||||
|
font-family: inherit;
|
||||||
|
outline: none;
|
||||||
|
transition: border-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel select:focus,
|
||||||
|
.control-panel input[type="text"]:focus {
|
||||||
|
border-color: #00a4dc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel select { min-width: 160px; }
|
||||||
|
.control-panel input[type="text"] { width: 160px; }
|
||||||
|
|
||||||
|
.control-panel button {
|
||||||
|
background: #00a4dc;
|
||||||
|
color: #fff;
|
||||||
|
border: none;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 0.5em 1.2em;
|
||||||
|
font-size: 0.9em;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background 0.2s;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel button:hover { background: #008bbd; }
|
||||||
|
|
||||||
|
.control-panel button.btn-secondary {
|
||||||
|
background: #333;
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel button.btn-secondary:hover { background: #444; }
|
||||||
|
|
||||||
|
.custom-fields {
|
||||||
|
display: none;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.custom-fields.visible { display: flex; }
|
||||||
|
|
||||||
|
/* ── Mock Content ── */
|
||||||
|
.mock-content {
|
||||||
|
padding: 2em 1.5em 6em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mock-content h2 {
|
||||||
|
font-size: 1.4em;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mock-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));
|
||||||
|
gap: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mock-card {
|
||||||
|
background: #1a1a1a;
|
||||||
|
border-radius: 8px;
|
||||||
|
aspect-ratio: 2/3;
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mock-card::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 40%;
|
||||||
|
background: linear-gradient(transparent, rgba(0,0,0,0.7));
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-bar {
|
||||||
|
position: fixed;
|
||||||
|
bottom: 65px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
background: rgba(0, 164, 220, 0.15);
|
||||||
|
border-top: 1px solid rgba(0, 164, 220, 0.3);
|
||||||
|
padding: 0.5em 1.5em;
|
||||||
|
z-index: 199;
|
||||||
|
font-size: 0.8em;
|
||||||
|
color: #aaa;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-bar a {
|
||||||
|
color: #00a4dc;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<!-- Mock Header -->
|
||||||
|
<div class="skinHeader">
|
||||||
|
<div class="skinHeader-content">
|
||||||
|
<span>Jellyfin</span>
|
||||||
|
</div>
|
||||||
|
<div class="headerRight">
|
||||||
|
<button class="paper-icon-button-light" title="Search">
|
||||||
|
<span class="material-icons">search</span>
|
||||||
|
</button>
|
||||||
|
<button class="paper-icon-button-light" title="User">
|
||||||
|
<span class="material-icons">person</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Seasonals Container (themes inject here) -->
|
||||||
|
<div class="seasonals-container"></div>
|
||||||
|
|
||||||
|
<!-- Mock Library Content -->
|
||||||
|
<div class="mock-content">
|
||||||
|
<h2>My Media</h2>
|
||||||
|
<div class="mock-grid">
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
<div class="mock-card"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Info Bar -->
|
||||||
|
<div class="info-bar">
|
||||||
|
📖 See <a href="../../CONTRIBUTING.md">CONTRIBUTING.md</a> for how to create your own seasonal theme
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Control Panel -->
|
||||||
|
<div class="control-panel">
|
||||||
|
<label for="theme-select">Theme:</label>
|
||||||
|
<select id="theme-select">
|
||||||
|
<option value="" selected disabled>— Select a theme —</option>
|
||||||
|
<option value="snowfall">Snowfall</option>
|
||||||
|
<option value="snowflakes">Snowflakes</option>
|
||||||
|
<option value="snowstorm">Snowstorm</option>
|
||||||
|
<option value="fireworks">Fireworks</option>
|
||||||
|
<option value="halloween">Halloween</option>
|
||||||
|
<option value="hearts">Hearts</option>
|
||||||
|
<option value="christmas">Christmas</option>
|
||||||
|
<option value="santa">Santa</option>
|
||||||
|
<option value="autumn">Autumn</option>
|
||||||
|
<option value="easter">Easter</option>
|
||||||
|
<option value="resurrection">Resurrection</option>
|
||||||
|
<option value="custom">⚙ Custom (Local Files)</option>
|
||||||
|
</select>
|
||||||
|
|
||||||
|
<div id="custom-fields" class="custom-fields">
|
||||||
|
<p>Javascript:</p>
|
||||||
|
<input type="text" id="custom-js" placeholder="mytheme.js">
|
||||||
|
<p>CSS:</p>
|
||||||
|
<input type="text" id="custom-css" placeholder="mytheme.css">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button id="btn-load" onclick="loadTheme()">Load Theme</button>
|
||||||
|
<button id="btn-clear" class="btn-secondary" onclick="clearTheme()">Clear & Reload</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// ── Path Rewriter ──────────────────────────────────────────
|
||||||
|
// Theme JS files reference images using the production path:
|
||||||
|
// ../Seasonals/Resources/santa_images/gift1.png
|
||||||
|
// When testing locally, the images live next to this HTML file:
|
||||||
|
// ./santa_images/gift1.png
|
||||||
|
// This observer intercepts all new <img> elements and rewrites
|
||||||
|
// their src attribute so image-based themes (santa, halloween,
|
||||||
|
// autumn, easter, resurrection) work out of the box.
|
||||||
|
const PRODUCTION_PREFIX = '../Seasonals/Resources/';
|
||||||
|
const LOCAL_PREFIX = './';
|
||||||
|
|
||||||
|
function rewritePath(src) {
|
||||||
|
if (!src) return src;
|
||||||
|
// Handle both full URLs and relative paths
|
||||||
|
const idx = src.indexOf('Seasonals/Resources/');
|
||||||
|
if (idx !== -1) {
|
||||||
|
return LOCAL_PREFIX + src.substring(idx + 'Seasonals/Resources/'.length);
|
||||||
|
}
|
||||||
|
return src;
|
||||||
|
}
|
||||||
|
|
||||||
|
function rewriteElement(el) {
|
||||||
|
if (el.tagName === 'IMG' && el.src && el.src.includes('Seasonals/Resources/')) {
|
||||||
|
const newSrc = rewritePath(el.src);
|
||||||
|
console.log(`[Path Rewriter] ${el.src} → ${newSrc}`);
|
||||||
|
el.src = newSrc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Watch for dynamically added images and rewrite their paths
|
||||||
|
const pathRewriter = new MutationObserver((mutations) => {
|
||||||
|
for (const mutation of mutations) {
|
||||||
|
for (const node of mutation.addedNodes) {
|
||||||
|
if (node.nodeType !== 1) continue; // skip non-elements
|
||||||
|
rewriteElement(node);
|
||||||
|
// Also check children (e.g. a div with img inside)
|
||||||
|
if (node.querySelectorAll) {
|
||||||
|
node.querySelectorAll('img').forEach(rewriteElement);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Also catch src attribute changes on existing elements
|
||||||
|
if (mutation.type === 'attributes' && mutation.attributeName === 'src') {
|
||||||
|
rewriteElement(mutation.target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
pathRewriter.observe(document.body, {
|
||||||
|
childList: true,
|
||||||
|
subtree: true,
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ['src']
|
||||||
|
});
|
||||||
|
|
||||||
|
// ── Built-in theme map (local file paths for testing) ──
|
||||||
|
const themes = {
|
||||||
|
snowfall: { css: 'snowfall.css', js: 'snowfall.js', container: 'snowfall-container' },
|
||||||
|
snowflakes: { css: 'snowflakes.css', js: 'snowflakes.js', container: 'snowflakes' },
|
||||||
|
snowstorm: { css: 'snowstorm.css', js: 'snowstorm.js', container: 'snowstorm-container' },
|
||||||
|
fireworks: { css: 'fireworks.css', js: 'fireworks.js', container: 'fireworks' },
|
||||||
|
halloween: { css: 'halloween.css', js: 'halloween.js', container: 'halloween-container' },
|
||||||
|
hearts: { css: 'hearts.css', js: 'hearts.js', container: 'hearts-container' },
|
||||||
|
christmas: { css: 'christmas.css', js: 'christmas.js', container: 'christmas-container' },
|
||||||
|
santa: { css: 'santa.css', js: 'santa.js', container: 'santa-container' },
|
||||||
|
autumn: { css: 'autumn.css', js: 'autumn.js', container: 'autumn-container' },
|
||||||
|
easter: { css: 'easter.css', js: 'easter.js', container: 'easter-container' },
|
||||||
|
resurrection: { css: 'resurrection.css', js: 'resurrection.js', container: 'resurrection-container' },
|
||||||
|
};
|
||||||
|
|
||||||
|
const select = document.getElementById('theme-select');
|
||||||
|
const customFields = document.getElementById('custom-fields');
|
||||||
|
|
||||||
|
select.addEventListener('change', () => {
|
||||||
|
customFields.classList.toggle('visible', select.value === 'custom');
|
||||||
|
});
|
||||||
|
|
||||||
|
function clearTheme() {
|
||||||
|
// Remove injected CSS
|
||||||
|
document.querySelectorAll('link[data-seasonal]').forEach(el => el.remove());
|
||||||
|
// Remove injected JS
|
||||||
|
document.querySelectorAll('script[data-seasonal]').forEach(el => el.remove());
|
||||||
|
|
||||||
|
// Reset the seasonals container
|
||||||
|
const container = document.querySelector('.seasonals-container');
|
||||||
|
container.className = 'seasonals-container';
|
||||||
|
container.innerHTML = '';
|
||||||
|
|
||||||
|
// Remove any theme-created containers on body
|
||||||
|
const knownContainers = [
|
||||||
|
'.snowfall-container', '.snowflakes', '.snowstorm-container',
|
||||||
|
'.fireworks', '.halloween-container', '.hearts-container',
|
||||||
|
'.christmas-container', '.santa-container', '.autumn-container',
|
||||||
|
'.easter-container', '.resurrection-container'
|
||||||
|
];
|
||||||
|
knownContainers.forEach(sel => {
|
||||||
|
document.querySelectorAll(sel).forEach(el => {
|
||||||
|
if (!el.classList.contains('seasonals-container')) el.remove();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Remove any canvas elements left over
|
||||||
|
document.querySelectorAll('#snowfallCanvas, #snowstormCanvas').forEach(el => el.remove());
|
||||||
|
|
||||||
|
// Remove rabbit element
|
||||||
|
document.querySelectorAll('#rabbit, .hopping-rabbit').forEach(el => el.remove());
|
||||||
|
|
||||||
|
// Remove santa element
|
||||||
|
document.querySelectorAll('.santa, .present').forEach(el => el.remove());
|
||||||
|
|
||||||
|
console.log('[Test Site] Theme cleared.');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Track active animation frames and observers for cleanup
|
||||||
|
let activeAnimationFrames = [];
|
||||||
|
let activeBlobUrls = [];
|
||||||
|
|
||||||
|
// Patch requestAnimationFrame and MutationObserver to track them
|
||||||
|
const origRAF = window.requestAnimationFrame;
|
||||||
|
const origCAF = window.cancelAnimationFrame;
|
||||||
|
let trackingEnabled = false;
|
||||||
|
|
||||||
|
window.requestAnimationFrame = function(cb) {
|
||||||
|
const id = origRAF.call(window, cb);
|
||||||
|
if (trackingEnabled) activeAnimationFrames.push(id);
|
||||||
|
return id;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.cancelAnimationFrame = function(id) {
|
||||||
|
origCAF.call(window, id);
|
||||||
|
activeAnimationFrames = activeAnimationFrames.filter(f => f !== id);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Track MutationObservers created by themes
|
||||||
|
let activeObservers = [];
|
||||||
|
const OrigMO = window.MutationObserver;
|
||||||
|
window.MutationObserver = class extends OrigMO {
|
||||||
|
constructor(cb) {
|
||||||
|
super(cb);
|
||||||
|
if (trackingEnabled) activeObservers.push(this);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Track intervals created by themes
|
||||||
|
let activeIntervals = [];
|
||||||
|
const origSetInterval = window.setInterval;
|
||||||
|
const origClearInterval = window.clearInterval;
|
||||||
|
window.setInterval = function(...args) {
|
||||||
|
const id = origSetInterval.apply(window, args);
|
||||||
|
if (trackingEnabled) activeIntervals.push(id);
|
||||||
|
return id;
|
||||||
|
};
|
||||||
|
window.clearInterval = function(id) {
|
||||||
|
origClearInterval.call(window, id);
|
||||||
|
activeIntervals = activeIntervals.filter(i => i !== id);
|
||||||
|
};
|
||||||
|
|
||||||
|
function loadTheme() {
|
||||||
|
clearTheme();
|
||||||
|
|
||||||
|
// Cancel all tracked animation frames
|
||||||
|
activeAnimationFrames.forEach(id => origCAF.call(window, id));
|
||||||
|
activeAnimationFrames = [];
|
||||||
|
|
||||||
|
// Disconnect all tracked MutationObservers
|
||||||
|
activeObservers.forEach(obs => obs.disconnect());
|
||||||
|
activeObservers = [];
|
||||||
|
|
||||||
|
// Clear all tracked intervals
|
||||||
|
activeIntervals.forEach(id => origClearInterval.call(window, id));
|
||||||
|
activeIntervals = [];
|
||||||
|
|
||||||
|
// Revoke old blob URLs
|
||||||
|
activeBlobUrls.forEach(url => URL.revokeObjectURL(url));
|
||||||
|
activeBlobUrls = [];
|
||||||
|
|
||||||
|
const value = select.value;
|
||||||
|
if (!value || value === '') return;
|
||||||
|
|
||||||
|
let cssFile, jsFile, containerClass;
|
||||||
|
|
||||||
|
if (value === 'custom') {
|
||||||
|
cssFile = document.getElementById('custom-css').value.trim();
|
||||||
|
jsFile = document.getElementById('custom-js').value.trim();
|
||||||
|
containerClass = cssFile ? cssFile.replace('.css', '-container') : 'custom-container';
|
||||||
|
} else {
|
||||||
|
const theme = themes[value];
|
||||||
|
cssFile = theme.css;
|
||||||
|
jsFile = theme.js;
|
||||||
|
containerClass = theme.container;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the seasonals-container class
|
||||||
|
const container = document.querySelector('.seasonals-container');
|
||||||
|
container.className = `seasonals-container ${containerClass}`;
|
||||||
|
|
||||||
|
// Inject CSS
|
||||||
|
if (cssFile) {
|
||||||
|
const link = document.createElement('link');
|
||||||
|
link.rel = 'stylesheet';
|
||||||
|
link.href = cssFile;
|
||||||
|
link.setAttribute('data-seasonal', 'true');
|
||||||
|
link.onerror = () => console.error(`[Test Site] Failed to load CSS: ${cssFile}`);
|
||||||
|
document.head.appendChild(link);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Inject JS wrapped in IIFE to avoid const redeclaration errors
|
||||||
|
if (jsFile) {
|
||||||
|
setTimeout(async () => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(jsFile);
|
||||||
|
if (!response.ok) throw new Error(`HTTP ${response.status}`);
|
||||||
|
const code = await response.text();
|
||||||
|
|
||||||
|
// Wrap in IIFE so each theme has its own scope
|
||||||
|
const wrappedCode = `(function() {\n${code}\n})();`;
|
||||||
|
const blob = new Blob([wrappedCode], { type: 'application/javascript' });
|
||||||
|
const blobUrl = URL.createObjectURL(blob);
|
||||||
|
activeBlobUrls.push(blobUrl);
|
||||||
|
|
||||||
|
trackingEnabled = true;
|
||||||
|
const script = document.createElement('script');
|
||||||
|
script.src = blobUrl;
|
||||||
|
script.setAttribute('data-seasonal', 'true');
|
||||||
|
script.onerror = () => console.error(`[Test Site] Failed to execute JS: ${jsFile}`);
|
||||||
|
document.body.appendChild(script);
|
||||||
|
console.log(`[Test Site] Loaded theme: ${value} (${jsFile}) [IIFE-wrapped]`);
|
||||||
|
} catch (err) {
|
||||||
|
console.error(`[Test Site] Failed to load JS: ${jsFile}`, err);
|
||||||
|
}
|
||||||
|
}, 150);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
189
README.md
@@ -13,20 +13,17 @@ This plugin is based on my manual mod (see the [legacy branch](https://github.co
|
|||||||
- [Table of Contents](#table-of-contents)
|
- [Table of Contents](#table-of-contents)
|
||||||
- [Features](#features)
|
- [Features](#features)
|
||||||
- [Overview](#overview)
|
- [Overview](#overview)
|
||||||
|
- [Ideas for additional seasonals](#ideas-for-additional-seasonals)
|
||||||
- [Installation](#installation)
|
- [Installation](#installation)
|
||||||
- [Client Compatibility](#client-compatibility)
|
- [Client Compatibility](#client-compatibility)
|
||||||
- [Configuration](#configuration)
|
- [Configuration](#configuration)
|
||||||
- [Automatic Theme Selection](#automatic-theme-selection)
|
- [Automatic Theme Selection](#automatic-theme-selection)
|
||||||
- [Theme Settings](#theme-settings)
|
- [Theme Settings](#theme-settings)
|
||||||
- [Build The Plugin By Yourself](#build-the-plugin-by-yourself)
|
- [Build the plugin by yourself](#build-the-plugin-by-yourself)
|
||||||
- [Troubleshooting](#troubleshooting)
|
- [Troubleshooting](#troubleshooting)
|
||||||
- [Effects Not Showing](#effects-not-showing)
|
- [Effects Not Showing](#effects-not-showing)
|
||||||
- [Docker Permission Issues](#docker-permission-issues)
|
- [Docker Permission Issues](#docker-permission-issues)
|
||||||
- [Contributing](#contributing)
|
- [Contributing](#contributing)
|
||||||
- [Legacy Manual Installation](#legacy-manual-installation)
|
|
||||||
- [Installation](#installation-1)
|
|
||||||
- [Usage](#usage)
|
|
||||||
- [Additional Directory: Separate Single Seasonals](#additional-directory-separate-single-seasonals)
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -34,39 +31,83 @@ This plugin is based on my manual mod (see the [legacy branch](https://github.co
|
|||||||
|
|
||||||
- **Automatic Theme Selection**: Dynamically updates the theme based on the date (e.g., snowflakes in December, fireworks for new year's eve).
|
- **Automatic Theme Selection**: Dynamically updates the theme based on the date (e.g., snowflakes in December, fireworks for new year's eve).
|
||||||
- **Easy Integration**: No manual file editing required. The plugin injects everything automatically.
|
- **Easy Integration**: No manual file editing required. The plugin injects everything automatically.
|
||||||
- **Configuration UI**: Configure settings directly in the Jellyfin Dashboard (very basic for now, needs some work in the future).
|
- **Configuration UI**: Configure settings directly in the Jellyfin Dashboard.
|
||||||
|
<details>
|
||||||
|
<summary>Have a look:</summary>
|
||||||
|
<img width="852" height="782" alt="Admin-Settings" src="https://github.com/user-attachments/assets/03d04ea8-7dd9-418e-88f8-9ae2937c06bb" />
|
||||||
|
</details>
|
||||||
|
- **User Toggle**: Optionally allow users to enable/disable seasonal effects from their client.
|
||||||
|
<details>
|
||||||
|
<summary>Have a look:</summary>
|
||||||
|
<img width="467" height="263" alt="Client-Settings" src="https://github.com/user-attachments/assets/a8dfc90a-16c8-409c-9133-4139f6527b0b" />
|
||||||
|
</details>
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
Click on the following themes to expand them and see the theme in action:
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Easter</summary>
|
||||||
|
|
||||||
**Easter**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Autumn</summary>
|
||||||
|
|
||||||
**Autumn**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Santa</summary>
|
||||||
|
|
||||||
**Santa**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Christmas</summary>
|
||||||
|
|
||||||
**Christmas**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Fireworks</summary>
|
||||||
|
|
||||||
**Fireworks**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Halloween</summary>
|
||||||
|
|
||||||
**Halloween**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Hearts</summary>
|
||||||
|
|
||||||
**Hearts**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Snowfall</summary>
|
||||||
|
|
||||||
**Snowfall**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Snowflakes</summary>
|
||||||
|
|
||||||
**Snowflakes**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Snowstorm</summary>
|
||||||
|
|
||||||
**Snowstorm**
|
|
||||||

|

|
||||||
|
</details>
|
||||||
|
|
||||||
|
|
||||||
|
## Ideas for additional seasonals
|
||||||
|
If you have any (specific) ideas for additional seasonals, feel free to open an issue or create a pull request. See the [contributing guidelines](CONTRIBUTING.md) for more details.
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@@ -135,7 +176,7 @@ If automatic selection is enabled, the following themes are applied based on the
|
|||||||
## Theme Settings
|
## Theme Settings
|
||||||
Each theme contains additional settings to customize its behavior. Expand the advanced configuration section to configure each theme, adjust parameters like particle count, animation speed etc.
|
Each theme contains additional settings to customize its behavior. Expand the advanced configuration section to configure each theme, adjust parameters like particle count, animation speed etc.
|
||||||
|
|
||||||
## Build The Plugin By Yourself
|
## Build the plugin by yourself
|
||||||
|
|
||||||
If you want to build the plugin yourself:
|
If you want to build the plugin yourself:
|
||||||
|
|
||||||
@@ -202,117 +243,5 @@ volumes:
|
|||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
Feel free to contribute to this project by creating pull requests or reporting issues.
|
Feel free to contribute to this project by creating pull requests or reporting issues.
|
||||||
|
For detailed contribution guidelines, please refer to the [CONTRIBUTING.md](CONTRIBUTING.md) file.
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Legacy Manual Installation
|
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary>Click to expand instructions for the old manual installation method (without plugin)</summary>
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
|
|
||||||
> [!TIP]
|
|
||||||
> Take a look at [CodeDevMLH/Jellyfin-Mods-Automated-Script](https://github.com/CodeDevMLH/Jellyfin-Mods-Automated-Script)
|
|
||||||
|
|
||||||
1. **Add Seasonal Container to `index.html`**
|
|
||||||
Edit the `index.html` file of your Jellyfin web instance. Add the following code inside the `<body>` tag:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<div class="seasonals-container"></div>
|
|
||||||
<script src="seasonals/seasonals.js"></script>
|
|
||||||
```
|
|
||||||
2. **Deploy Files**
|
|
||||||
Place the seasonals folder (including seasonals.js, CSS, and additional JavaScript files for each theme [this one](https://github.com/CodeDevMLH/Jellyfin-Seasonals/tree/legacy/seasonals)) inside the Jellyfin web server directory (labeld with "web").
|
|
||||||
|
|
||||||
3. **Configure Themes**
|
|
||||||
Customize the theme-configs.js file to modify or add new themes. The default configuration is shown below:
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
const automateThemeSelection = true; // Set to false to disable automatic theme selection based on current date
|
|
||||||
const defaultTheme = 'none'; // Default theme if automatic selection is off
|
|
||||||
|
|
||||||
const themeConfigs = {
|
|
||||||
snowflakes: {
|
|
||||||
css: 'seasonals/snowflakes.css',
|
|
||||||
js: 'seasonals/snowflakes.js',
|
|
||||||
containerClass: 'snowflakes'
|
|
||||||
},
|
|
||||||
snowfall: {
|
|
||||||
css: 'seasonals/snowfall.css',
|
|
||||||
js: 'seasonals/snowfall.js',
|
|
||||||
containerClass: 'snowfall-container'
|
|
||||||
},
|
|
||||||
|
|
||||||
// more configs...
|
|
||||||
|
|
||||||
none: {
|
|
||||||
containerClass: 'none'
|
|
||||||
},
|
|
||||||
};
|
|
||||||
```
|
|
||||||
|
|
||||||
4. **Reload the web interface**
|
|
||||||
After making these changes, restart your Jellyfin server and/or refresh the web interface (ctrl+F5, sometimes you need to clear the browsers temp files/cache (every time with firefox ;-()) to see the changes.
|
|
||||||
|
|
||||||
### Theme Settings
|
|
||||||
Each theme's JavaScript file contains additional settings to customize its behavior. Here are examples for the `autumn` and `snowflakes` themes:
|
|
||||||
|
|
||||||
**Autumn Theme Settings**
|
|
||||||
```javascript
|
|
||||||
const leaves = true; // Enable/disable leaves
|
|
||||||
const randomLeaves = true; // Enable random leaves
|
|
||||||
const randomLeavesMobile = false; // Enable random leaves on mobile devices
|
|
||||||
const enableDiffrentDuration = true; // Enable different animation duration for random leaves
|
|
||||||
const leafCount = 25; // Number of random extra leaves
|
|
||||||
```
|
|
||||||
|
|
||||||
**Snowflakes Theme Settings**
|
|
||||||
```javascript
|
|
||||||
const snowflakes = true; // Enable/disable snowflakes
|
|
||||||
const randomSnowflakes = true; // Enable random snowflakes
|
|
||||||
const randomSnowflakesMobile = false; // Enable random snowflakes on mobile devices
|
|
||||||
const enableColoredSnowflakes = true; // Enable colored snowflakes on mobile devices
|
|
||||||
const enableDiffrentDuration = true; // Enable different animation duration for random snowflakes
|
|
||||||
const snowflakeCount = 25; // Number of random extra snowflakes
|
|
||||||
```
|
|
||||||
|
|
||||||
### Usage
|
|
||||||
**Automatic Theme Selection**
|
|
||||||
By default, the theme is automatically selected based on the date. For example:
|
|
||||||
|
|
||||||
Snowfall: January
|
|
||||||
Hearts: February (Valentine's Day)
|
|
||||||
Halloween: October
|
|
||||||
|
|
||||||
Modify the determineCurrentTheme() function in seasonals.js to adjust date-based logic.
|
|
||||||
|
|
||||||
**Manual Theme Selection**
|
|
||||||
To use a fixed theme, set automateThemeSelection to false in the theme-configs.js file and specify a defaultTheme.
|
|
||||||
|
|
||||||
**Custom Themes**
|
|
||||||
1. Add your CSS and JavaScript files to the seasonals folder.
|
|
||||||
|
|
||||||
2. Extend the themeConfigs object with your theme details:
|
|
||||||
```javascript
|
|
||||||
myTheme: {
|
|
||||||
css: 'seasonals/my-theme.css',
|
|
||||||
js: 'seasonals/my-theme.js',
|
|
||||||
containerClass: 'my-theme-container',
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
### Additional Directory: Separate Single Seasonals
|
|
||||||
For users who prefer not to use the automatic seasonal theme selection, individual seasonals are available in the `separate single seasonals` folder. Each seasonal theme can be independently loaded and used without relying on the main automatic selection system.
|
|
||||||
|
|
||||||
but this requires to the modify of the `index.html` with adding the html in `add_to_index_html`.
|
|
||||||
|
|
||||||
To use a single seasonal theme, include its specific CSS and JS files in your `index.html` inside the `<body> </body>` tags provided by `add_to_index_html.html` in the sub-theme-folders as shown below:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<div class="seasonalsname-container"></div>
|
|
||||||
<script src="separate single seasonals/snowflakes.js"></script>
|
|
||||||
<link rel="stylesheet" href="separate single seasonals/snowflakes.css">
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|||||||
@@ -26,8 +26,11 @@ git merge main
|
|||||||
Setzt deine Commits neu auf die Spitze eines anderen Branches. Die Commit-IDs werden dabei neu geschrieben.
|
Setzt deine Commits neu auf die Spitze eines anderen Branches. Die Commit-IDs werden dabei neu geschrieben.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
git stash # (optional) Änderungen parken.
|
||||||
git checkout dev
|
git checkout dev
|
||||||
git rebase main
|
git fetch origin
|
||||||
|
git rebase origin/main
|
||||||
|
git stash pop # (optional) Änderungen zurückholen.
|
||||||
```
|
```
|
||||||
|
|
||||||
| Details | |
|
| Details | |
|
||||||
@@ -38,6 +41,60 @@ git rebase main
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## **Git: dev zurücksetzen & „main ist Chef“**
|
||||||
|
|
||||||
|
Problem
|
||||||
|
|
||||||
|
Beim Arbeiten mit Git passiert oft Folgendes:
|
||||||
|
|
||||||
|
- `dev` hat Commits, die nicht in `main` sind
|
||||||
|
- diese Commits brauche ich nicht mehr
|
||||||
|
- beim `git rebase origin/main` will Git trotzdem mergen oder Konflikte lösen
|
||||||
|
|
||||||
|
⚠️ Wichtiges Missverständnis:
|
||||||
|
|
||||||
|
Das sind keine lokalen Änderungen, sondern Commits, die auf `dev` existieren.
|
||||||
|
|
||||||
|
`git reset --hard origin/dev` entfernt nur lokale, nicht gepushte Commits,
|
||||||
|
aber nicht Commits, die bereits auf `origin/dev` liegen.
|
||||||
|
|
||||||
|
Praktische Befehle & sichere Vorgehensweise
|
||||||
|
|
||||||
|
- Prüfen — welche Commits würden entfernt werden:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git fetch origin
|
||||||
|
git log --oneline origin/main..origin/dev
|
||||||
|
```
|
||||||
|
|
||||||
|
- Optional: Backup des aktuellen `origin/dev`, falls etwas schiefgeht:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git fetch origin
|
||||||
|
git branch backup/dev origin/dev
|
||||||
|
```
|
||||||
|
|
||||||
|
- Sicheres Zurücksetzen von `dev` auf `main` (lokal + remote):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git checkout dev
|
||||||
|
git fetch origin
|
||||||
|
git reset --hard origin/main
|
||||||
|
git push --force-with-lease origin dev
|
||||||
|
```
|
||||||
|
|
||||||
|
- Alternative (ohne lokalen Checkout): Push von `main` direkt nach `dev`:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git fetch origin
|
||||||
|
git push --force-with-lease origin origin/main:refs/heads/dev
|
||||||
|
```
|
||||||
|
|
||||||
|
- Warnung: Diese Aktionen schreiben die Remote‑History um. Koordiniere dich mit
|
||||||
|
dem Team bevor du ein Force‑Push ausführst. Verwende `--force-with-lease`
|
||||||
|
anstelle von `--force` für etwas mehr Sicherheit.
|
||||||
|
|
||||||
|
|
||||||
## 📦 Temporäres Speichern & Spezialbefehle
|
## 📦 Temporäres Speichern & Spezialbefehle
|
||||||
|
|
||||||
### Stash (Das "Regal")
|
### Stash (Das "Regal")
|
||||||
|
|||||||
154
manifest.json
@@ -9,12 +9,20 @@
|
|||||||
"imageUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/raw/branch/main/logo.png",
|
"imageUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/raw/branch/main/logo.png",
|
||||||
"versions": [
|
"versions": [
|
||||||
{
|
{
|
||||||
"version": "1.6.11.0",
|
"version": "1.7.0.15",
|
||||||
"changelog": "- feat: Add client-side toggle option for seasonal settings",
|
"changelog": "- feat: add customizable auto seasonal list via config page\n- feat: add new season theme 'resurrection' by Bioflash257",
|
||||||
"targetAbi": "10.11.0.0",
|
"targetAbi": "10.11.0.0",
|
||||||
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/releases/download/v1.6.11.0/Jellyfin.Plugin.Seasonals.zip",
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/releases/download/v1.7.0.15/Jellyfin.Plugin.Seasonals.zip",
|
||||||
"checksum": "b3eb3bea787bd6b5bf01ccfe9f8cd61e",
|
"checksum": "d1fc094710efe45ea8cc885bc6a826c4",
|
||||||
"timestamp": "2026-02-03T21:10:33Z"
|
"timestamp": "2026-02-17T13:11:21Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.6.3.0",
|
||||||
|
"changelog": "- fix path issue on subpath installations",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/releases/download/v1.6.3.0/Jellyfin.Plugin.Seasonals.zip",
|
||||||
|
"checksum": "e9a1fb6c91b8b48978efb43c72e462a0",
|
||||||
|
"timestamp": "2026-02-15T01:12:57Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"version": "1.5.1.0",
|
"version": "1.5.1.0",
|
||||||
@@ -107,6 +115,142 @@
|
|||||||
"category": "General",
|
"category": "General",
|
||||||
"imageUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/raw/branch/main/logo.png",
|
"imageUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/raw/branch/main/logo.png",
|
||||||
"versions": [
|
"versions": [
|
||||||
|
{
|
||||||
|
"version": "1.4.0.12",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.12/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "26edee51b52dcee4ecf388aa376f3869",
|
||||||
|
"timestamp": "2026-02-04T18:07:40Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.11",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.11/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "ca0b3270eba5871e7a23db6b45bc5048",
|
||||||
|
"timestamp": "2026-02-04T17:58:14Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.10",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.10/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "9618570053f7acef445a034c1e2e044b",
|
||||||
|
"timestamp": "2026-02-04T17:51:25Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.9",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.9/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "29b3ffb9caeab135df88b6313032fc50",
|
||||||
|
"timestamp": "2026-02-04T17:39:11Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.8",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.8/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "ec343204a7cd2c1af4013e645bdddcd3",
|
||||||
|
"timestamp": "2026-02-04T17:27:22Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.7",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.7/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "450e5977228d08b8451b6047e4a6be94",
|
||||||
|
"timestamp": "2026-02-04T17:09:28Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.6",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.6/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "348ebf449ac77fd156e2afbd03e80fce",
|
||||||
|
"timestamp": "2026-02-04T16:40:21Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.5",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.5/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "3ba68bae1c492767bddab2dee2540226",
|
||||||
|
"timestamp": "2026-02-04T16:23:14Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.4",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.4/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "30a15dd883de7656c4480cfa932e9858",
|
||||||
|
"timestamp": "2026-02-04T16:17:15Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.3",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.3/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "9abf21c095e1ae99cdbeb51edb08f370",
|
||||||
|
"timestamp": "2026-02-04T15:52:11Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.2",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.2/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "6026fb8878a51f6dbe18aab1ac006df8",
|
||||||
|
"timestamp": "2026-02-04T15:45:39Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.1",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.1/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "4068c03b1ab809906d64d4faed1c1b0e",
|
||||||
|
"timestamp": "2026-02-04T15:01:50Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.4.0.0",
|
||||||
|
"changelog": "- feat: Add client-side settings feature for selected media bar settings",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.4.0.0/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "20faa2a703dbb46591f4bd09e6ab7ec3",
|
||||||
|
"timestamp": "2026-02-04T12:49:45Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.3.0.3",
|
||||||
|
"changelog": "- feat: Enhance custom media ID functionality with manual trailer override support",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.3.0.3/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "1d9e0a8342d46f84aed3f7bd1bee32d3",
|
||||||
|
"timestamp": "2026-02-04T01:41:35Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.3.0.2",
|
||||||
|
"changelog": "- feat: Enhance custom media ID functionality with manual trailer override support",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.3.0.2/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "22e79daa5f433ca09a3db4f8e37679b4",
|
||||||
|
"timestamp": "2026-02-04T01:27:55Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.3.0.1",
|
||||||
|
"changelog": "- feat: Enhance custom media ID functionality with manual trailer override support",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.3.0.1/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "5a4f555e29c733dabd51169f6ace56eb",
|
||||||
|
"timestamp": "2026-02-04T01:14:19Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.3.0.0",
|
||||||
|
"changelog": "- feat: Enhance custom media ID functionality with manual trailer override support",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/jellyfin-plugin-media-bar-enhanced/releases/download/v1.3.0.0/Jellyfin.Plugin.MediaBarEnhanced.zip",
|
||||||
|
"checksum": "83c26ba8f7ad6e1a7fe73c7190f532f3",
|
||||||
|
"timestamp": "2026-02-04T00:07:15Z"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"version": "1.2.3.7",
|
"version": "1.2.3.7",
|
||||||
"changelog": "- Fixes the issue where buttons were cut off on smaller screens such as on S24/S25.\n- Update mediaBarEnhanced.js and mediaBarEnhanced.css with version 3.0.9 from original repo",
|
"changelog": "- Fixes the issue where buttons were cut off on smaller screens such as on S24/S25.\n- Update mediaBarEnhanced.js and mediaBarEnhanced.css with version 3.0.9 from original repo",
|
||||||
|
|||||||