Compare commits
69 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ec89f2d48d | ||
|
|
61b21de566 | ||
|
|
590f2c3606 | ||
|
|
fdadc00a0c | ||
|
|
2ab88fd5ac | ||
|
|
9a41c0a2ce | ||
|
|
816f58cf02 | ||
|
|
5be9a60eed | ||
|
|
133808105e | ||
|
|
c631aca44f | ||
|
|
241450d132 | ||
|
|
d50d71bde1 | ||
|
|
262dd98519 | ||
|
|
b45ec73a67 | ||
|
|
4e8a37540f | ||
|
|
cde5201991 | ||
|
|
b2420b8eb4 | ||
|
|
dacec7d03c | ||
|
|
65f8261fb7 | ||
|
|
78872e7f96 | ||
|
|
45c9a199c2 | ||
|
|
1df6fb37b1 | ||
|
|
82a1e8a178 | ||
|
|
22bf887d10 | ||
|
|
07600766cf | ||
|
|
56298487f4 | ||
|
|
89fc1c38f0 | ||
|
|
4c168a5ec2 | ||
|
|
92d9e1a9ad | ||
|
|
007e55a612 | ||
|
|
20da9899e4 | ||
|
|
9b9cad1caa | ||
|
|
e8e3424cc9 | ||
|
|
0eeed99508 | ||
|
|
a0f261f597 | ||
|
|
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 |
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`](./Jellyfin.Plugin.Seasonals/Web/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`](./Jellyfin.Plugin.Seasonals/Web/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`](./Jellyfin.Plugin.Seasonals/Web/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! 🎉
|
||||||
207
Injector_new.cs
Normal file
@@ -0,0 +1,207 @@
|
|||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.Loader;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using MediaBrowser.Common.Configuration;
|
||||||
|
using Jellyfin.Plugin.Seasonals.Helpers;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.Seasonals;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Handles the injection of the Seasonals script into the Jellyfin web interface.
|
||||||
|
/// </summary>
|
||||||
|
public class ScriptInjector
|
||||||
|
{
|
||||||
|
private readonly IApplicationPaths _appPaths;
|
||||||
|
private readonly ILogger<ScriptInjector> _logger;
|
||||||
|
public const string ScriptTag = "<script src=\"../Seasonals/Resources/seasonals.js\" defer></script>";
|
||||||
|
public const string Marker = "</body>";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="ScriptInjector"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="appPaths">The application paths.</param>
|
||||||
|
/// <param name="logger">The logger.</param>
|
||||||
|
public ScriptInjector(IApplicationPaths appPaths, ILogger<ScriptInjector> logger)
|
||||||
|
{
|
||||||
|
_appPaths = appPaths;
|
||||||
|
_logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Injects the script tag into index.html if it's not already present.
|
||||||
|
/// </summary>
|
||||||
|
public void Inject()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var webPath = GetWebPath();
|
||||||
|
if (string.IsNullOrEmpty(webPath))
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Could not find Jellyfin web path. Script injection skipped. Attempting fallback.");
|
||||||
|
RegisterFileTransformation();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var indexPath = Path.Combine(webPath, "index.html");
|
||||||
|
if (!File.Exists(indexPath))
|
||||||
|
{
|
||||||
|
_logger.LogWarning("index.html not found at {Path}. Script injection skipped. Attempting fallback.", indexPath);
|
||||||
|
RegisterFileTransformation();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var content = File.ReadAllText(indexPath);
|
||||||
|
if (!content.Contains(ScriptTag))
|
||||||
|
{
|
||||||
|
var index = content.IndexOf(Marker, StringComparison.OrdinalIgnoreCase);
|
||||||
|
if (index != -1)
|
||||||
|
{
|
||||||
|
content = content.Insert(index, ScriptTag + Environment.NewLine);
|
||||||
|
File.WriteAllText(indexPath, content);
|
||||||
|
_logger.LogInformation("Successfully injected Seasonals script into index.html.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Script already present in index.html. Or could not be injected.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (UnauthorizedAccessException)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Unauthorized access when attempting to inject script into index.html. Automatic injection failed. Attempting fallback now...");
|
||||||
|
RegisterFileTransformation();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error injecting Seasonals script. Attempting fallback.");
|
||||||
|
RegisterFileTransformation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Removes the script tag from index.html.
|
||||||
|
/// </summary>
|
||||||
|
public void Remove()
|
||||||
|
{
|
||||||
|
UnregisterFileTransformation();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var webPath = GetWebPath();
|
||||||
|
if (string.IsNullOrEmpty(webPath))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var indexPath = Path.Combine(webPath, "index.html");
|
||||||
|
if (!File.Exists(indexPath))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var content = File.ReadAllText(indexPath);
|
||||||
|
if (content.Contains(ScriptTag))
|
||||||
|
{
|
||||||
|
content = content.Replace(ScriptTag + Environment.NewLine, "").Replace(ScriptTag, "");
|
||||||
|
File.WriteAllText(indexPath, content);
|
||||||
|
_logger.LogInformation("Successfully removed Seasonals script from index.html.");
|
||||||
|
} else {
|
||||||
|
_logger.LogInformation("Seasonals script tag not found in index.html. No removal necessary.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (UnauthorizedAccessException)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Unauthorized access when attempting to remove script from index.html.");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error removing Seasonals script.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves the path to the Jellyfin web interface directory.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The path to the web directory, or null if not found.</returns>
|
||||||
|
private string? GetWebPath()
|
||||||
|
{
|
||||||
|
// Use reflection to access WebPath property to ensure compatibility across different Jellyfin versions
|
||||||
|
var prop = _appPaths.GetType().GetProperty("WebPath", BindingFlags.Instance | BindingFlags.Public);
|
||||||
|
return prop?.GetValue(_appPaths) as string;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RegisterFileTransformation()
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Seasonals Fallback. Registering file transformations.");
|
||||||
|
|
||||||
|
List<JObject> payloads = new List<JObject>();
|
||||||
|
|
||||||
|
{
|
||||||
|
JObject payload = new JObject();
|
||||||
|
payload.Add("id", "ef1e863f-cbb0-4e47-9f23-f0cbb1826ad4");
|
||||||
|
payload.Add("fileNamePattern", "index.html");
|
||||||
|
payload.Add("callbackAssembly", GetType().Assembly.FullName);
|
||||||
|
payload.Add("callbackClass", typeof(TransformationPatches).FullName);
|
||||||
|
payload.Add("callbackMethod", nameof(TransformationPatches.IndexHtml));
|
||||||
|
|
||||||
|
payloads.Add(payload);
|
||||||
|
}
|
||||||
|
|
||||||
|
Assembly? fileTransformationAssembly =
|
||||||
|
AssemblyLoadContext.All.SelectMany(x => x.Assemblies).FirstOrDefault(x =>
|
||||||
|
x.FullName?.Contains(".FileTransformation") ?? false);
|
||||||
|
|
||||||
|
if (fileTransformationAssembly != null)
|
||||||
|
{
|
||||||
|
Type? pluginInterfaceType = fileTransformationAssembly.GetType("Jellyfin.Plugin.FileTransformation.PluginInterface");
|
||||||
|
|
||||||
|
if (pluginInterfaceType != null)
|
||||||
|
{
|
||||||
|
foreach (JObject payload in payloads)
|
||||||
|
{
|
||||||
|
pluginInterfaceType.GetMethod("RegisterTransformation")?.Invoke(null, new object?[] { payload });
|
||||||
|
}
|
||||||
|
_logger.LogInformation("File transformations registered successfully.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning("FileTransformation plugin found but PluginInterface type missing.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning("FileTransformation plugin assembly not found. Fallback injection skipped.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UnregisterFileTransformation()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Assembly? fileTransformationAssembly =
|
||||||
|
AssemblyLoadContext.All.SelectMany(x => x.Assemblies).FirstOrDefault(x =>
|
||||||
|
x.FullName?.Contains(".FileTransformation") ?? false);
|
||||||
|
|
||||||
|
if (fileTransformationAssembly != null)
|
||||||
|
{
|
||||||
|
Type? pluginInterfaceType = fileTransformationAssembly.GetType("Jellyfin.Plugin.FileTransformation.PluginInterface");
|
||||||
|
|
||||||
|
if (pluginInterfaceType != null)
|
||||||
|
{
|
||||||
|
Guid id = Guid.Parse("ef1e863f-cbb0-4e47-9f23-f0cbb1826ad4");
|
||||||
|
pluginInterfaceType.GetMethod("RemoveTransformation")?.Invoke(null, new object?[] { id });
|
||||||
|
_logger.LogInformation("File transformation unregistered successfully.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "Error attempting to unregister file transformation. It might not have been registered.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -28,6 +28,10 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
Santa = new SantaOptions();
|
Santa = new SantaOptions();
|
||||||
Easter = new EasterOptions();
|
Easter = new EasterOptions();
|
||||||
Resurrection = new ResurrectionOptions();
|
Resurrection = new ResurrectionOptions();
|
||||||
|
Spring = new SpringOptions();
|
||||||
|
Summer = new SummerOptions();
|
||||||
|
CherryBlossom = new CherryBlossomOptions();
|
||||||
|
Carnival = new CarnivalOptions();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -53,7 +57,7 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the seasonal rules configuration as JSON.
|
/// Gets or sets the seasonal rules configuration as JSON.
|
||||||
/// </summary>
|
/// </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\"}]";
|
public string SeasonalRules { get; set; } = "[{\"Name\":\"New Year Fireworks\",\"StartDay\":28,\"StartMonth\":12,\"EndDay\":5,\"EndMonth\":1,\"Theme\":\"fireworks\"},{\"Name\":\"Carnival\",\"StartDay\":19,\"StartMonth\":2,\"EndDay\":28,\"EndMonth\":2,\"Theme\":\"carnival\"},{\"Name\":\"Valentine's Day\",\"StartDay\":10,\"StartMonth\":2,\"EndDay\":18,\"EndMonth\":2,\"Theme\":\"hearts\"},{\"Name\":\"Spring\",\"StartDay\":1,\"StartMonth\":3,\"EndDay\":31,\"EndMonth\":5,\"Theme\":\"spring\"},{\"Name\":\"Summer\",\"StartDay\":1,\"StartMonth\":6,\"EndDay\":31,\"EndMonth\":8,\"Theme\":\"summer\"},{\"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.
|
||||||
@@ -69,6 +73,10 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
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 ResurrectionOptions Resurrection { get; set; }
|
||||||
|
public SpringOptions Spring { get; set; }
|
||||||
|
public SummerOptions Summer { get; set; }
|
||||||
|
public CherryBlossomOptions CherryBlossom { get; set; }
|
||||||
|
public CarnivalOptions Carnival { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class AutumnOptions
|
public class AutumnOptions
|
||||||
@@ -182,3 +190,47 @@ public class ResurrectionOptions
|
|||||||
public bool EnableRandomSymbolsMobile { get; set; } = false;
|
public bool EnableRandomSymbolsMobile { get; set; } = false;
|
||||||
public bool EnableDifferentDuration { get; set; } = true;
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public class SpringOptions
|
||||||
|
{
|
||||||
|
public int PollenCount { get; set; } = 30;
|
||||||
|
public int SunbeamCount { get; set; } = 5;
|
||||||
|
public int BirdCount { get; set; } = 4;
|
||||||
|
public int ButterflyCount { get; set; } = 4;
|
||||||
|
public int BeeCount { get; set; } = 2;
|
||||||
|
public int LadybugCount { get; set; } = 2;
|
||||||
|
public bool EnableSpring { get; set; } = true;
|
||||||
|
public bool EnableSpringSunbeams { get; set; } = true;
|
||||||
|
public bool EnableRandomSpring { get; set; } = true;
|
||||||
|
public bool EnableRandomSpringMobile { get; set; } = false;
|
||||||
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class SummerOptions
|
||||||
|
{
|
||||||
|
public int BubbleCount { get; set; } = 20;
|
||||||
|
public int DustCount { get; set; } = 50;
|
||||||
|
public bool EnableSummer { get; set; } = true;
|
||||||
|
public bool EnableRandomSummer { get; set; } = true;
|
||||||
|
public bool EnableRandomSummerMobile { get; set; } = false;
|
||||||
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class CarnivalOptions
|
||||||
|
{
|
||||||
|
public int ObjectCount { get; set; } = 25;
|
||||||
|
public bool EnableCarnival { get; set; } = true;
|
||||||
|
public bool EnableRandomCarnival { get; set; } = true;
|
||||||
|
public bool EnableRandomCarnivalMobile { get; set; } = false;
|
||||||
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
|
public bool EnableCarnivalSway { get; set; } = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class CherryBlossomOptions
|
||||||
|
{
|
||||||
|
public int PetalCount { get; set; } = 25;
|
||||||
|
public bool EnableCherryBlossom { get; set; } = true;
|
||||||
|
public bool EnableRandomCherryBlossom { get; set; } = true;
|
||||||
|
public bool EnableRandomCherryBlossomMobile { get; set; } = false;
|
||||||
|
public bool EnableDifferentDuration { get; set; } = true;
|
||||||
|
}
|
||||||
|
|||||||
@@ -62,18 +62,21 @@
|
|||||||
<option value="halloween">Halloween</option>
|
<option value="halloween">Halloween</option>
|
||||||
<option value="hearts">Hearts</option>
|
<option value="hearts">Hearts</option>
|
||||||
<option value="christmas">Christmas</option>
|
<option value="christmas">Christmas</option>
|
||||||
<option value="santa">Santa</option>
|
<option value="santa">Santa (flying santa & snowfall)</option>
|
||||||
<option value="autumn">Autumn</option>
|
<option value="autumn">Autumn (falling leaves)</option>
|
||||||
<option value="easter">Easter</option>
|
<option value="easter">Easter</option>
|
||||||
<option value="resurrection">Resurrection</option>
|
<option value="summer">Summer (Bubbles)</option>
|
||||||
<option value="summer" disabled>Summer (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="spring">Spring</option>
|
||||||
<option value="spring" disabled>Spring (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="carnival">Carnival (Confetti)</option>
|
||||||
<option value="oktoberfest" disabled>Oktoberfest (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="cherryblossom">Cherry Blossom</option>
|
||||||
<option value="carnival" disabled>Carnival (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="resurrection">Resurrection by Bioflash257</option>
|
||||||
<option value="championships" disabled>European/World Championships (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="championships" disabled>European/World Championships (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="patrick" disabled>St. Patrick's Day (not implemented yet. Please commit ideas in a issue or PR)</option>
|
<option value="patrick" disabled>St. Patrick's Day (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
<option value="thanksgiving" disabled>Thanksgiving (not implemented yet. Please commit ideas in a issue or PR)</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>
|
<option value="pride" disabled>Pride (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
|
<option value="pride" disabled>Oscar Awards (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
|
<option value="pride" disabled>Eurovison Awards (not implemented yet. Please commit ideas in a issue or PR)</option>
|
||||||
|
<option value="pride" disabled>Sugar Feast (Eid al-Fitr, Ramadan) (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 or no "Auto Selection" rule matches the current date.</div>
|
<div class="fieldDescription">The season to display if automation is disabled or no "Auto Selection" rule matches the current date.</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -88,17 +91,20 @@
|
|||||||
|
|
||||||
<!-- Auto Selection Tab -->
|
<!-- Auto Selection Tab -->
|
||||||
<div id="seasonals-auto-selection" class="seasonals-tab-content" style="display: none;">
|
<div id="seasonals-auto-selection" class="seasonals-tab-content" style="display: none;">
|
||||||
<div style="background-color: rgba(255, 255, 255, 0.05); border-left: 4px solid #00a4dc; border-radius: 4px; padding: 1em 1.5em; margin-bottom: 1.5em;">
|
<h2>Auto Selection Rules</h2>
|
||||||
<h3>Auto Selection Rules</h3>
|
|
||||||
<p>Define rules to automatically select a season based on the date. Rules are evaluated from top to bottom. The first matching rule wins.</p>
|
<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>
|
||||||
|
|
||||||
<div id="seasonalRulesList">
|
<div id="seasonalRulesList">
|
||||||
<!-- Rules will be injected here via JS -->
|
<!-- Rules will be injected here via JS -->
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button is="emby-button" type="button" class="raised button-accent block" onclick="SeasonalsConfigPage.addRule();">
|
<button is="emby-button" type="button" class="raised emby-button" onclick="SeasonalsConfigPage.addRule();"
|
||||||
<i class="material-icons button-icon button-icon-left">add</i>
|
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>
|
<span>Add Rule</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -573,6 +579,202 @@
|
|||||||
<div class="fieldDescription">Randomize the movement speed.</div>
|
<div class="fieldDescription">Randomize the movement speed.</div>
|
||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Spring</summary>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableSpring" name="EnableSpring" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span class="checkboxLabel">Enable Spring</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enables the Spring theme (grass, pollen).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomSpring" name="EnableRandomSpring" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Animation Assets</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enables animated spring assets (birds, butterflies, bees, etc.).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomSpringMobile" name="EnableRandomSpringMobile" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Animation Assets on Mobile</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays animated assets on mobile devices. Warning: High values may affect performance.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringPollenCount">Pollen Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringPollenCount" name="SpringPollenCount" />
|
||||||
|
<div class="fieldDescription">Number of pollen particles (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringSunbeamCount">Sunbeam Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringSunbeamCount" name="SpringSunbeamCount" />
|
||||||
|
<div class="fieldDescription">Number of sunbeams (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableSpringSunbeams" name="EnableSpringSunbeams" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Sunbeams</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Display sunbeams at the top of the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringBirdCount">Bird Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringBirdCount" name="SpringBirdCount" />
|
||||||
|
<div class="fieldDescription">Number of birds flying across the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringButterflyCount">Butterfly Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringButterflyCount" name="SpringButterflyCount" />
|
||||||
|
<div class="fieldDescription">Number of butterflies.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringBeeCount">Bee Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringBeeCount" name="SpringBeeCount" />
|
||||||
|
<div class="fieldDescription">Number of bees.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SpringLadybugCount">Ladybug Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SpringLadybugCount" name="SpringLadybugCount" />
|
||||||
|
<div class="fieldDescription">Number of ladybugs walking along the bottom.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableDifferentDurationSpring" name="EnableDifferentDurationSpring" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Different Duration</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Randomize the animations duration.</div>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Summer</summary>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableSummer" name="EnableSummer" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Summer Seasonal</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enable the Summer theme in general (e.g. for automation).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomSummer" name="EnableRandomSummer" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Bubbles and Dust</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional bubbles and dust particles rising across the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomSummerMobile" name="EnableRandomSummerMobile" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Bubbles and Dust on Mobile</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional bubbles and dust particles rising across the screen on mobile devices. Warning: High values may affect performance.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SummerBubbleCount">Bubbles Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SummerBubbleCount" name="SummerBubbleCount" />
|
||||||
|
<div class="fieldDescription">Number of bubbles (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="SummerDustCount">Dust Count</label>
|
||||||
|
<input is="emby-input" type="number" id="SummerDustCount" name="SummerDustCount" />
|
||||||
|
<div class="fieldDescription">Number of dust particles (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableDifferentDurationSummer" name="EnableDifferentDurationSummer" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Different Rising Speed</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Randomize the rising speed of bubbles.</div>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Carnival</summary>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableCarnival" name="EnableCarnival" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Carnival Seasonal</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enable the Carnival theme in general (e.g. for automation).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomCarnival" name="EnableRandomCarnival" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Confetti</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional confetti falling and fluttering across the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomCarnivalMobile" name="EnableRandomCarnivalMobile" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Confetti on Mobile</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional confetti falling and fluttering across the screen on mobile devices. Warning: High values may affect performance.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="CarnivalObjectCount">Confetti Count</label>
|
||||||
|
<input is="emby-input" type="number" id="CarnivalObjectCount" name="CarnivalObjectCount" />
|
||||||
|
<div class="fieldDescription">Number of additional confetti pieces (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableDifferentDurationCarnival" name="EnableDifferentDurationCarnival" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Different Falling Speed</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Randomize the falling speed of confetti.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableCarnivalSway" name="EnableCarnivalSway" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Sway</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enable sway animation for carnival confetti.</div>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
<hr style="max-width: 800px; margin: 1em 0;">
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Cherry Blossom</summary>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableCherryBlossom" name="EnableCherryBlossom" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Cherry Blossom Seasonal</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Enable the Cherry Blossom theme in general (e.g. for automation).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomCherryBlossom" name="EnableRandomCherryBlossom" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Cherry Blossoms</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional cherry blossoms falling and fluttering across the screen.</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableRandomCherryBlossomMobile" name="EnableRandomCherryBlossomMobile" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Additional Random Cherry Blossoms on Mobile</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Displays additional cherry blossoms falling and fluttering across the screen on mobile devices. Warning: High values may affect performance.</div>
|
||||||
|
</div>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel" for="CherryBlossomPetalCount">Petal Count</label>
|
||||||
|
<input is="emby-input" type="number" id="CherryBlossomPetalCount" name="CherryBlossomPetalCount" />
|
||||||
|
<div class="fieldDescription">Number of additional cherry blossoms (if enabled).</div>
|
||||||
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="EnableDifferentDurationCherryBlossom" name="EnableDifferentDurationCherryBlossom" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Enable Different Falling Speed</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Randomize the falling speed of cherry blossoms.</div>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
</div>
|
</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;">
|
||||||
@@ -613,8 +815,7 @@
|
|||||||
}
|
}
|
||||||
.seasonal-rule-content {
|
.seasonal-rule-content {
|
||||||
display: grid;
|
display: grid;
|
||||||
/* grid-template-columns: 2fr 1fr 1fr 2fr; */
|
grid-template-columns: 2fr 1.3fr 1.3fr 2fr;
|
||||||
grid-template-columns: 2fr 1.2fr 1.2fr 1.6fr;
|
|
||||||
gap: 15px;
|
gap: 15px;
|
||||||
align-items: end;
|
align-items: end;
|
||||||
}
|
}
|
||||||
@@ -626,7 +827,13 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
align-items: flex-end;
|
align-items: flex-end;
|
||||||
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
|
.date-range-group > .inputContainer,
|
||||||
|
.date-range-group > .selectContainer {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
@media (max-width: 800px) {
|
@media (max-width: 800px) {
|
||||||
.seasonal-rule-content {
|
.seasonal-rule-content {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
@@ -639,10 +846,10 @@
|
|||||||
border-bottom: 2px solid #00a4dc !important;
|
border-bottom: 2px solid #00a4dc !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Disabled options in selects
|
/* Disabled options in selects */
|
||||||
select option:disabled {
|
select option:disabled {
|
||||||
color: #a3a3a3 !important;
|
color: #a3a3a3 !important;
|
||||||
} */
|
}
|
||||||
</style>
|
</style>
|
||||||
<script>
|
<script>
|
||||||
function showSeasonalsTab(tabId, btn) {
|
function showSeasonalsTab(tabId, btn) {
|
||||||
@@ -677,86 +884,117 @@
|
|||||||
var endMonth = data ? (data.EndMonth !== undefined ? data.EndMonth : (data.endMonth !== undefined ? data.endMonth : 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 theme = data ? (data.Theme || data.theme || 'none') : 'none';
|
||||||
|
|
||||||
div.innerHTML = `
|
var days = [];
|
||||||
<div class="seasonal-rule-header">
|
for (var i = 1; i <= 31; i++) days.push(i);
|
||||||
<div style="font-weight: bold; font-size: 1.1em;" class="rule-title">${name}</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="this.closest('.seasonal-rule').querySelector('.rule-title').innerText = this.value" />
|
|
||||||
</div>
|
|
||||||
<div class="date-range-group">
|
|
||||||
<div class="inputContainer" style="margin:0;">
|
|
||||||
<label class="inputLabel">Start Day</label>
|
|
||||||
<input is="emby-input" type="number" class="rule-start-day" min="1" max="31" />
|
|
||||||
</div>
|
|
||||||
<div class="inputContainer" style="margin:0;">
|
|
||||||
<label class="inputLabel">Month</label>
|
|
||||||
<input is="emby-input" type="number" class="rule-start-month" min="1" max="12" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="date-range-group">
|
var months = [
|
||||||
<div class="inputContainer" style="margin:0;">
|
{ v: 1, n: 'Jan' }, { v: 2, n: 'Feb' }, { v: 3, n: 'Mar' }, { v: 4, n: 'Apr' },
|
||||||
<label class="inputLabel">End Day</label>
|
{ v: 5, n: 'May' }, { v: 6, n: 'Jun' }, { v: 7, n: 'Jul' }, { v: 8, n: 'Aug' },
|
||||||
<input is="emby-input" type="number" class="rule-end-day" min="1" max="31" />
|
{ v: 9, n: 'Sep' }, { v: 10, n: 'Oct' }, { v: 11, n: 'Nov' }, { v: 12, n: 'Dec' }
|
||||||
</div>
|
];
|
||||||
<div class="inputContainer" style="margin:0;">
|
|
||||||
<label class="inputLabel">Month</label>
|
|
||||||
<input is="emby-input" type="number" class="rule-end-month" min="1" max="12" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="selectContainer" style="margin:0;">
|
// Build select HTML via string concatenation to avoid Jellyfin's ${} localization processing
|
||||||
<label class="selectLabel">Theme</label>
|
function mkSelect(val, opts, cls) {
|
||||||
<select is="emby-select" class="rule-theme" style="width: 100%;">
|
var h = '<select class="emby-select emby-select-withcolor ' + cls + '" style="width: 100%; -webkit-appearance: menulist; appearance: menulist;">';
|
||||||
<option value="none">None</option>
|
opts.forEach(function(o) {
|
||||||
<option value="snowflakes">Snowflakes</option>
|
var v = o.v || o;
|
||||||
<option value="snowfall">Snowfall</option>
|
var n = o.n || o;
|
||||||
<option value="snowstorm">Snowstorm</option>
|
h += '<option value="' + v + '" ' + (v == val ? 'selected' : '') + '>' + n + '</option>';
|
||||||
<option value="fireworks">Fireworks</option>
|
});
|
||||||
<option value="halloween">Halloween</option>
|
h += '</select>';
|
||||||
<option value="hearts">Hearts</option>
|
return h;
|
||||||
<option value="christmas">Christmas</option>
|
}
|
||||||
<option value="santa">Santa</option>
|
|
||||||
<option value="autumn">Autumn</option>
|
div.innerHTML =
|
||||||
<option value="easter">Easter</option>
|
'<div class="seasonal-rule-header">' +
|
||||||
</select>
|
' <div style="font-weight: bold; font-size: 1.1em;" class="rule-title"></div>' +
|
||||||
</div>
|
' <div class="rule-actions">' +
|
||||||
</div>
|
' <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 (flying santa & snowfall)</option>' +
|
||||||
|
' <option value="autumn">Autumn (falling leaves)</option>' +
|
||||||
|
' <option value="easter">Easter</option>' +
|
||||||
|
' <option value="summer">Summer (Bubbles)</option>' +
|
||||||
|
' <option value="spring">Spring</option>' +
|
||||||
|
' <option value="carnival">Carnival (Confetti)</option>' +
|
||||||
|
' <option value="cherryblossom">Cherry Blossom</option>' +
|
||||||
|
' <option value="resurrection">Resurrection by Bioflash257</option>' +
|
||||||
|
' </select>' +
|
||||||
|
' </div>' +
|
||||||
|
'</div>';
|
||||||
|
|
||||||
container.appendChild(div);
|
container.appendChild(div);
|
||||||
|
|
||||||
// Set values programmatically to ensure they are correctly populated
|
// Set values programmatically
|
||||||
div.querySelector('.rule-name').value = name;
|
div.querySelector('.rule-name').value = name;
|
||||||
div.querySelector('.rule-title').innerText = name;
|
|
||||||
div.querySelector('.rule-start-day').value = startDay;
|
|
||||||
div.querySelector('.rule-start-month').value = startMonth;
|
|
||||||
div.querySelector('.rule-end-day').value = endDay;
|
|
||||||
div.querySelector('.rule-end-month').value = endMonth;
|
|
||||||
div.querySelector('.rule-theme').value = theme;
|
div.querySelector('.rule-theme').value = theme;
|
||||||
|
|
||||||
// Bind events
|
// Bind events
|
||||||
div.querySelector('.btn-remove').addEventListener('click', function() {
|
div.querySelector('.btn-remove').addEventListener('click', function() {
|
||||||
div.remove();
|
div.remove();
|
||||||
|
SeasonalsConfigPage.updateRuleTitles();
|
||||||
});
|
});
|
||||||
div.querySelector('.btn-move-up').addEventListener('click', function() {
|
div.querySelector('.btn-move-up').addEventListener('click', function() {
|
||||||
if (div.previousElementSibling) {
|
if (div.previousElementSibling) {
|
||||||
container.insertBefore(div, div.previousElementSibling);
|
container.insertBefore(div, div.previousElementSibling);
|
||||||
|
SeasonalsConfigPage.updateRuleTitles();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
div.querySelector('.btn-move-down').addEventListener('click', function() {
|
div.querySelector('.btn-move-down').addEventListener('click', function() {
|
||||||
if (div.nextElementSibling) {
|
if (div.nextElementSibling) {
|
||||||
container.insertBefore(div.nextElementSibling, div);
|
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) {
|
renderRules: function(rules) {
|
||||||
@@ -892,6 +1130,42 @@
|
|||||||
document.querySelector('#EnableRandomResurrectionMobile').checked = config.Resurrection.EnableRandomSymbolsMobile;
|
document.querySelector('#EnableRandomResurrectionMobile').checked = config.Resurrection.EnableRandomSymbolsMobile;
|
||||||
document.querySelector('#EnableDifferentDurationResurrection').checked = config.Resurrection.EnableDifferentDuration;
|
document.querySelector('#EnableDifferentDurationResurrection').checked = config.Resurrection.EnableDifferentDuration;
|
||||||
|
|
||||||
|
// Spring
|
||||||
|
document.querySelector('#EnableSpring').checked = config.Spring.EnableSpring;
|
||||||
|
document.querySelector('#EnableSpringSunbeams').checked = config.Spring.EnableSpringSunbeams !== undefined ? config.Spring.EnableSpringSunbeams : true;
|
||||||
|
document.querySelector('#SpringPollenCount').value = config.Spring.PollenCount;
|
||||||
|
document.querySelector('#SpringSunbeamCount').value = config.Spring.SunbeamCount;
|
||||||
|
document.querySelector('#SpringBirdCount').value = config.Spring.BirdCount !== undefined ? config.Spring.BirdCount : 3;
|
||||||
|
document.querySelector('#SpringButterflyCount').value = config.Spring.ButterflyCount !== undefined ? config.Spring.ButterflyCount : 2;
|
||||||
|
document.querySelector('#SpringBeeCount').value = config.Spring.BeeCount !== undefined ? config.Spring.BeeCount : 1;
|
||||||
|
document.querySelector('#SpringLadybugCount').value = config.Spring.LadybugCount !== undefined ? config.Spring.LadybugCount : 1;
|
||||||
|
document.querySelector('#EnableRandomSpring').checked = config.Spring.EnableRandomSpring;
|
||||||
|
document.querySelector('#EnableRandomSpringMobile').checked = config.Spring.EnableRandomSpringMobile;
|
||||||
|
document.querySelector('#EnableDifferentDurationSpring').checked = config.Spring.EnableDifferentDuration;
|
||||||
|
|
||||||
|
// Summer
|
||||||
|
document.querySelector('#EnableSummer').checked = config.Summer.EnableSummer;
|
||||||
|
document.querySelector('#SummerBubbleCount').value = config.Summer.BubbleCount;
|
||||||
|
document.querySelector('#SummerDustCount').value = config.Summer.DustCount;
|
||||||
|
document.querySelector('#EnableRandomSummer').checked = config.Summer.EnableRandomSummer;
|
||||||
|
document.querySelector('#EnableRandomSummerMobile').checked = config.Summer.EnableRandomSummerMobile;
|
||||||
|
document.querySelector('#EnableDifferentDurationSummer').checked = config.Summer.EnableDifferentDuration;
|
||||||
|
|
||||||
|
// Carnival
|
||||||
|
document.querySelector('#EnableCarnival').checked = config.Carnival.EnableCarnival;
|
||||||
|
document.querySelector('#EnableCarnivalSway').checked = config.Carnival.EnableCarnivalSway !== undefined ? config.Carnival.EnableCarnivalSway : true;
|
||||||
|
document.querySelector('#CarnivalObjectCount').value = config.Carnival.ObjectCount;
|
||||||
|
document.querySelector('#EnableRandomCarnival').checked = config.Carnival.EnableRandomCarnival;
|
||||||
|
document.querySelector('#EnableRandomCarnivalMobile').checked = config.Carnival.EnableRandomCarnivalMobile;
|
||||||
|
document.querySelector('#EnableDifferentDurationCarnival').checked = config.Carnival.EnableDifferentDuration;
|
||||||
|
|
||||||
|
// Cherry Blossom
|
||||||
|
document.querySelector('#EnableCherryBlossom').checked = config.CherryBlossom.EnableCherryBlossom;
|
||||||
|
document.querySelector('#CherryBlossomPetalCount').value = config.CherryBlossom.PetalCount;
|
||||||
|
document.querySelector('#EnableRandomCherryBlossom').checked = config.CherryBlossom.EnableRandomCherryBlossom;
|
||||||
|
document.querySelector('#EnableRandomCherryBlossomMobile').checked = config.CherryBlossom.EnableRandomCherryBlossomMobile;
|
||||||
|
document.querySelector('#EnableDifferentDurationCherryBlossom').checked = config.CherryBlossom.EnableDifferentDuration;
|
||||||
|
|
||||||
Dashboard.hideLoadingMsg();
|
Dashboard.hideLoadingMsg();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -999,6 +1273,42 @@
|
|||||||
config.Resurrection.EnableRandomSymbolsMobile = document.querySelector('#EnableRandomResurrectionMobile').checked;
|
config.Resurrection.EnableRandomSymbolsMobile = document.querySelector('#EnableRandomResurrectionMobile').checked;
|
||||||
config.Resurrection.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationResurrection').checked;
|
config.Resurrection.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationResurrection').checked;
|
||||||
|
|
||||||
|
// Spring
|
||||||
|
config.Spring.EnableSpring = document.querySelector('#EnableSpring').checked;
|
||||||
|
config.Spring.EnableSpringSunbeams = document.querySelector('#EnableSpringSunbeams').checked;
|
||||||
|
config.Spring.PollenCount = parseInt(document.querySelector('#SpringPollenCount').value);
|
||||||
|
config.Spring.SunbeamCount = parseInt(document.querySelector('#SpringSunbeamCount').value);
|
||||||
|
config.Spring.BirdCount = parseInt(document.querySelector('#SpringBirdCount').value);
|
||||||
|
config.Spring.ButterflyCount = parseInt(document.querySelector('#SpringButterflyCount').value);
|
||||||
|
config.Spring.BeeCount = parseInt(document.querySelector('#SpringBeeCount').value);
|
||||||
|
config.Spring.LadybugCount = parseInt(document.querySelector('#SpringLadybugCount').value);
|
||||||
|
config.Spring.EnableRandomSpring = document.querySelector('#EnableRandomSpring').checked;
|
||||||
|
config.Spring.EnableRandomSpringMobile = document.querySelector('#EnableRandomSpringMobile').checked;
|
||||||
|
config.Spring.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationSpring').checked;
|
||||||
|
|
||||||
|
// Summer
|
||||||
|
config.Summer.EnableSummer = document.querySelector('#EnableSummer').checked;
|
||||||
|
config.Summer.BubbleCount = parseInt(document.querySelector('#SummerBubbleCount').value);
|
||||||
|
config.Summer.DustCount = parseInt(document.querySelector('#SummerDustCount').value);
|
||||||
|
config.Summer.EnableRandomSummer = document.querySelector('#EnableRandomSummer').checked;
|
||||||
|
config.Summer.EnableRandomSummerMobile = document.querySelector('#EnableRandomSummerMobile').checked;
|
||||||
|
config.Summer.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationSummer').checked;
|
||||||
|
|
||||||
|
// Carnival
|
||||||
|
config.Carnival.EnableCarnival = document.querySelector('#EnableCarnival').checked;
|
||||||
|
config.Carnival.EnableCarnivalSway = document.querySelector('#EnableCarnivalSway').checked;
|
||||||
|
config.Carnival.ObjectCount = parseInt(document.querySelector('#CarnivalObjectCount').value);
|
||||||
|
config.Carnival.EnableRandomCarnival = document.querySelector('#EnableRandomCarnival').checked;
|
||||||
|
config.Carnival.EnableRandomCarnivalMobile = document.querySelector('#EnableRandomCarnivalMobile').checked;
|
||||||
|
config.Carnival.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationCarnival').checked;
|
||||||
|
|
||||||
|
// Cherry Blossom
|
||||||
|
config.CherryBlossom.EnableCherryBlossom = document.querySelector('#EnableCherryBlossom').checked;
|
||||||
|
config.CherryBlossom.PetalCount = parseInt(document.querySelector('#CherryBlossomPetalCount').value);
|
||||||
|
config.CherryBlossom.EnableRandomCherryBlossom = document.querySelector('#EnableRandomCherryBlossom').checked;
|
||||||
|
config.CherryBlossom.EnableRandomCherryBlossomMobile = document.querySelector('#EnableRandomCherryBlossomMobile').checked;
|
||||||
|
config.CherryBlossom.EnableDifferentDuration = document.querySelector('#EnableDifferentDurationCherryBlossom').checked;
|
||||||
|
|
||||||
ApiClient.updatePluginConfiguration(SeasonalsConfigPage.pluginUniqueId, config).then(function (result) {
|
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.7.0.6</Version>
|
<Version>1.7.1.5</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" />
|
||||||
|
|||||||
@@ -56,6 +56,18 @@ public class ScriptInjector
|
|||||||
}
|
}
|
||||||
|
|
||||||
var content = File.ReadAllText(indexPath);
|
var content = File.ReadAllText(indexPath);
|
||||||
|
|
||||||
|
|
||||||
|
// MARK: Legacy Tags, remove in future versions
|
||||||
|
bool modified = false;
|
||||||
|
// Cleanup legacy tags first to avoid duplicates or conflicts
|
||||||
|
content = RemoveLegacyTags(content, ref modified);
|
||||||
|
if (modified)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Removed legacy tags from index.html.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!content.Contains(ScriptTag))
|
if (!content.Contains(ScriptTag))
|
||||||
{
|
{
|
||||||
var index = content.IndexOf(Marker, StringComparison.OrdinalIgnoreCase);
|
var index = content.IndexOf(Marker, StringComparison.OrdinalIgnoreCase);
|
||||||
@@ -113,6 +125,17 @@ public class ScriptInjector
|
|||||||
} else {
|
} else {
|
||||||
_logger.LogInformation("Seasonals script tag not found in index.html. No removal necessary.");
|
_logger.LogInformation("Seasonals script tag not found in index.html. No removal necessary.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: Legacy Tags, remove in future versions
|
||||||
|
// Remove legacy tags
|
||||||
|
bool modified = false;
|
||||||
|
content = RemoveLegacyTags(content, ref modified);
|
||||||
|
if (modified)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Removed legacy tags from index.html.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
catch (UnauthorizedAccessException)
|
catch (UnauthorizedAccessException)
|
||||||
{
|
{
|
||||||
@@ -204,4 +227,21 @@ public class ScriptInjector
|
|||||||
_logger.LogWarning(ex, "Error attempting to unregister file transformation. It might not have been registered.");
|
_logger.LogWarning(ex, "Error attempting to unregister file transformation. It might not have been registered.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: Legacy Tags, remove in future versions
|
||||||
|
/// <summary>
|
||||||
|
/// Removes legacy script tags from the content.
|
||||||
|
/// </summary>
|
||||||
|
private string RemoveLegacyTags(string content, ref bool modified)
|
||||||
|
{
|
||||||
|
// Legacy tags (used in versions prior to 1.6.3.0 where paths started with / instead of ../)
|
||||||
|
const string LegacyScriptTag = "<script src=\"/Seasonals/Resources/seasonals.js\" defer></script>";
|
||||||
|
|
||||||
|
if (content.Contains(LegacyScriptTag))
|
||||||
|
{
|
||||||
|
content = content.Replace(LegacyScriptTag + Environment.NewLine, "").Replace(LegacyScriptTag, "");
|
||||||
|
modified = true;
|
||||||
|
}
|
||||||
|
return content;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
85
Jellyfin.Plugin.Seasonals/Web/carnival.css
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
.carnival-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 10;
|
||||||
|
perspective: 600px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-wrapper {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 15;
|
||||||
|
top: -20px;
|
||||||
|
will-change: top;
|
||||||
|
animation-name: carnival-fall;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: 1;
|
||||||
|
animation-fill-mode: forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-sway-wrapper {
|
||||||
|
will-change: transform;
|
||||||
|
animation-name: carnival-sway;
|
||||||
|
animation-timing-function: ease-in-out;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
animation-direction: alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-confetti {
|
||||||
|
width: 8px;
|
||||||
|
height: 16px;
|
||||||
|
background-color: #f0f;
|
||||||
|
will-change: transform;
|
||||||
|
animation-name: carnival-flutter;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-confetti.circle {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-confetti.square {
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.carnival-confetti.triangle {
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
clip-path: polygon(50% 0%, 0% 100%, 100% 100%);
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes carnival-fall {
|
||||||
|
0% {
|
||||||
|
top: -10%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
top: 110%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes carnival-sway {
|
||||||
|
0% {
|
||||||
|
transform: translateX(calc(var(--sway-amount, 50px) * -1));
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: translateX(var(--sway-amount, 50px));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes carnival-flutter {
|
||||||
|
0% {
|
||||||
|
transform: rotate3d(var(--rx, 1), var(--ry, 1), var(--rz, 0), 0deg);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: rotate3d(var(--rx, 1), var(--ry, 1), var(--rz, 0), var(--rot-dir, 360deg));
|
||||||
|
}
|
||||||
|
}
|
||||||
197
Jellyfin.Plugin.Seasonals/Web/carnival.js
Normal file
@@ -0,0 +1,197 @@
|
|||||||
|
const config = window.SeasonalsPluginConfig?.Carnival || {};
|
||||||
|
|
||||||
|
const carnival = config.EnableCarnival !== undefined ? config.EnableCarnival : true; // Enable/disable carnival
|
||||||
|
const randomCarnival = config.EnableRandomCarnival !== undefined ? config.EnableRandomCarnival : true; // Enable random carnival objects
|
||||||
|
const randomCarnivalMobile = config.EnableRandomCarnivalMobile !== undefined ? config.EnableRandomCarnivalMobile : false; // Enable random carnival objects on mobile
|
||||||
|
const enableDifferentDuration = config.EnableDifferentDuration !== undefined ? config.EnableDifferentDuration : true; // Randomize falling and flutter speeds
|
||||||
|
const enableSway = config.EnableCarnivalSway !== undefined ? config.EnableCarnivalSway : true; // Enable side-to-side sway animation
|
||||||
|
const carnivalCount = config.ObjectCount || 120; // Number of confetti pieces to spawn
|
||||||
|
|
||||||
|
let msgPrinted = false;
|
||||||
|
|
||||||
|
// function to check and control the carnival animation
|
||||||
|
function toggleCarnival() {
|
||||||
|
const carnivalContainer = document.querySelector('.carnival-container');
|
||||||
|
if (!carnivalContainer) 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');
|
||||||
|
|
||||||
|
// hide carnival if video/trailer player is active or dashboard is visible
|
||||||
|
if (videoPlayer || trailerPlayer || isDashboard || hasUserMenu) {
|
||||||
|
carnivalContainer.style.display = 'none'; // hide carnival
|
||||||
|
if (!msgPrinted) {
|
||||||
|
console.log('Carnival hidden');
|
||||||
|
msgPrinted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
carnivalContainer.style.display = 'block'; // show carnival
|
||||||
|
if (msgPrinted) {
|
||||||
|
console.log('Carnival visible');
|
||||||
|
msgPrinted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// observe changes in the DOM
|
||||||
|
const observer = new MutationObserver(toggleCarnival);
|
||||||
|
|
||||||
|
// start observation
|
||||||
|
observer.observe(document.body, {
|
||||||
|
childList: true, // observe adding/removing of child elements
|
||||||
|
subtree: true, // observe all levels of the DOM tree
|
||||||
|
attributes: true // observe changes to attributes (e.g. class changes)
|
||||||
|
});
|
||||||
|
|
||||||
|
const confettiColors = [
|
||||||
|
'#fce18a', '#ff726d', '#b48def', '#f4306d',
|
||||||
|
'#36c5f0', '#2ccc5d', '#e9b31d', '#9b59b6',
|
||||||
|
'#3498db', '#e74c3c', '#1abc9c', '#f1c40f'
|
||||||
|
];
|
||||||
|
|
||||||
|
function createConfettiPiece(container, isInitial = false) {
|
||||||
|
const wrapper = document.createElement('div');
|
||||||
|
wrapper.classList.add('carnival-wrapper');
|
||||||
|
|
||||||
|
let swayWrapper = wrapper;
|
||||||
|
|
||||||
|
if (enableSway) {
|
||||||
|
swayWrapper = document.createElement('div');
|
||||||
|
swayWrapper.classList.add('carnival-sway-wrapper');
|
||||||
|
wrapper.appendChild(swayWrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
const confetti = document.createElement('div');
|
||||||
|
confetti.classList.add('carnival-confetti');
|
||||||
|
|
||||||
|
// Random color
|
||||||
|
const color = confettiColors[Math.floor(Math.random() * confettiColors.length)];
|
||||||
|
confetti.style.backgroundColor = color;
|
||||||
|
|
||||||
|
// Random shape
|
||||||
|
const shape = Math.random();
|
||||||
|
if (shape > 0.8) {
|
||||||
|
confetti.classList.add('circle');
|
||||||
|
} else if (shape > 0.6) {
|
||||||
|
confetti.classList.add('square');
|
||||||
|
} else if (shape > 0.4) {
|
||||||
|
confetti.classList.add('triangle');
|
||||||
|
} else {
|
||||||
|
confetti.classList.add('rect');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Random position
|
||||||
|
wrapper.style.left = `${Math.random() * 100}%`;
|
||||||
|
|
||||||
|
// Random dimensions
|
||||||
|
// MARK: CONFETTI SIZE (RECTANGLES)
|
||||||
|
if (!confetti.classList.contains('circle') && !confetti.classList.contains('square') && !confetti.classList.contains('triangle')) {
|
||||||
|
const width = Math.random() * 3 + 4; // 4-7px
|
||||||
|
const height = Math.random() * 5 + 8; // 8-13px
|
||||||
|
confetti.style.width = `${width}px`;
|
||||||
|
confetti.style.height = `${height}px`;
|
||||||
|
} else if (confetti.classList.contains('circle') || confetti.classList.contains('square')) {
|
||||||
|
// MARK: CONFETTI SIZE (CIRCLES/SQUARES)
|
||||||
|
const size = Math.random() * 5 + 5; // 5-10px
|
||||||
|
confetti.style.width = `${size}px`;
|
||||||
|
confetti.style.height = `${size}px`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Animation settings
|
||||||
|
// MARK: CONFETTI FALLING SPEED (in seconds)
|
||||||
|
const duration = Math.random() * 5 + 5;
|
||||||
|
|
||||||
|
let delay = 0;
|
||||||
|
if (isInitial) {
|
||||||
|
delay = -Math.random() * duration;
|
||||||
|
} else {
|
||||||
|
delay = Math.random() * 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.style.animationDelay = `${delay}s`;
|
||||||
|
wrapper.style.animationDuration = `${duration}s`;
|
||||||
|
|
||||||
|
if (enableSway) {
|
||||||
|
// Random sway duration
|
||||||
|
const swayDuration = Math.random() * 2 + 3; // 3-5s per cycle
|
||||||
|
swayWrapper.style.animationDuration = `${swayDuration}s`;
|
||||||
|
swayWrapper.style.animationDelay = `-${Math.random() * 5}s`;
|
||||||
|
|
||||||
|
// Random sway amplitude (using CSS variable for dynamic keyframe)
|
||||||
|
// MARK: SWAY DISTANCE RANGE (in px)
|
||||||
|
const swayAmount = Math.random() * 70 + 30; // 30-100px
|
||||||
|
const direction = Math.random() > 0.5 ? 1 : -1;
|
||||||
|
swayWrapper.style.setProperty('--sway-amount', `${swayAmount * direction}px`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flutter speed and random 3D rotation axis
|
||||||
|
// MARK: CONFETTI FLUTTER ROTATION SPEED
|
||||||
|
confetti.style.animationDuration = `${Math.random() * 2 + 1}s`;
|
||||||
|
confetti.style.setProperty('--rx', Math.random().toFixed(2));
|
||||||
|
confetti.style.setProperty('--ry', Math.random().toFixed(2));
|
||||||
|
confetti.style.setProperty('--rz', (Math.random() * 0.5).toFixed(2));
|
||||||
|
|
||||||
|
// Random direction for 3D rotation
|
||||||
|
const rotDir = Math.random() > 0.5 ? 1 : -1;
|
||||||
|
confetti.style.setProperty('--rot-dir', `${rotDir * 360}deg`);
|
||||||
|
|
||||||
|
if (enableSway) {
|
||||||
|
swayWrapper.appendChild(confetti);
|
||||||
|
wrapper.appendChild(swayWrapper);
|
||||||
|
} else {
|
||||||
|
wrapper.appendChild(confetti);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Respawn confetti when it hits the bottom
|
||||||
|
wrapper.addEventListener('animationend', (e) => {
|
||||||
|
if (e.animationName === 'carnival-fall') {
|
||||||
|
wrapper.remove();
|
||||||
|
createConfettiPiece(container, false); // respawn without initial huge delay
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
function addRandomCarnivalObjects(count) {
|
||||||
|
const carnivalContainer = document.querySelector('.carnival-container');
|
||||||
|
if (!carnivalContainer) return;
|
||||||
|
|
||||||
|
console.log('Adding random carnival confetti');
|
||||||
|
|
||||||
|
for (let i = 0; i < count; i++) {
|
||||||
|
createConfettiPiece(carnivalContainer, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// initialize standard carnival objects
|
||||||
|
function initCarnivalObjects() {
|
||||||
|
let container = document.querySelector('.carnival-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement("div");
|
||||||
|
container.className = "carnival-container";
|
||||||
|
container.setAttribute("aria-hidden", "true");
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initial confetti
|
||||||
|
for (let i = 0; i < 60; i++) {
|
||||||
|
createConfettiPiece(container, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// initialize carnival
|
||||||
|
function initializeCarnival() {
|
||||||
|
if (!carnival) return; // exit if carnival is disabled
|
||||||
|
initCarnivalObjects();
|
||||||
|
toggleCarnival();
|
||||||
|
|
||||||
|
const screenWidth = window.innerWidth;
|
||||||
|
if (randomCarnival && (screenWidth > 768 || randomCarnivalMobile)) {
|
||||||
|
addRandomCarnivalObjects(carnivalCount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeCarnival();
|
||||||
58
Jellyfin.Plugin.Seasonals/Web/cherryblossom.css
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
.cherryblossom-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Petals */
|
||||||
|
.cherryblossom-petal {
|
||||||
|
position: fixed;
|
||||||
|
top: -20px;
|
||||||
|
z-index: 1005;
|
||||||
|
width: 15px;
|
||||||
|
height: 10px;
|
||||||
|
background-color: #ffc0cb;
|
||||||
|
border-radius: 15px 0px 15px 0px;
|
||||||
|
|
||||||
|
will-change: transform, top;
|
||||||
|
animation-name: cherryblossom-fall, cherryblossom-sway;
|
||||||
|
animation-timing-function: linear, ease-in-out;
|
||||||
|
animation-iteration-count: infinite, infinite;
|
||||||
|
animation-duration: 10s, 3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cherryblossom-petal.lighter {
|
||||||
|
background-color: #ffd1dc;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cherryblossom-petal.darker {
|
||||||
|
background-color: #ffb7c5;
|
||||||
|
opacity: 0.9;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cherryblossom-petal.type2 {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 10px 0px 10px 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes cherryblossom-fall {
|
||||||
|
0% { top: -10%; }
|
||||||
|
100% { top: 100%; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes cherryblossom-sway {
|
||||||
|
0%, 100% {
|
||||||
|
transform: translateX(0) rotate(0deg);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: translateX(30px) rotate(45deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
101
Jellyfin.Plugin.Seasonals/Web/cherryblossom.js
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
const config = window.SeasonalsPluginConfig?.CherryBlossom || {};
|
||||||
|
|
||||||
|
const cherryBlossom = config.EnableCherryBlossom !== undefined ? config.EnableCherryBlossom : true;
|
||||||
|
const petalCount = config.PetalCount || 25;
|
||||||
|
const randomCherryBlossom = config.EnableRandomCherryBlossom !== undefined ? config.EnableRandomCherryBlossom : true;
|
||||||
|
const randomCherryBlossomMobile = config.EnableRandomCherryBlossomMobile !== undefined ? config.EnableRandomCherryBlossomMobile : false;
|
||||||
|
const enableDifferentDuration = config.EnableDifferentDuration !== undefined ? config.EnableDifferentDuration : true;
|
||||||
|
|
||||||
|
let msgPrinted = false;
|
||||||
|
|
||||||
|
function toggleCherryBlossom() {
|
||||||
|
const container = document.querySelector('.cherryblossom-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('CherryBlossom hidden');
|
||||||
|
msgPrinted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
container.style.display = 'block';
|
||||||
|
if (msgPrinted) {
|
||||||
|
console.log('CherryBlossom visible');
|
||||||
|
msgPrinted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const observer = new MutationObserver(toggleCherryBlossom);
|
||||||
|
observer.observe(document.body, { childList: true, subtree: true, attributes: true });
|
||||||
|
|
||||||
|
function createPetal(container) {
|
||||||
|
const petal = document.createElement('div');
|
||||||
|
petal.classList.add('cherryblossom-petal');
|
||||||
|
|
||||||
|
const type = Math.random() > 0.5 ? 'type1' : 'type2';
|
||||||
|
petal.classList.add(type);
|
||||||
|
|
||||||
|
const color = Math.random() > 0.7 ? 'darker' : 'lighter';
|
||||||
|
petal.classList.add(color);
|
||||||
|
|
||||||
|
const randomLeft = Math.random() * 100;
|
||||||
|
petal.style.left = `${randomLeft}%`;
|
||||||
|
|
||||||
|
const size = Math.random() * 0.5 + 0.5;
|
||||||
|
petal.style.transform = `scale(${size})`;
|
||||||
|
|
||||||
|
const duration = Math.random() * 5 + 8;
|
||||||
|
const delay = Math.random() * 10;
|
||||||
|
const swayDuration = Math.random() * 2 + 2;
|
||||||
|
|
||||||
|
if (enableDifferentDuration) {
|
||||||
|
petal.style.animationDuration = `${duration}s, ${swayDuration}s`;
|
||||||
|
}
|
||||||
|
petal.style.animationDelay = `${delay}s, ${Math.random() * 3}s`;
|
||||||
|
|
||||||
|
container.appendChild(petal);
|
||||||
|
}
|
||||||
|
|
||||||
|
function addRandomObjects() {
|
||||||
|
const container = document.querySelector('.cherryblossom-container');
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
for (let i = 0; i < petalCount; i++) {
|
||||||
|
createPetal(container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initObjects() {
|
||||||
|
let container = document.querySelector('.cherryblossom-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement("div");
|
||||||
|
container.className = "cherryblossom-container";
|
||||||
|
container.setAttribute("aria-hidden", "true");
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initial batch
|
||||||
|
for (let i = 0; i < 15; i++) {
|
||||||
|
createPetal(container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeCherryBlossom() {
|
||||||
|
if (!cherryBlossom) return;
|
||||||
|
initObjects();
|
||||||
|
toggleCherryBlossom();
|
||||||
|
|
||||||
|
const screenWidth = window.innerWidth;
|
||||||
|
if (randomCherryBlossom && (screenWidth > 768 || randomCherryBlossomMobile)) {
|
||||||
|
addRandomObjects();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeCherryBlossom();
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
.ressurection-container {
|
.resurrection-container {
|
||||||
display: block;
|
display: block;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
@@ -10,20 +10,20 @@
|
|||||||
z-index: 10;
|
z-index: 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ressurection-symbol {
|
.resurrection-symbol {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
z-index: 15;
|
z-index: 15;
|
||||||
top: -15%;
|
top: -15%;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
cursor: default;
|
cursor: default;
|
||||||
animation-name: ressurection-fall, ressurection-sway;
|
animation-name: resurrection-fall, resurrection-sway;
|
||||||
animation-timing-function: linear, ease-in-out;
|
animation-timing-function: linear, ease-in-out;
|
||||||
animation-iteration-count: infinite, infinite;
|
animation-iteration-count: infinite, infinite;
|
||||||
will-change: transform, top;
|
will-change: transform, top;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ressurection-symbol img {
|
.resurrection-symbol img {
|
||||||
z-index: 15;
|
z-index: 15;
|
||||||
height: auto;
|
height: auto;
|
||||||
width: 56px;
|
width: 56px;
|
||||||
@@ -32,12 +32,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 768px) {
|
||||||
.ressurection-symbol img {
|
.resurrection-symbol img {
|
||||||
width: 42px;
|
width: 42px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@keyframes ressurection-fall {
|
@keyframes resurrection-fall {
|
||||||
0% {
|
0% {
|
||||||
top: -15%;
|
top: -15%;
|
||||||
}
|
}
|
||||||
@@ -47,7 +47,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@keyframes ressurection-sway {
|
@keyframes resurrection-sway {
|
||||||
0%,
|
0%,
|
||||||
100% {
|
100% {
|
||||||
transform: translateX(0);
|
transform: translateX(0);
|
||||||
|
|||||||
@@ -68,6 +68,16 @@ const ThemeConfigs = {
|
|||||||
js: '../Seasonals/Resources/spring.js',
|
js: '../Seasonals/Resources/spring.js',
|
||||||
containerClass: 'spring-container'
|
containerClass: 'spring-container'
|
||||||
},
|
},
|
||||||
|
carnival: {
|
||||||
|
css: '../Seasonals/Resources/carnival.css',
|
||||||
|
js: '../Seasonals/Resources/carnival.js',
|
||||||
|
containerClass: 'carnival-container'
|
||||||
|
},
|
||||||
|
cherryblossom: {
|
||||||
|
css: '../Seasonals/Resources/cherryblossom.css',
|
||||||
|
js: '../Seasonals/Resources/cherryblossom.js',
|
||||||
|
containerClass: 'cherryblossom-container'
|
||||||
|
},
|
||||||
none: {
|
none: {
|
||||||
containerClass: 'none'
|
containerClass: 'none'
|
||||||
},
|
},
|
||||||
@@ -241,6 +251,12 @@ const SeasonalsManager = {
|
|||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
this.config = await response.json();
|
this.config = await response.json();
|
||||||
window.SeasonalsPluginConfig = this.config;
|
window.SeasonalsPluginConfig = this.config;
|
||||||
|
|
||||||
|
if (this.config.IsEnabled === false) {
|
||||||
|
console.log('Seasonals: Plugin is disabled globally.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
console.log('Seasonals: Seasonals Config loaded:', this.config);
|
console.log('Seasonals: Seasonals Config loaded:', this.config);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
|||||||
212
Jellyfin.Plugin.Seasonals/Web/spring.css
Normal file
@@ -0,0 +1,212 @@
|
|||||||
|
.spring-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Pollen */
|
||||||
|
.spring-pollen {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 14;
|
||||||
|
background-color: #fffacd;
|
||||||
|
border-radius: 50%;
|
||||||
|
opacity: 0.6;
|
||||||
|
box-shadow: 0 0 4px rgba(255, 250, 205, 0.4);
|
||||||
|
|
||||||
|
will-change: transform;
|
||||||
|
animation-name: spring-float;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Sunbeams */
|
||||||
|
.spring-sunbeam {
|
||||||
|
position: fixed;
|
||||||
|
top: -50%;
|
||||||
|
height: 200%;
|
||||||
|
background: linear-gradient(to bottom, rgba(255, 255, 255, 0), rgba(255, 255, 200, 0.08) 50%, rgba(255, 255, 255, 0));
|
||||||
|
z-index: 5;
|
||||||
|
transform-origin: top center;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Grass */
|
||||||
|
.spring-grass-container {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 80px;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spring-grass {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
width: 3px;
|
||||||
|
border-radius: 100% 0 0 0;
|
||||||
|
transform-origin: bottom center;
|
||||||
|
background-color: #4caf50;
|
||||||
|
|
||||||
|
will-change: transform;
|
||||||
|
animation-name: spring-grass-sway;
|
||||||
|
animation-timing-function: ease-in-out;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Birds */
|
||||||
|
.spring-bird {
|
||||||
|
position: static !important;
|
||||||
|
display: block;
|
||||||
|
z-index: 1001;
|
||||||
|
/* MARK: BIRD SIZE */
|
||||||
|
width: 80px;
|
||||||
|
height: auto;
|
||||||
|
will-change: transform;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Butterflies */
|
||||||
|
.spring-butterfly {
|
||||||
|
position: static !important;
|
||||||
|
display: block;
|
||||||
|
z-index: 1001;
|
||||||
|
/* MARK: BUTTERFLY SIZE */
|
||||||
|
width: 40px;
|
||||||
|
height: auto;
|
||||||
|
will-change: transform;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bee */
|
||||||
|
.spring-bee {
|
||||||
|
position: static !important;
|
||||||
|
display: block;
|
||||||
|
z-index: 1001;
|
||||||
|
/* MARK: BEE SIZE */
|
||||||
|
width: 30px;
|
||||||
|
height: auto;
|
||||||
|
will-change: transform;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ladybug */
|
||||||
|
.spring-ladybug-gif {
|
||||||
|
position: static !important;
|
||||||
|
display: block;
|
||||||
|
/* MARK: LADYBUG SIZE */
|
||||||
|
width: 30px;
|
||||||
|
height: auto;
|
||||||
|
will-change: transform;
|
||||||
|
animation-timing-function: linear;
|
||||||
|
animation-iteration-count: infinite;
|
||||||
|
--bug-rotation: -55deg;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spring-ladybug-wrapper {
|
||||||
|
z-index: 1002;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Generic Wrappers */
|
||||||
|
.spring-anim-wrapper {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 1001;
|
||||||
|
will-change: transform;
|
||||||
|
top: 0; left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spring-mirror-wrapper {
|
||||||
|
width: 100%; height: 100%;
|
||||||
|
will-change: transform;
|
||||||
|
transform-origin: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@keyframes spring-float {
|
||||||
|
0% { transform: translateX(0) translateY(0); }
|
||||||
|
25% { transform: translateX(20px) translateY(-10px); }
|
||||||
|
50% { transform: translateX(40px) translateY(0); }
|
||||||
|
75% { transform: translateX(20px) translateY(10px); }
|
||||||
|
100% { transform: translateX(0) translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spring-beam-pulse {
|
||||||
|
0% { opacity: 0; transform: rotate(var(--beam-rotation, 45deg)) scaleX(0.8); }
|
||||||
|
50% { opacity: 0.6; transform: rotate(var(--beam-rotation, 45deg)) scaleX(1.2); }
|
||||||
|
100% { opacity: 0; transform: rotate(var(--beam-rotation, 45deg)) scaleX(0.8); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spring-grass-sway {
|
||||||
|
0% { transform: rotate(0deg); }
|
||||||
|
50% { transform: rotate(8deg); }
|
||||||
|
100% { transform: rotate(0deg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wrapper animations (Flight across screen) */
|
||||||
|
@keyframes spring-fly-right-wrapper {
|
||||||
|
0% { transform: translateX(-10vw); }
|
||||||
|
100% { transform: translateX(110vw); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spring-fly-left-wrapper {
|
||||||
|
0% { transform: translateX(110vw); }
|
||||||
|
100% { transform: translateX(-10vw); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Vertical Drift for Sloped Flight */
|
||||||
|
@keyframes spring-vertical-drift {
|
||||||
|
0% { top: var(--start-y, 10%); }
|
||||||
|
100% { top: var(--end-y, 10%); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Inner animations (Bobbing/Fluttering) */
|
||||||
|
@keyframes spring-bob {
|
||||||
|
0% { transform: translateY(0); }
|
||||||
|
50% { transform: translateY(-20px); }
|
||||||
|
100% { transform: translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spring-flutter {
|
||||||
|
0% { transform: translateY(0) rotate(0deg); }
|
||||||
|
25% { transform: translateY(-5px) rotate(5deg); }
|
||||||
|
50% { transform: translateY(0) rotate(0deg); }
|
||||||
|
75% { transform: translateY(5px) rotate(-5deg); }
|
||||||
|
100% { transform: translateY(0) rotate(0deg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bee Buzz - Reduced Intensity */
|
||||||
|
@keyframes spring-buzz {
|
||||||
|
0% { transform: translate(0, 0); }
|
||||||
|
25% { transform: translate(2px, -2px); }
|
||||||
|
50% { transform: translate(0, 2px); }
|
||||||
|
75% { transform: translate(-2px, -2px); }
|
||||||
|
100% { transform: translate(0, 0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ladybug Walk (Wrapper handles X) */
|
||||||
|
@keyframes spring-walk-right {
|
||||||
|
0% { transform: translateX(-10vw); }
|
||||||
|
100% { transform: translateX(110vw); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spring-walk-left {
|
||||||
|
0% { transform: translateX(110vw); }
|
||||||
|
100% { transform: translateX(-10vw); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ladybug Crawl (Inner Wobble) */
|
||||||
|
@keyframes spring-crawl {
|
||||||
|
0% { transform: translateY(0) rotate(var(--bug-rotation, 0deg)); }
|
||||||
|
25% { transform: translateY(-3px) rotate(calc(var(--bug-rotation, 0deg) + 8deg)); }
|
||||||
|
50% { transform: translateY(0) rotate(var(--bug-rotation, 0deg)); }
|
||||||
|
75% { transform: translateY(-3px) rotate(calc(var(--bug-rotation, 0deg) - 8deg)); }
|
||||||
|
100% { transform: translateY(0) rotate(var(--bug-rotation, 0deg)); }
|
||||||
|
}
|
||||||
396
Jellyfin.Plugin.Seasonals/Web/spring.js
Normal file
@@ -0,0 +1,396 @@
|
|||||||
|
const config = window.SeasonalsPluginConfig?.Spring || {};
|
||||||
|
|
||||||
|
const spring = config.EnableSpring !== undefined ? config.EnableSpring : true; // Enable/disable spring
|
||||||
|
const pollenCount = config.PollenCount || 30; // Number of pollen particles
|
||||||
|
const sunbeamCount = config.SunbeamCount || 5; // Number of sunbeams
|
||||||
|
const enableSunbeams = config.EnableSpringSunbeams !== undefined ? config.EnableSpringSunbeams : true; // Enable/disable sunbeams
|
||||||
|
const birdCount = config.BirdCount !== undefined ? config.BirdCount : 3; // Number of birds
|
||||||
|
const butterflyCount = config.ButterflyCount !== undefined ? config.ButterflyCount : 4; // Number of butterflies
|
||||||
|
const beeCount = config.BeeCount !== undefined ? config.BeeCount : 2; // Number of bees
|
||||||
|
const ladybugCount = config.LadybugCount !== undefined ? config.LadybugCount : 2; // Number of ladybugs
|
||||||
|
const randomSpring = config.EnableRandomSpring !== undefined ? config.EnableRandomSpring : true; // Enable random spring objects
|
||||||
|
|
||||||
|
const birdImages = [
|
||||||
|
'../Seasonals/Resources/spring_assets/Bird_1.gif',
|
||||||
|
'../Seasonals/Resources/spring_assets/Bird_2.gif',
|
||||||
|
'../Seasonals/Resources/spring_assets/Bird_3.gif'
|
||||||
|
];
|
||||||
|
|
||||||
|
const butterflyImages = [
|
||||||
|
'../Seasonals/Resources/spring_assets/Butterfly_1.gif',
|
||||||
|
'../Seasonals/Resources/spring_assets/Butterfly_2.gif'
|
||||||
|
];
|
||||||
|
|
||||||
|
const beeImage = '../Seasonals/Resources/spring_assets/Bee.gif';
|
||||||
|
const ladybugImage = '../Seasonals/Resources/spring_assets/ladybug.gif';
|
||||||
|
|
||||||
|
let msgPrinted = false;
|
||||||
|
|
||||||
|
function toggleSpring() {
|
||||||
|
const springContainer = document.querySelector('.spring-container');
|
||||||
|
if (!springContainer) 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) {
|
||||||
|
springContainer.style.display = 'none';
|
||||||
|
if (!msgPrinted) {
|
||||||
|
console.log('Spring hidden');
|
||||||
|
msgPrinted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
springContainer.style.display = 'block';
|
||||||
|
if (msgPrinted) {
|
||||||
|
console.log('Spring visible');
|
||||||
|
msgPrinted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const observer = new MutationObserver(toggleSpring);
|
||||||
|
observer.observe(document.body, { childList: true, subtree: true, attributes: true });
|
||||||
|
|
||||||
|
|
||||||
|
function createPollen(container) {
|
||||||
|
const pollen = document.createElement('div');
|
||||||
|
pollen.classList.add('spring-pollen');
|
||||||
|
|
||||||
|
// MARK: POLLEN START VERTICAL POSITION (in %)
|
||||||
|
const startY = Math.random() * 60 + 20;
|
||||||
|
pollen.style.top = `${startY}%`;
|
||||||
|
pollen.style.left = `${Math.random() * 100}%`;
|
||||||
|
|
||||||
|
// MARK: POLLEN SIZE
|
||||||
|
const size = Math.random() * 3 + 1; // 1-4px
|
||||||
|
pollen.style.width = `${size}px`;
|
||||||
|
pollen.style.height = `${size}px`;
|
||||||
|
|
||||||
|
const duration = Math.random() * 20 + 20;
|
||||||
|
pollen.style.animationDuration = `${duration}s`;
|
||||||
|
pollen.style.animationDelay = `-${Math.random() * 20}s`;
|
||||||
|
|
||||||
|
container.appendChild(pollen);
|
||||||
|
}
|
||||||
|
|
||||||
|
function spawnSunbeamGroup(container, count) {
|
||||||
|
if (!enableSunbeams) return;
|
||||||
|
|
||||||
|
const rotate = Math.random() * 30 - 15 + 45;
|
||||||
|
let beamsActive = count;
|
||||||
|
|
||||||
|
for (let i = 0; i < count; i++) {
|
||||||
|
const beam = document.createElement('div');
|
||||||
|
beam.classList.add('spring-sunbeam');
|
||||||
|
|
||||||
|
const left = Math.random() * 100;
|
||||||
|
beam.style.left = `${left}%`;
|
||||||
|
|
||||||
|
// MARK: SUNBEAM WIDTH (in px)
|
||||||
|
const width = Math.random() * 12 + 8; // 8-20px wide
|
||||||
|
beam.style.width = `${width}px`;
|
||||||
|
|
||||||
|
beam.style.setProperty('--beam-rotation', `${rotate}deg`);
|
||||||
|
|
||||||
|
const duration = Math.random() * 7 + 8; // 8-15s
|
||||||
|
beam.style.animation = `spring-beam-pulse ${duration}s ease-in-out forwards`;
|
||||||
|
|
||||||
|
beam.style.animationDelay = `${Math.random() * 3}s`;
|
||||||
|
|
||||||
|
beam.addEventListener('animationend', () => {
|
||||||
|
beam.remove();
|
||||||
|
beamsActive--;
|
||||||
|
if (beamsActive === 0) {
|
||||||
|
spawnSunbeamGroup(container, count);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
container.appendChild(beam);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function createGrass(container) {
|
||||||
|
let grassContainer = container.querySelector('.spring-grass-container');
|
||||||
|
if (!grassContainer) {
|
||||||
|
grassContainer = document.createElement('div');
|
||||||
|
grassContainer.className = 'spring-grass-container';
|
||||||
|
container.appendChild(grassContainer);
|
||||||
|
}
|
||||||
|
|
||||||
|
grassContainer.innerHTML = '';
|
||||||
|
|
||||||
|
const bladeCount = window.innerWidth / 3;
|
||||||
|
for (let i = 0; i < bladeCount; i++) {
|
||||||
|
const blade = document.createElement('div');
|
||||||
|
blade.classList.add('spring-grass');
|
||||||
|
|
||||||
|
// MARK: GRASS HEIGHT
|
||||||
|
const height = Math.random() * 40 + 20; // 20-60px height
|
||||||
|
blade.style.height = `${height}px`;
|
||||||
|
blade.style.left = `${i * 3 + Math.random() * 2}px`;
|
||||||
|
|
||||||
|
const duration = Math.random() * 2 + 3;
|
||||||
|
blade.style.animationDuration = `${duration}s`;
|
||||||
|
blade.style.animationDelay = `-${Math.random() * 5}s`;
|
||||||
|
|
||||||
|
const hue = 100 + Math.random() * 40;
|
||||||
|
blade.style.backgroundColor = `hsl(${hue}, 60%, 40%)`;
|
||||||
|
|
||||||
|
// Random z-index to interleave with Ladybug (1002)
|
||||||
|
// Values: 1001 (behind) or 1003 (front)
|
||||||
|
const z = Math.random() > 0.5 ? 1001 : 1003;
|
||||||
|
blade.style.zIndex = z;
|
||||||
|
|
||||||
|
grassContainer.appendChild(blade);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initSpringObjects() {
|
||||||
|
let container = document.querySelector('.spring-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement("div");
|
||||||
|
container.className = "spring-container";
|
||||||
|
container.setAttribute("aria-hidden", "true");
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
createGrass(container);
|
||||||
|
|
||||||
|
if (enableSunbeams) {
|
||||||
|
spawnSunbeamGroup(container, sunbeamCount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeSpring() {
|
||||||
|
if (!spring) {
|
||||||
|
console.warn('Spring is disabled.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
initSpringObjects();
|
||||||
|
toggleSpring();
|
||||||
|
|
||||||
|
const container = document.querySelector('.spring-container');
|
||||||
|
if (container) {
|
||||||
|
if (randomSpring) {
|
||||||
|
// Add Pollen
|
||||||
|
for (let i = 0; i < pollenCount; i++) {
|
||||||
|
createPollen(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add Birds
|
||||||
|
for (let i = 0; i < birdCount; i++) {
|
||||||
|
setTimeout(() => createBird(container), Math.random() * 1000); // 0-1s desync
|
||||||
|
}
|
||||||
|
// Add Butterflies
|
||||||
|
for (let i = 0; i < butterflyCount; i++) {
|
||||||
|
setTimeout(() => createButterfly(container), Math.random() * 1000); // 0-1s desync
|
||||||
|
}
|
||||||
|
// Add Bees
|
||||||
|
for (let i = 0; i < beeCount; i++) {
|
||||||
|
setTimeout(() => createBee(container), Math.random() * 1000); // 0-1s desync
|
||||||
|
}
|
||||||
|
// Add Ladybugs
|
||||||
|
for (let i = 0; i < ladybugCount; i++) {
|
||||||
|
setTimeout(() => createLadybugGif(container), Math.random() * 1000); // 0-1s desync
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function createBird(container) {
|
||||||
|
const wrapper = document.createElement('div');
|
||||||
|
wrapper.classList.add('spring-anim-wrapper');
|
||||||
|
wrapper.classList.add('spring-bird-wrapper');
|
||||||
|
|
||||||
|
const mirror = document.createElement('div');
|
||||||
|
mirror.classList.add('spring-mirror-wrapper');
|
||||||
|
|
||||||
|
const bird = document.createElement('img');
|
||||||
|
bird.classList.add('spring-bird');
|
||||||
|
|
||||||
|
const randomSrc = birdImages[Math.floor(Math.random() * birdImages.length)];
|
||||||
|
bird.src = randomSrc;
|
||||||
|
|
||||||
|
const direction = Math.random() > 0.5 ? 'right' : 'left';
|
||||||
|
// MARK: BIRD SPEED (10-15s)
|
||||||
|
const duration = Math.random() * 5 + 10;
|
||||||
|
|
||||||
|
// MARK: BIRD HEIGHT RANGE (in %)
|
||||||
|
const startY = Math.random() * 55 + 5; // Start 5-60%
|
||||||
|
const endY = Math.random() * 55 + 5; // End 5-60%
|
||||||
|
wrapper.style.setProperty('--start-y', `${startY}%`);
|
||||||
|
wrapper.style.setProperty('--end-y', `${endY}%`);
|
||||||
|
|
||||||
|
if (direction === 'right') {
|
||||||
|
wrapper.style.animation = `spring-fly-right-wrapper ${duration}s linear forwards, spring-vertical-drift ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(-1)';
|
||||||
|
} else {
|
||||||
|
wrapper.style.animation = `spring-fly-left-wrapper ${duration}s linear forwards, spring-vertical-drift ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(1)';
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.addEventListener('animationend', (e) => {
|
||||||
|
if (e.animationName.includes('fly-')) {
|
||||||
|
wrapper.remove();
|
||||||
|
createBird(container);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
bird.style.animation = `spring-bob 2s ease-in-out infinite`;
|
||||||
|
|
||||||
|
wrapper.style.top = `${startY}%`;
|
||||||
|
|
||||||
|
mirror.appendChild(bird);
|
||||||
|
wrapper.appendChild(mirror);
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
function createButterfly(container) {
|
||||||
|
const wrapper = document.createElement('div');
|
||||||
|
wrapper.classList.add('spring-anim-wrapper');
|
||||||
|
wrapper.classList.add('spring-butterfly-wrapper');
|
||||||
|
|
||||||
|
const mirror = document.createElement('div');
|
||||||
|
mirror.classList.add('spring-mirror-wrapper');
|
||||||
|
|
||||||
|
const butterfly = document.createElement('img');
|
||||||
|
butterfly.classList.add('spring-butterfly');
|
||||||
|
|
||||||
|
const randomSrc = butterflyImages[Math.floor(Math.random() * butterflyImages.length)];
|
||||||
|
butterfly.src = randomSrc;
|
||||||
|
|
||||||
|
const duration = Math.random() * 15 + 25;
|
||||||
|
const direction = Math.random() > 0.5 ? 'right' : 'left';
|
||||||
|
|
||||||
|
if (direction === 'right') {
|
||||||
|
wrapper.style.animation = `spring-fly-right-wrapper ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(1)';
|
||||||
|
} else {
|
||||||
|
wrapper.style.animation = `spring-fly-left-wrapper ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(-1)';
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.addEventListener('animationend', (e) => {
|
||||||
|
if (e.animationName.includes('fly-')) {
|
||||||
|
wrapper.remove();
|
||||||
|
createButterfly(container);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// MARK: BUTTERFLY FLUTTER RHYTHM
|
||||||
|
butterfly.style.animation = `spring-flutter 3s ease-in-out infinite`;
|
||||||
|
butterfly.style.animationDelay = `-${Math.random() * 3}s`;
|
||||||
|
|
||||||
|
// MARK: BUTTERFLY HEIGHT (in %)
|
||||||
|
const top = Math.random() * 35 + 30; // 30-65%
|
||||||
|
wrapper.style.top = `${top}%`;
|
||||||
|
|
||||||
|
mirror.appendChild(butterfly);
|
||||||
|
wrapper.appendChild(mirror);
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
function createBee(container) {
|
||||||
|
const wrapper = document.createElement('div');
|
||||||
|
wrapper.classList.add('spring-anim-wrapper');
|
||||||
|
wrapper.classList.add('spring-bee-wrapper');
|
||||||
|
|
||||||
|
const mirror = document.createElement('div');
|
||||||
|
mirror.classList.add('spring-mirror-wrapper');
|
||||||
|
|
||||||
|
const bee = document.createElement('img');
|
||||||
|
bee.classList.add('spring-bee');
|
||||||
|
bee.src = beeImage;
|
||||||
|
|
||||||
|
const duration = Math.random() * 10 + 15;
|
||||||
|
const direction = Math.random() > 0.5 ? 'right' : 'left';
|
||||||
|
|
||||||
|
if (direction === 'right') {
|
||||||
|
wrapper.style.animation = `spring-fly-right-wrapper ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(1)';
|
||||||
|
} else {
|
||||||
|
wrapper.style.animation = `spring-fly-left-wrapper ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(-1)';
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.addEventListener('animationend', (e) => {
|
||||||
|
if (e.animationName.includes('fly-')) {
|
||||||
|
wrapper.remove();
|
||||||
|
createBee(container);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// MARK: BEE HEIGHT (in %)
|
||||||
|
const top = Math.random() * 60 + 20; // 20-80%
|
||||||
|
wrapper.style.top = `${top}%`;
|
||||||
|
|
||||||
|
mirror.appendChild(bee);
|
||||||
|
wrapper.appendChild(mirror);
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
function createLadybugGif(container) {
|
||||||
|
const wrapper = document.createElement('div');
|
||||||
|
wrapper.classList.add('spring-anim-wrapper');
|
||||||
|
wrapper.classList.add('spring-ladybug-wrapper');
|
||||||
|
|
||||||
|
const mirror = document.createElement('div');
|
||||||
|
mirror.classList.add('spring-mirror-wrapper');
|
||||||
|
|
||||||
|
const bug = document.createElement('img');
|
||||||
|
bug.classList.add('spring-ladybug-gif');
|
||||||
|
bug.src = ladybugImage;
|
||||||
|
|
||||||
|
const direction = Math.random() > 0.5 ? 'right' : 'left';
|
||||||
|
const duration = Math.random() * 20 + 30;
|
||||||
|
|
||||||
|
if (direction === 'right') {
|
||||||
|
wrapper.style.animation = `spring-walk-right ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(1)';
|
||||||
|
} else {
|
||||||
|
wrapper.style.animation = `spring-walk-left ${duration}s linear forwards`;
|
||||||
|
mirror.style.transform = 'scaleX(-1)';
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.addEventListener('animationend', (e) => {
|
||||||
|
if (e.animationName.includes('walk-')) {
|
||||||
|
wrapper.remove();
|
||||||
|
createLadybugGif(container);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
bug.style.animation = `spring-crawl 2s ease-in-out infinite`;
|
||||||
|
|
||||||
|
wrapper.style.top = 'auto';
|
||||||
|
wrapper.style.bottom = '5px';
|
||||||
|
|
||||||
|
mirror.appendChild(bug);
|
||||||
|
wrapper.appendChild(mirror);
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
function debounce(func, wait) {
|
||||||
|
let timeout;
|
||||||
|
return function executedFunction(...args) {
|
||||||
|
const later = () => {
|
||||||
|
clearTimeout(timeout);
|
||||||
|
func(...args);
|
||||||
|
};
|
||||||
|
clearTimeout(timeout);
|
||||||
|
timeout = setTimeout(later, wait);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleResize = debounce(() => {
|
||||||
|
const container = document.querySelector('.spring-container');
|
||||||
|
if (container) {
|
||||||
|
createGrass(container);
|
||||||
|
}
|
||||||
|
}, 250);
|
||||||
|
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
|
|
||||||
|
initializeSpring();
|
||||||
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Bee.gif
Normal file
|
After Width: | Height: | Size: 132 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Bird_1.gif
Normal file
|
After Width: | Height: | Size: 135 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Bird_2.gif
Normal file
|
After Width: | Height: | Size: 133 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Bird_3.gif
Normal file
|
After Width: | Height: | Size: 135 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Butterfly_1.gif
Normal file
|
After Width: | Height: | Size: 502 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Butterfly_2.gif
Normal file
|
After Width: | Height: | Size: 514 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/Rotkehlchen.gif
Normal file
|
After Width: | Height: | Size: 208 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/ladybug.gif
Normal file
|
After Width: | Height: | Size: 74 KiB |
BIN
Jellyfin.Plugin.Seasonals/Web/spring_assets/wasp.gif
Normal file
|
After Width: | Height: | Size: 4.3 KiB |
80
Jellyfin.Plugin.Seasonals/Web/summer.css
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
.summer-container {
|
||||||
|
display: block;
|
||||||
|
position: fixed;
|
||||||
|
overflow: hidden;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
.summer-bubble {
|
||||||
|
position: fixed;
|
||||||
|
bottom: -50px;
|
||||||
|
z-index: 15;
|
||||||
|
background: radial-gradient(circle at 30% 30%, rgba(255, 255, 255, 0.8), rgba(255, 255, 255, 0.1) 60%, rgba(255, 255, 255, 0.05));
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.3);
|
||||||
|
border-radius: 50%;
|
||||||
|
box-shadow: 0 0 10px rgba(255, 255, 255, 0.2);
|
||||||
|
|
||||||
|
will-change: transform, bottom;
|
||||||
|
animation-name: summer-rise, summer-wobble;
|
||||||
|
animation-timing-function: linear, ease-in-out;
|
||||||
|
animation-iteration-count: infinite, infinite;
|
||||||
|
animation-duration: 10s, 3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.summer-dust {
|
||||||
|
position: fixed;
|
||||||
|
bottom: -10px;
|
||||||
|
z-index: 12;
|
||||||
|
background-color: rgba(255, 223, 186, 0.6);
|
||||||
|
border-radius: 50%;
|
||||||
|
box-shadow: 0 0 5px rgba(255, 223, 186, 0.4);
|
||||||
|
|
||||||
|
will-change: transform, bottom;
|
||||||
|
animation-name: summer-rise, summer-drift;
|
||||||
|
animation-timing-function: linear, ease-in-out;
|
||||||
|
animation-iteration-count: infinite, infinite;
|
||||||
|
animation-duration: 20s, 5s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes summer-rise {
|
||||||
|
0% {
|
||||||
|
bottom: -10%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
bottom: 110%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes summer-wobble {
|
||||||
|
0% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
33% {
|
||||||
|
transform: translateX(15px);
|
||||||
|
}
|
||||||
|
66% {
|
||||||
|
transform: translateX(-15px);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes summer-drift {
|
||||||
|
0% {
|
||||||
|
transform: translateX(0) translateY(0);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: translateX(30px) translateY(-20px);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: translateX(0) translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
147
Jellyfin.Plugin.Seasonals/Web/summer.js
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
const config = window.SeasonalsPluginConfig?.Summer || {};
|
||||||
|
|
||||||
|
const summer = config.EnableSummer !== undefined ? config.EnableSummer : true; // Enable/disable summer theme
|
||||||
|
const bubbleCount = config.BubbleCount || 30; // Number of bubbles
|
||||||
|
const dustCount = config.DustCount || 50; // Number of dust particles
|
||||||
|
const randomSummer = config.EnableRandomSummer !== undefined ? config.EnableRandomSummer : true; // Enable random generating objects
|
||||||
|
const randomSummerMobile = config.EnableRandomSummerMobile !== undefined ? config.EnableRandomSummerMobile : false; // Enable random generating objects on mobile
|
||||||
|
const enableDifferentDuration = config.EnableDifferentDuration !== undefined ? config.EnableDifferentDuration : true; // Randomize animation duration of bubbles and dust
|
||||||
|
|
||||||
|
let msgPrinted = false;
|
||||||
|
|
||||||
|
function toggleSummer() {
|
||||||
|
const summerContainer = document.querySelector('.summer-container');
|
||||||
|
if (!summerContainer) 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) {
|
||||||
|
summerContainer.style.display = 'none';
|
||||||
|
if (!msgPrinted) {
|
||||||
|
console.log('Summer hidden');
|
||||||
|
msgPrinted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
summerContainer.style.display = 'block';
|
||||||
|
if (msgPrinted) {
|
||||||
|
console.log('Summer visible');
|
||||||
|
msgPrinted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const observer = new MutationObserver(toggleSummer);
|
||||||
|
observer.observe(document.body, { childList: true, subtree: true, attributes: true });
|
||||||
|
|
||||||
|
function createBubble(container, isDust = false) {
|
||||||
|
const bubble = document.createElement('div');
|
||||||
|
|
||||||
|
if (isDust) {
|
||||||
|
bubble.classList.add('summer-dust');
|
||||||
|
} else {
|
||||||
|
bubble.classList.add('summer-bubble');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Random horizontal position
|
||||||
|
const randomLeft = Math.random() * 100;
|
||||||
|
bubble.style.left = `${randomLeft}%`;
|
||||||
|
|
||||||
|
// Random size
|
||||||
|
if (!isDust) {
|
||||||
|
// MARK: BUBBLE SIZE
|
||||||
|
const size = Math.random() * 20 + 10; // 10-30px bubbles
|
||||||
|
bubble.style.width = `${size}px`;
|
||||||
|
bubble.style.height = `${size}px`;
|
||||||
|
} else {
|
||||||
|
// MARK: DUST SIZE
|
||||||
|
const size = Math.random() * 3 + 1; // 1-4px dust
|
||||||
|
bubble.style.width = `${size}px`;
|
||||||
|
bubble.style.height = `${size}px`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Animation properties
|
||||||
|
const duration = isDust ? (Math.random() * 20 + 10) : (Math.random() * 10 + 5); // Dust is slower
|
||||||
|
const delay = Math.random() * 10;
|
||||||
|
|
||||||
|
if (enableDifferentDuration) {
|
||||||
|
bubble.style.animationDuration = `${duration}s`;
|
||||||
|
}
|
||||||
|
bubble.style.animationDelay = `${delay}s`;
|
||||||
|
|
||||||
|
container.appendChild(bubble);
|
||||||
|
}
|
||||||
|
|
||||||
|
function addRandomSummerObjects() {
|
||||||
|
const container = document.querySelector('.summer-container');
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
// Add bubbles
|
||||||
|
for (let i = 0; i < bubbleCount; i++) {
|
||||||
|
createBubble(container, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add some dust particles
|
||||||
|
for (let i = 0; i < dustCount; i++) {
|
||||||
|
createBubble(container, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initSummerObjects() {
|
||||||
|
let container = document.querySelector('.summer-container');
|
||||||
|
if (!container) {
|
||||||
|
container = document.createElement("div");
|
||||||
|
container.className = "summer-container";
|
||||||
|
container.setAttribute("aria-hidden", "true");
|
||||||
|
document.body.appendChild(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initial bubbles/dust
|
||||||
|
for (let i = 0; i < 10; i++) {
|
||||||
|
const bubble = document.createElement('div');
|
||||||
|
const isDust = Math.random() > 0.5;
|
||||||
|
if (isDust) {
|
||||||
|
bubble.classList.add('summer-dust');
|
||||||
|
} else {
|
||||||
|
bubble.classList.add('summer-bubble');
|
||||||
|
}
|
||||||
|
|
||||||
|
const randomLeft = Math.random() * 100;
|
||||||
|
bubble.style.left = `${randomLeft}%`;
|
||||||
|
|
||||||
|
if (!isDust) {
|
||||||
|
// MARK: BUBBLE SIZE
|
||||||
|
const size = Math.random() * 20 + 10;
|
||||||
|
bubble.style.width = `${size}px`;
|
||||||
|
bubble.style.height = `${size}px`;
|
||||||
|
} else {
|
||||||
|
// MARK: DUST SIZE
|
||||||
|
const size = Math.random() * 3 + 1;
|
||||||
|
bubble.style.width = `${size}px`;
|
||||||
|
bubble.style.height = `${size}px`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const duration = isDust ? (Math.random() * 20 + 10) : (Math.random() * 10 + 5);
|
||||||
|
if (enableDifferentDuration) {
|
||||||
|
bubble.style.animationDuration = `${duration}s`;
|
||||||
|
}
|
||||||
|
|
||||||
|
bubble.style.animationDelay = `-${Math.random() * 10}s`;
|
||||||
|
container.appendChild(bubble);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeSummer() {
|
||||||
|
if (!summer) return;
|
||||||
|
initSummerObjects();
|
||||||
|
toggleSummer();
|
||||||
|
|
||||||
|
const screenWidth = window.innerWidth;
|
||||||
|
if (randomSummer && (screenWidth > 768 || randomSummerMobile)) {
|
||||||
|
addRandomSummerObjects();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initializeSummer();
|
||||||
@@ -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>
|
|
||||||
508
Jellyfin.Plugin.Seasonals/Web/test-site.html
Normal file
@@ -0,0 +1,508 @@
|
|||||||
|
<!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="spring">Spring</option>
|
||||||
|
<option value="summer">Summer (Bubbles)</option>
|
||||||
|
<option value="carnival">Carnival (Confetti)</option>
|
||||||
|
<option value="cherryblossom">Cherryblossom</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' },
|
||||||
|
spring: { css: 'spring.css', js: 'spring.js', container: 'spring-container' },
|
||||||
|
summer: { css: 'summer.css', js: 'summer.js', container: 'summer-container' },
|
||||||
|
carnival: { css: 'carnival.css', js: 'carnival.js', container: 'carnival-container' },
|
||||||
|
cherryblossom: { css: 'cherryblossom.css', js: 'cherryblossom.js', container: 'cherryblossom-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',
|
||||||
|
'.christmas-container', '.santa-container', '.autumn-container',
|
||||||
|
'.easter-container', '.resurrection-container', '.spring-container',
|
||||||
|
'.summer-container', '.carnival-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>
|
||||||
@@ -107,7 +107,7 @@ Click on the following themes to expand them and see the theme in action:
|
|||||||
|
|
||||||
|
|
||||||
## Ideas for additional seasonals
|
## Ideas for additional seasonals
|
||||||
If you have any (specific) ideas for additional seasonals, feel free to open an issue or create a pull request.
|
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
|
||||||
|
|
||||||
@@ -243,4 +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.
|
||||||
|
|
||||||
|
|||||||
@@ -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.7.0.5",
|
"version": "1.7.1.5",
|
||||||
|
"changelog": "- feat: add summer, spring and carnival themes",
|
||||||
|
"targetAbi": "10.11.0.0",
|
||||||
|
"sourceUrl": "https://git.mahom03-spacecloud.de/CodeDevMLH/Jellyfin-Seasonals-Plugin/releases/download/v1.7.1.5/Jellyfin.Plugin.Seasonals.zip",
|
||||||
|
"checksum": "f6447d476189e69fb96fe1675c55a1a0",
|
||||||
|
"timestamp": "2026-02-21T14:28:30Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.7.0.15",
|
||||||
"changelog": "- feat: add customizable auto seasonal list via config page\n- feat: add new season theme 'resurrection' by Bioflash257",
|
"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.7.0.5/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": "dc018048e121b88469b8d8340970b2a6",
|
"checksum": "d1fc094710efe45ea8cc885bc6a826c4",
|
||||||
"timestamp": "2026-02-16T17:16:02Z"
|
"timestamp": "2026-02-17T13:11:21Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"version": "1.6.3.0",
|
"version": "1.6.3.0",
|
||||||
|
|||||||