parent
c8b0d59508
commit
cc830abae4
@ -6,6 +6,7 @@
|
|||||||
* Support selecting which libraries are analyzed
|
* Support selecting which libraries are analyzed
|
||||||
* Support customizing [introduction requirements](README.md#introduction-requirements)
|
* Support customizing [introduction requirements](README.md#introduction-requirements)
|
||||||
* Changing these settings will increase episode analysis times
|
* Changing these settings will increase episode analysis times
|
||||||
|
* Support adding and editing intro timestamps
|
||||||
* Report how CPU time is being spent while analyzing episodes
|
* Report how CPU time is being spent while analyzing episodes
|
||||||
* CPU time reports can be viewed under "Analysis Statistics (experimental)" in the plugin configuration page
|
* CPU time reports can be viewed under "Analysis Statistics (experimental)" in the plugin configuration page
|
||||||
* Sped up fingerprint analysis (not including fingerprint generation time) by 40%
|
* Sped up fingerprint analysis (not including fingerprint generation time) by 40%
|
||||||
|
@ -84,7 +84,8 @@
|
|||||||
<a href="https://kodi.wiki/view/Edit_decision_list">MPlayer compatible EDL files</a>
|
<a href="https://kodi.wiki/view/Edit_decision_list">MPlayer compatible EDL files</a>
|
||||||
alongside your episode files. <br />
|
alongside your episode files. <br />
|
||||||
|
|
||||||
If this value is changed after EDL files are generated, you must check the "Regenerate EDL files" checkbox below.
|
If this value is changed after EDL files are generated, you must check the
|
||||||
|
"Regenerate EDL files" checkbox below.
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -95,7 +96,8 @@
|
|||||||
</label>
|
</label>
|
||||||
|
|
||||||
<div class="fieldDescription">
|
<div class="fieldDescription">
|
||||||
If checked, the plugin will <strong>overwrite all EDL files</strong> associated with your episodes with the currently discovered introduction timestamps and EDL action.
|
If checked, the plugin will <strong>overwrite all EDL files</strong> associated with
|
||||||
|
your episodes with the currently discovered introduction timestamps and EDL action.
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
@ -276,6 +278,23 @@
|
|||||||
<br />
|
<br />
|
||||||
<br />
|
<br />
|
||||||
|
|
||||||
|
<div id="timestampEditor" style="display:none">
|
||||||
|
<p>Introduction timestamp editor. All times are in seconds.</p>
|
||||||
|
<p id="editLeftEpisodeTitle" style="margin-bottom:0"></p>
|
||||||
|
<input style="width:4em" type="number" min="0" id="editLeftEpisodeStart"> to
|
||||||
|
<input style="width:4em;margin-bottom:10px" type="number" min="0" id="editLeftEpisodeEnd">
|
||||||
|
|
||||||
|
<p id="editRightEpisodeTitle" style="margin-top:0;margin-bottom:0"></p>
|
||||||
|
<input style="width:4em" type="number" min="0" id="editRightEpisodeStart"> to
|
||||||
|
<input style="width:4em;margin-bottom:10px" type="number" min="0" id="editRightEpisodeEnd">
|
||||||
|
|
||||||
|
<button id="btnUpdateTimestamps" type="button">
|
||||||
|
Update timestamps
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<br />
|
||||||
|
<br />
|
||||||
|
|
||||||
<canvas id="troubleshooter"></canvas>
|
<canvas id="troubleshooter"></canvas>
|
||||||
<span id="timestampContainer">
|
<span id="timestampContainer">
|
||||||
<span id="timestamps"></span> <br />
|
<span id="timestamps"></span> <br />
|
||||||
@ -324,6 +343,7 @@
|
|||||||
var txtOffset = document.querySelector("input#offset");
|
var txtOffset = document.querySelector("input#offset");
|
||||||
var txtSuggested = document.querySelector("span#suggestedShifts");
|
var txtSuggested = document.querySelector("span#suggestedShifts");
|
||||||
var btnSeasonEraseTimestamps = document.querySelector("button#btnEraseSeasonTimestamps");
|
var btnSeasonEraseTimestamps = document.querySelector("button#btnEraseSeasonTimestamps");
|
||||||
|
var btnUpdateTimestamps = document.querySelector("button#btnUpdateTimestamps");
|
||||||
var timeContainer = document.querySelector("span#timestampContainer");
|
var timeContainer = document.querySelector("span#timestampContainer");
|
||||||
|
|
||||||
var windowHashInterval = 0;
|
var windowHashInterval = 0;
|
||||||
@ -446,14 +466,50 @@
|
|||||||
|
|
||||||
Dashboard.showLoadingMsg();
|
Dashboard.showLoadingMsg();
|
||||||
|
|
||||||
lhs = await getJson("Intros/Fingerprint/" + selectEpisode1.value);
|
lhs = await getJson("Intros/Episode/" + selectEpisode1.value + "/Chromaprint");
|
||||||
rhs = await getJson("Intros/Fingerprint/" + selectEpisode2.value);
|
rhs = await getJson("Intros/Episode/" + selectEpisode2.value + "/Chromaprint");
|
||||||
|
|
||||||
Dashboard.hideLoadingMsg();
|
Dashboard.hideLoadingMsg();
|
||||||
|
|
||||||
refreshBounds();
|
refreshBounds();
|
||||||
renderTroubleshooter();
|
renderTroubleshooter();
|
||||||
findExactMatches();
|
findExactMatches();
|
||||||
|
updateTimestampEditor();
|
||||||
|
}
|
||||||
|
|
||||||
|
// updates the timestamp editor
|
||||||
|
async function updateTimestampEditor() {
|
||||||
|
// Get the title and ID of the left and right episodes
|
||||||
|
const leftEpisode = selectEpisode1.options[selectEpisode1.selectedIndex];
|
||||||
|
const rightEpisode = selectEpisode2.options[selectEpisode2.selectedIndex];
|
||||||
|
|
||||||
|
// Try to get the timestamps of each intro, falling back a default value of zero if no intro was found
|
||||||
|
let leftEpisodeIntro = {
|
||||||
|
IntroStart: 0,
|
||||||
|
IntroEnd: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
let rightEpisodeIntro = {
|
||||||
|
IntroStart: 0,
|
||||||
|
IntroEnd: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
leftEpisodeIntro = await getJson("Episode/" + leftEpisode.value + "/IntroTimestamps/v1");
|
||||||
|
rightEpisodeIntro = await getJson("Episode/" + rightEpisode.value + "/IntroTimestamps/v1");
|
||||||
|
} catch {
|
||||||
|
// Ignore the server provided intro and use the defaults from above
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the editor for the first and second episodes
|
||||||
|
document.querySelector("#timestampEditor").style.display = "unset";
|
||||||
|
document.querySelector("#editLeftEpisodeTitle").textContent = leftEpisode.text;
|
||||||
|
document.querySelector("#editLeftEpisodeStart").value = Math.round(leftEpisodeIntro.IntroStart);
|
||||||
|
document.querySelector("#editLeftEpisodeEnd").value = Math.round(leftEpisodeIntro.IntroEnd);
|
||||||
|
|
||||||
|
document.querySelector("#editRightEpisodeTitle").textContent = rightEpisode.text;
|
||||||
|
document.querySelector("#editRightEpisodeStart").value = Math.round(rightEpisodeIntro.IntroStart);
|
||||||
|
document.querySelector("#editRightEpisodeEnd").value = Math.round(rightEpisodeIntro.IntroEnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
// adds an item to a dropdown
|
// adds an item to a dropdown
|
||||||
@ -472,20 +528,26 @@
|
|||||||
|
|
||||||
// make an authenticated GET to the server and parse the response as JSON
|
// make an authenticated GET to the server and parse the response as JSON
|
||||||
async function getJson(url, method = "GET") {
|
async function getJson(url, method = "GET") {
|
||||||
|
return await fetchWithAuth(url, method).then(r => { return r.json(); });
|
||||||
|
}
|
||||||
|
|
||||||
|
// make an authenticated fetch to the server
|
||||||
|
async function fetchWithAuth(url, method, body) {
|
||||||
url = ApiClient.serverAddress() + "/" + url;
|
url = ApiClient.serverAddress() + "/" + url;
|
||||||
|
|
||||||
const reqInit = {
|
const reqInit = {
|
||||||
method: method,
|
method: method,
|
||||||
headers: {
|
headers: {
|
||||||
"Authorization": "MediaBrowser Token=" + ApiClient.accessToken()
|
"Authorization": "MediaBrowser Token=" + ApiClient.accessToken()
|
||||||
}
|
},
|
||||||
|
body: body,
|
||||||
};
|
};
|
||||||
|
|
||||||
return await
|
if (method === "POST") {
|
||||||
fetch(url, reqInit)
|
reqInit.headers["Content-Type"] = "application/json";
|
||||||
.then(r => {
|
}
|
||||||
return r.json();
|
|
||||||
});
|
return await fetch(url, reqInit);
|
||||||
}
|
}
|
||||||
|
|
||||||
// key pressed
|
// key pressed
|
||||||
@ -626,6 +688,24 @@
|
|||||||
|
|
||||||
Dashboard.alert("Erased timestamps for " + season + " of " + show);
|
Dashboard.alert("Erased timestamps for " + season + " of " + show);
|
||||||
});
|
});
|
||||||
|
btnUpdateTimestamps.addEventListener("click", () => {
|
||||||
|
const lhsId = selectEpisode1.options[selectEpisode1.selectedIndex].value;
|
||||||
|
const newLhsIntro = {
|
||||||
|
IntroStart: document.querySelector("#editLeftEpisodeStart").value,
|
||||||
|
IntroEnd: document.querySelector("#editLeftEpisodeEnd").value,
|
||||||
|
};
|
||||||
|
|
||||||
|
const rhsId = selectEpisode2.options[selectEpisode2.selectedIndex].value;
|
||||||
|
const newRhsIntro = {
|
||||||
|
IntroStart: document.querySelector("#editRightEpisodeStart").value,
|
||||||
|
IntroEnd: document.querySelector("#editRightEpisodeEnd").value,
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchWithAuth("Intros/Episode/" + lhsId + "/UpdateIntroTimestamps", "POST", JSON.stringify(newLhsIntro));
|
||||||
|
fetchWithAuth("Intros/Episode/" + rhsId + "/UpdateIntroTimestamps", "POST", JSON.stringify(newRhsIntro));
|
||||||
|
|
||||||
|
Dashboard.alert("New introduction timestamps saved");
|
||||||
|
});
|
||||||
document.addEventListener("keydown", keyDown);
|
document.addEventListener("keydown", keyDown);
|
||||||
windowHashInterval = setInterval(checkWindowHash, 2500);
|
windowHashInterval = setInterval(checkWindowHash, 2500);
|
||||||
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.ObjectModel;
|
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Net.Mime;
|
using System.Net.Mime;
|
||||||
using Microsoft.AspNetCore.Authorization;
|
using Microsoft.AspNetCore.Authorization;
|
||||||
@ -102,7 +101,7 @@ public class VisualizationController : ControllerBase
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="id">Episode id.</param>
|
/// <param name="id">Episode id.</param>
|
||||||
/// <returns>Read only collection of fingerprint points.</returns>
|
/// <returns>Read only collection of fingerprint points.</returns>
|
||||||
[HttpGet("Fingerprint/{Id}")]
|
[HttpGet("Episode/{Id}/Chromaprint")]
|
||||||
public ActionResult<uint[]> GetEpisodeFingerprint([FromRoute] Guid id)
|
public ActionResult<uint[]> GetEpisodeFingerprint([FromRoute] Guid id)
|
||||||
{
|
{
|
||||||
var queue = Plugin.Instance!.AnalysisQueue;
|
var queue = Plugin.Instance!.AnalysisQueue;
|
||||||
@ -150,6 +149,23 @@ public class VisualizationController : ControllerBase
|
|||||||
return NoContent();
|
return NoContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Updates the timestamps for the provided episode.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="id">Episode ID to update timestamps for.</param>
|
||||||
|
/// <param name="timestamps">New introduction start and end times.</param>
|
||||||
|
/// <response code="204">New introduction timestamps saved.</response>
|
||||||
|
/// <returns>No content.</returns>
|
||||||
|
[HttpPost("Episode/{Id}/UpdateIntroTimestamps")]
|
||||||
|
public ActionResult UpdateTimestamps([FromRoute] Guid id, [FromBody] Intro timestamps)
|
||||||
|
{
|
||||||
|
var tr = new TimeRange(timestamps.IntroStart, timestamps.IntroEnd);
|
||||||
|
Plugin.Instance!.Intros[id] = new Intro(id, tr);
|
||||||
|
Plugin.Instance!.SaveTimestamps();
|
||||||
|
|
||||||
|
return NoContent();
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Returns the statistics for the most recent analysis.
|
/// Returns the statistics for the most recent analysis.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Globalization;
|
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using ConfusedPolarBear.Plugin.IntroSkipper.Configuration;
|
using ConfusedPolarBear.Plugin.IntroSkipper.Configuration;
|
||||||
using MediaBrowser.Common.Configuration;
|
using MediaBrowser.Common.Configuration;
|
||||||
@ -17,6 +16,7 @@ namespace ConfusedPolarBear.Plugin.IntroSkipper;
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class Plugin : BasePlugin<PluginConfiguration>, IHasWebPages
|
public class Plugin : BasePlugin<PluginConfiguration>, IHasWebPages
|
||||||
{
|
{
|
||||||
|
private readonly object _serializationLock = new object();
|
||||||
private IXmlSerializer _xmlSerializer;
|
private IXmlSerializer _xmlSerializer;
|
||||||
private ILibraryManager _libraryManager;
|
private ILibraryManager _libraryManager;
|
||||||
private string _introPath;
|
private string _introPath;
|
||||||
@ -110,14 +110,17 @@ public class Plugin : BasePlugin<PluginConfiguration>, IHasWebPages
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public void SaveTimestamps()
|
public void SaveTimestamps()
|
||||||
{
|
{
|
||||||
var introList = new List<Intro>();
|
lock (_serializationLock)
|
||||||
|
|
||||||
foreach (var intro in Plugin.Instance!.Intros)
|
|
||||||
{
|
{
|
||||||
introList.Add(intro.Value);
|
var introList = new List<Intro>();
|
||||||
}
|
|
||||||
|
|
||||||
_xmlSerializer.SerializeToFile(introList, _introPath);
|
foreach (var intro in Plugin.Instance!.Intros)
|
||||||
|
{
|
||||||
|
introList.Add(intro.Value);
|
||||||
|
}
|
||||||
|
|
||||||
|
_xmlSerializer.SerializeToFile(introList, _introPath);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user