mirror of
https://github.com/Theodor-Springmann-Stiftung/hamann-ausgabe-core.git
synced 2025-10-29 09:15:33 +00:00
Fixed a few bugs; fixed marginals; fixed google indexing
This commit is contained in:
1
HaWeb/.gitignore
vendored
1
HaWeb/.gitignore
vendored
@@ -23,3 +23,4 @@ bin/*
|
|||||||
*.njsproj
|
*.njsproj
|
||||||
*.sln
|
*.sln
|
||||||
*.sw?
|
*.sw?
|
||||||
|
testdata/
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ using System.Xml.Linq;
|
|||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
|
|
||||||
public class HaDocumentWrapper : IHaDocumentWrappper {
|
public class HaDocumentWrapper : IHaDocumentWrappper {
|
||||||
private IFileInfo _ActiveFile;
|
private IFileInfo? _ActiveFile;
|
||||||
private ILibrary? Library;
|
private ILibrary? Library;
|
||||||
private IXMLInteractionService _xmlService;
|
private IXMLInteractionService _xmlService;
|
||||||
private int _startYear;
|
private int _startYear;
|
||||||
@@ -51,7 +51,7 @@ public class HaDocumentWrapper : IHaDocumentWrappper {
|
|||||||
// 1. Parse the Document, create search Index
|
// 1. Parse the Document, create search Index
|
||||||
var sw = new Stopwatch();
|
var sw = new Stopwatch();
|
||||||
sw.Start();
|
sw.Start();
|
||||||
if (_xmlService != null)
|
if (_xmlService != null)
|
||||||
_xmlService.CreateCollections(doc);
|
_xmlService.CreateCollections(doc);
|
||||||
sw.Stop();
|
sw.Stop();
|
||||||
Console.WriteLine("Parsed Collections, elapsed: " + sw.ElapsedMilliseconds);
|
Console.WriteLine("Parsed Collections, elapsed: " + sw.ElapsedMilliseconds);
|
||||||
@@ -60,7 +60,8 @@ public class HaDocumentWrapper : IHaDocumentWrappper {
|
|||||||
// 2. Set ILibrary
|
// 2. Set ILibrary
|
||||||
try {
|
try {
|
||||||
Library = HaDocument.Document.Create(new HaWeb.Settings.HaDocumentOptions() { HamannXMLFilePath = path, AvailableYearRange = (_startYear, _endYear) }, doc.Root);
|
Library = HaDocument.Document.Create(new HaWeb.Settings.HaDocumentOptions() { HamannXMLFilePath = path, AvailableYearRange = (_startYear, _endYear) }, doc.Root);
|
||||||
} catch (Exception ex) {
|
}
|
||||||
|
catch (Exception ex) {
|
||||||
if (ModelState != null) ModelState.AddModelError("Error", "Das Dokument konnte nicht geparst werden: " + ex.Message);
|
if (ModelState != null) ModelState.AddModelError("Error", "Das Dokument konnte nicht geparst werden: " + ex.Message);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -87,4 +88,4 @@ public class HaDocumentWrapper : IHaDocumentWrappper {
|
|||||||
public ILibrary? GetLibrary() {
|
public ILibrary? GetLibrary() {
|
||||||
return Library;
|
return Library;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,10 +3,8 @@ using Microsoft.Extensions.FileProviders;
|
|||||||
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
||||||
using HaWeb.Models;
|
using HaWeb.Models;
|
||||||
using HaWeb.XMLParser;
|
using HaWeb.XMLParser;
|
||||||
using HaWeb.XMLTests;
|
|
||||||
using System.Xml.Linq;
|
using System.Xml.Linq;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
using System.Diagnostics;
|
|
||||||
using Microsoft.Extensions.Primitives;
|
using Microsoft.Extensions.Primitives;
|
||||||
|
|
||||||
// XMLProvider provides a wrapper around the available XML data on a FILE basis
|
// XMLProvider provides a wrapper around the available XML data on a FILE basis
|
||||||
@@ -17,7 +15,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
private IFileProvider _hamannFileProvider;
|
private IFileProvider _hamannFileProvider;
|
||||||
private IFileProvider _bareRepositoryFileProvider;
|
private IFileProvider _bareRepositoryFileProvider;
|
||||||
private IFileProvider _workingTreeFileProvider;
|
private IFileProvider _workingTreeFileProvider;
|
||||||
|
|
||||||
public event EventHandler<GitState?> FileChange;
|
public event EventHandler<GitState?> FileChange;
|
||||||
public event EventHandler ConfigReload;
|
public event EventHandler ConfigReload;
|
||||||
public event EventHandler<XMLParsingState?> NewState;
|
public event EventHandler<XMLParsingState?> NewState;
|
||||||
@@ -28,7 +26,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
|
|
||||||
private List<IFileInfo>? _WorkingTreeFiles;
|
private List<IFileInfo>? _WorkingTreeFiles;
|
||||||
private List<IFileInfo>? _HamannFiles;
|
private List<IFileInfo>? _HamannFiles;
|
||||||
|
|
||||||
private GitState? _GitState;
|
private GitState? _GitState;
|
||||||
private System.Timers.Timer? _changeTokenTimer;
|
private System.Timers.Timer? _changeTokenTimer;
|
||||||
|
|
||||||
@@ -42,13 +40,13 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
_URL = config.GetValue<string>("RepositoryURL");
|
_URL = config.GetValue<string>("RepositoryURL");
|
||||||
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) {
|
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) {
|
||||||
_hamannFileProvider = new PhysicalFileProvider(config.GetValue<string>("HamannFileStoreWindows"));
|
_hamannFileProvider = new PhysicalFileProvider(config.GetValue<string>("HamannFileStoreWindows"));
|
||||||
_bareRepositoryFileProvider = new PhysicalFileProvider(config.GetValue<string>("BareRepositoryPathWindows"));
|
_bareRepositoryFileProvider = new PhysicalFileProvider(config.GetValue<string>("BareRepositoryPathWindows"));
|
||||||
_workingTreeFileProvider = new PhysicalFileProvider(config.GetValue<string>("WorkingTreePathWindows"));
|
_workingTreeFileProvider = new PhysicalFileProvider(config.GetValue<string>("WorkingTreePathWindows"));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
_hamannFileProvider = new PhysicalFileProvider(config.GetValue<string>("HamannFileStoreLinux"));
|
_hamannFileProvider = new PhysicalFileProvider(config.GetValue<string>("HamannFileStoreLinux"));
|
||||||
_bareRepositoryFileProvider = new PhysicalFileProvider(config.GetValue<string>("BareRepositoryPathLinux"));
|
_bareRepositoryFileProvider = new PhysicalFileProvider(config.GetValue<string>("BareRepositoryPathLinux"));
|
||||||
_workingTreeFileProvider = new PhysicalFileProvider(config.GetValue<string>("WorkingTreePathLinux"));
|
_workingTreeFileProvider = new PhysicalFileProvider(config.GetValue<string>("WorkingTreePathLinux"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create File Lists; Here and in xmlservice, which does preliminary checking
|
// Create File Lists; Here and in xmlservice, which does preliminary checking
|
||||||
@@ -58,16 +56,16 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
xmlservice.SetState(state);
|
xmlservice.SetState(state);
|
||||||
}
|
}
|
||||||
_HamannFiles = _ScanHamannFiles();
|
_HamannFiles = _ScanHamannFiles();
|
||||||
|
|
||||||
_RegisterChangeToken();
|
_RegisterChangeToken();
|
||||||
// Check if hamann file already is current working tree status
|
// Check if hamann file already is current working tree status
|
||||||
// -> YES: Load up the file via _lib.SetLibrary();
|
// -> YES: Load up the file via _lib.SetLibrary();
|
||||||
if (_IsAlreadyParsed()) {
|
if (_IsAlreadyParsed()) {
|
||||||
_Lib.SetLibrary(_HamannFiles!.First(), null, null);
|
_Lib.SetLibrary(_HamannFiles!.First(), null, null);
|
||||||
if (_Lib.GetLibrary() != null) return;
|
if (_Lib.GetLibrary() != null) return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> NO: Try to create a new file
|
// -> NO: Try to create a new file
|
||||||
var created = _XMLService.TryCreate(_XMLService.GetState());
|
var created = _XMLService.TryCreate(_XMLService.GetState());
|
||||||
if (created != null) {
|
if (created != null) {
|
||||||
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
||||||
@@ -82,11 +80,11 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
_Lib.SetLibrary(_HamannFiles.First(), null, null);
|
_Lib.SetLibrary(_HamannFiles.First(), null, null);
|
||||||
if (_Lib.GetLibrary() != null) return;
|
if (_Lib.GetLibrary() != null) return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> There is none? Use Fallback:
|
// -> There is none? Use Fallback:
|
||||||
else {
|
else {
|
||||||
var options = new HaWeb.Settings.HaDocumentOptions();
|
var options = new HaWeb.Settings.HaDocumentOptions();
|
||||||
if (_Lib.SetLibrary(null, null, null) == null) {
|
if (_Lib.SetLibrary(null, null, null) == null) {
|
||||||
throw new Exception("Die Fallback Hamann.xml unter " + options.HamannXMLFilePath + " kann nicht geparst werden.");
|
throw new Exception("Die Fallback Hamann.xml unter " + options.HamannXMLFilePath + " kann nicht geparst werden.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -116,7 +114,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
if (_Lib.GetLibrary() != null) return;
|
if (_Lib.GetLibrary() != null) return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> NO: Try to create a new file
|
// -> NO: Try to create a new file
|
||||||
var created = _XMLService.TryCreate(_XMLService.GetState());
|
var created = _XMLService.TryCreate(_XMLService.GetState());
|
||||||
if (created != null) {
|
if (created != null) {
|
||||||
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
||||||
@@ -131,11 +129,11 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
_Lib.SetLibrary(_HamannFiles.First(), null, null);
|
_Lib.SetLibrary(_HamannFiles.First(), null, null);
|
||||||
if (_Lib.GetLibrary() != null) return;
|
if (_Lib.GetLibrary() != null) return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> There is none? Use Fallback:
|
// -> There is none? Use Fallback:
|
||||||
else {
|
else {
|
||||||
var options = new HaWeb.Settings.HaDocumentOptions();
|
var options = new HaWeb.Settings.HaDocumentOptions();
|
||||||
if (_Lib.SetLibrary(null, null, null) == null) {
|
if (_Lib.SetLibrary(null, null, null) == null) {
|
||||||
throw new Exception("Die Fallback Hamann.xml unter " + options.HamannXMLFilePath + " kann nicht geparst werden.");
|
throw new Exception("Die Fallback Hamann.xml unter " + options.HamannXMLFilePath + " kann nicht geparst werden.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -165,7 +163,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
|
|
||||||
public IFileInfo? SaveHamannFile(XElement element, string basefilepath, ModelStateDictionary? ModelState) {
|
public IFileInfo? SaveHamannFile(XElement element, string basefilepath, ModelStateDictionary? ModelState) {
|
||||||
if (_GitState == null) return null;
|
if (_GitState == null) return null;
|
||||||
var filename = "hamann_" + _GitState.PullTime.Year + "-" + _GitState.PullTime.Month + "-" + _GitState.PullTime.Day + "_" + _GitState.PullTime.Hour + "-" + _GitState.PullTime.Minute + "." + _GitState.Commit.Substring(0,7) + ".xml";
|
var filename = "hamann_" + _GitState.PullTime.Year + "-" + _GitState.PullTime.Month + "-" + _GitState.PullTime.Day + "_" + _GitState.PullTime.Hour + "-" + _GitState.PullTime.Minute + "." + _GitState.Commit.Substring(0, 7) + ".xml";
|
||||||
var path = Path.Combine(basefilepath, filename);
|
var path = Path.Combine(basefilepath, filename);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -173,7 +171,8 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
Directory.CreateDirectory(basefilepath);
|
Directory.CreateDirectory(basefilepath);
|
||||||
using (var targetStream = System.IO.File.Create(path))
|
using (var targetStream = System.IO.File.Create(path))
|
||||||
element.Save(targetStream, SaveOptions.DisableFormatting);
|
element.Save(targetStream, SaveOptions.DisableFormatting);
|
||||||
} catch (Exception ex) {
|
}
|
||||||
|
catch (Exception ex) {
|
||||||
if (ModelState != null) ModelState.AddModelError("Error", "Die Datei konnte nicht gespeichert werden: " + ex.Message);
|
if (ModelState != null) ModelState.AddModelError("Error", "Die Datei konnte nicht gespeichert werden: " + ex.Message);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -237,7 +236,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
private bool _IsAlreadyParsed() {
|
private bool _IsAlreadyParsed() {
|
||||||
if (_HamannFiles == null || !_HamannFiles.Any() || _GitState == null) return false;
|
if (_HamannFiles == null || !_HamannFiles.Any() || _GitState == null) return false;
|
||||||
var fhash = _GetHashFromHamannFilename(_HamannFiles.First().Name);
|
var fhash = _GetHashFromHamannFilename(_HamannFiles.First().Name);
|
||||||
var ghash = _GitState.Commit.Substring(0,7);
|
var ghash = _GitState.Commit.Substring(0, 7);
|
||||||
return fhash == ghash;
|
return fhash == ghash;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -261,8 +260,8 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
_XMLService.SetState(state);
|
_XMLService.SetState(state);
|
||||||
OnNewState(state);
|
OnNewState(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> Try to create a new file
|
// -> Try to create a new file
|
||||||
var created = _XMLService.TryCreate(_XMLService.GetState());
|
var created = _XMLService.TryCreate(_XMLService.GetState());
|
||||||
if (created != null) {
|
if (created != null) {
|
||||||
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
var file = SaveHamannFile(created, _hamannFileProvider.GetFileInfo("./").PhysicalPath, null);
|
||||||
@@ -285,7 +284,7 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
protected virtual void OnFileChange(GitState? state) {
|
protected virtual void OnFileChange(GitState? state) {
|
||||||
EventHandler<GitState?> eh = FileChange;
|
EventHandler<GitState?> eh = FileChange;
|
||||||
eh?.Invoke(this, state);
|
eh?.Invoke(this, state);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual void OnNewState(XMLParsingState? state) {
|
protected virtual void OnNewState(XMLParsingState? state) {
|
||||||
EventHandler<XMLParsingState?> eh = NewState;
|
EventHandler<XMLParsingState?> eh = NewState;
|
||||||
@@ -301,4 +300,4 @@ public class XMLFileProvider : IXMLFileProvider {
|
|||||||
EventHandler eh = NewData;
|
EventHandler eh = NewData;
|
||||||
eh?.Invoke(this, System.EventArgs.Empty);
|
eh?.Invoke(this, System.EventArgs.Empty);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ var app = builder.Build();
|
|||||||
|
|
||||||
// // Websockets for realtime notification of changes
|
// // Websockets for realtime notification of changes
|
||||||
app.UseWebSockets(new WebSocketOptions {
|
app.UseWebSockets(new WebSocketOptions {
|
||||||
KeepAliveInterval = TimeSpan.FromMinutes(180),
|
KeepAliveInterval = TimeSpan.FromMinutes(30),
|
||||||
});
|
});
|
||||||
app.UseMiddleware<WebSocketMiddleware>();
|
app.UseMiddleware<WebSocketMiddleware>();
|
||||||
|
|
||||||
|
|||||||
@@ -13,13 +13,13 @@
|
|||||||
},
|
},
|
||||||
"AllowedWebSocketConnections": "*",
|
"AllowedWebSocketConnections": "*",
|
||||||
"AllowedHosts": "*",
|
"AllowedHosts": "*",
|
||||||
"HamannFileStoreLinux": "/home/simon/test/",
|
"HamannFileStoreLinux": "/home/simon/source/hamann-ausgabe-core/HaWeb/testdata/",
|
||||||
"HamannFileStoreWindows": "C:/Users/simon/Downloads/test/",
|
"HamannFileStoreWindows": "C:/Users/simon/Downloads/test/",
|
||||||
"BareRepositoryPathLinux": "/home/simon/source/hamann-xml/.git/",
|
"BareRepositoryPathLinux": "/home/simon/source/hamann-xml/.git/",
|
||||||
"BareRepositoryPathWindows": "D:/Simon/source/hamann-xml/.git/",
|
"BareRepositoryPathWindows": "D:/Simon/source/hamann-xml/.git/",
|
||||||
"WorkingTreePathLinux": "/home/simon/source/hamann-xml/",
|
"WorkingTreePathLinux": "/home/simon/source/hamann-xml/",
|
||||||
"WorkingTreePathWindows": "D:/Simon/source/hamann-xml/",
|
"WorkingTreePathWindows": "D:/Simon/source/hamann-xml/",
|
||||||
"RepositoryBranch": "Release",
|
"RepositoryBranch": "Main",
|
||||||
"RepositoryURL": "https://github.com/Theodor-Springmann-Stiftung/hamann-xml",
|
"RepositoryURL": "https://github.com/Theodor-Springmann-Stiftung/hamann-xml",
|
||||||
"StoredPDFPathWindows": "",
|
"StoredPDFPathWindows": "",
|
||||||
"StoredPDFPathLinux": "",
|
"StoredPDFPathLinux": "",
|
||||||
@@ -27,4 +27,4 @@
|
|||||||
"AvailableStartYear": 1700,
|
"AvailableStartYear": 1700,
|
||||||
"AvailableEndYear": 1800,
|
"AvailableEndYear": 1800,
|
||||||
"LettersOnPage": 80
|
"LettersOnPage": 80
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,210 +1,221 @@
|
|||||||
// Script for auto collapsing marginal boxes
|
// Script for auto collapsing marginal boxes
|
||||||
const startup_marginals = function () {
|
const startup_marginals = function () {
|
||||||
let debounce_resize;
|
let debounce_resize;
|
||||||
let collapsedboxes = [];
|
let collapsedboxes = [];
|
||||||
|
|
||||||
const getLineHeight = function (element) {
|
const getLineHeight = function (element) {
|
||||||
var temp = document.createElement(element.nodeName),
|
var temp = document.createElement(element.nodeName),
|
||||||
ret;
|
ret;
|
||||||
temp.setAttribute("class", element.className);
|
temp.setAttribute("class", element.className);
|
||||||
temp.innerHTML = "Üj";
|
temp.innerHTML = "Üj";
|
||||||
element.parentNode.appendChild(temp);
|
element.parentNode.appendChild(temp);
|
||||||
ret = temp.getBoundingClientRect().height;
|
ret = temp.getBoundingClientRect().height;
|
||||||
temp.parentNode.removeChild(temp);
|
temp.parentNode.removeChild(temp);
|
||||||
return ret;
|
return ret;
|
||||||
};
|
};
|
||||||
|
|
||||||
const collapsebox = function (element, height, lineheight) {
|
const collapsebox = function (element, height, lineheight) {
|
||||||
element.style.maxHeight = height + "px";
|
element.style.maxHeight = height + "px";
|
||||||
element.classList.add("ha-collapsed-box");
|
element.classList.add("ha-collapsed-box");
|
||||||
element.classList.remove("ha-expanded-box");
|
element.classList.remove("ha-expanded-box");
|
||||||
setTimeout(function () {
|
setTimeout(function () {
|
||||||
element.classList.remove("transition-all");
|
element.classList.remove("transition-all");
|
||||||
}, 130);
|
}, 130);
|
||||||
};
|
element.style.bottom = "unset";
|
||||||
|
};
|
||||||
|
|
||||||
const uncollapsebox = function (element) {
|
const uncollapsebox = function (element, topmove = false) {
|
||||||
element.classList.add("transition-all");
|
element.classList.add("transition-all");
|
||||||
element.classList.remove("ha-collapsed-box");
|
element.classList.remove("ha-collapsed-box");
|
||||||
element.classList.add("ha-expanded-box");
|
element.classList.add("ha-expanded-box");
|
||||||
};
|
if (topmove > 0) element.style.bottom = "5px";
|
||||||
|
};
|
||||||
|
|
||||||
const addbuttoncaollapsebox = function (element, height, hoverfunction, topmove) {
|
const addbuttoncaollapsebox = function (
|
||||||
let btn = document.createElement("div");
|
element,
|
||||||
btn.classList.add("ha-btn-collapsed-box");
|
height,
|
||||||
|
hoverfunction,
|
||||||
|
topmove = false,
|
||||||
|
) {
|
||||||
|
let btn = document.createElement("div");
|
||||||
|
btn.classList.add("ha-btn-collapsed-box");
|
||||||
|
|
||||||
if (element.classList.contains("ha-collapsed-box")) {
|
if (element.classList.contains("ha-collapsed-box")) {
|
||||||
btn.classList.add("ha-open-btn-collapsed-box");
|
btn.classList.add("ha-open-btn-collapsed-box");
|
||||||
} else {
|
} else {
|
||||||
btn.classList.add("ha-close-btn-collapsed-box");
|
btn.classList.add("ha-close-btn-collapsed-box");
|
||||||
}
|
}
|
||||||
|
|
||||||
btn.addEventListener("click", function (ev) {
|
btn.addEventListener("click", function (ev) {
|
||||||
ev.stopPropagation();
|
ev.stopPropagation();
|
||||||
if (element.classList.contains("ha-collapsed-box")) {
|
if (element.classList.contains("ha-collapsed-box")) {
|
||||||
uncollapsebox(element);
|
uncollapsebox(element, topmove);
|
||||||
if (topmove > 0) element.style.bottom = "5px";
|
btn.classList.add("ha-close-btn-collapsed-box");
|
||||||
btn.classList.add("ha-close-btn-collapsed-box");
|
btn.classList.add("ha-collapsed-box-manually-toggled");
|
||||||
btn.classList.add("ha-collapsed-box-manually-toggled");
|
} else {
|
||||||
} else {
|
collapsebox(element, height, 0);
|
||||||
collapsebox(element, height, 0);
|
btn.classList.remove("ha-close-btn-collapsed-box");
|
||||||
|
btn.classList.remove("ha-collapsed-box-manually-toggled");
|
||||||
btn.classList.remove("ha-close-btn-collapsed-box");
|
}
|
||||||
btn.classList.remove("ha-collapsed-box-manually-toggled");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (hoverfunction) {
|
|
||||||
let timer = null;
|
|
||||||
|
|
||||||
element.addEventListener("mouseenter", function (ev) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
timer = setTimeout(function () {
|
|
||||||
if (element.classList.contains("ha-collapsed-box")) {
|
|
||||||
uncollapsebox(element);
|
|
||||||
if (topmove > 0) element.style.bottom = "5px";
|
|
||||||
btn.classList.add("ha-close-btn-collapsed-box");
|
|
||||||
}
|
|
||||||
}, 80);
|
|
||||||
});
|
|
||||||
|
|
||||||
element.addEventListener("mouseleave", function (ev) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
if (timer != null) {
|
|
||||||
clearTimeout(timer);
|
|
||||||
}
|
|
||||||
if (
|
|
||||||
element.classList.contains("ha-expanded-box") &&
|
|
||||||
!btn.classList.contains("ha-collapsed-box-manually-toggled")
|
|
||||||
) {
|
|
||||||
collapsebox(element, height, 0);
|
|
||||||
btn.classList.remove("ha-close-btn-collapsed-box");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
element.parentNode.insertBefore(btn, element);
|
|
||||||
};
|
|
||||||
|
|
||||||
const overlappingcollapsebox = function (selector, hoverfunction, containerid) {
|
|
||||||
let container = document.getElementById(containerid);
|
|
||||||
if (!container) return;
|
|
||||||
container.classList.add("overflow-hidden");
|
|
||||||
let containerrect = document.getElementById(containerid).getBoundingClientRect();;
|
|
||||||
let boxes = document.querySelectorAll(selector);
|
|
||||||
let lineheight = 1;
|
|
||||||
|
|
||||||
if (boxes.length >= 1) {
|
|
||||||
lineheight = getLineHeight(boxes[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (var i = 0; i < boxes.length; i++) {
|
|
||||||
let element = boxes[i];
|
|
||||||
let thisrect = element.getBoundingClientRect();
|
|
||||||
let overlap = -2;
|
|
||||||
let topmove = 0;
|
|
||||||
if (thisrect.bottom > containerrect.bottom) {
|
|
||||||
overlap = thisrect.bottom - containerrect.bottom;
|
|
||||||
topmove = thisrect.bottom - containerrect.bottom;
|
|
||||||
console.log("topmove", topmove);
|
|
||||||
} else if (i < boxes.length - 1) {
|
|
||||||
let nextrect = boxes[i + 1].getBoundingClientRect();
|
|
||||||
overlap = thisrect.bottom - nextrect.top;
|
|
||||||
}
|
|
||||||
if (
|
|
||||||
// -1 for catching lines that perfectly close up on each other
|
|
||||||
overlap >= -1 &&
|
|
||||||
!(window.getComputedStyle(element).display === "none")
|
|
||||||
) {
|
|
||||||
let newlength = 0;
|
|
||||||
if (overlap >= 0)
|
|
||||||
newlength = thisrect.height - overlap;
|
|
||||||
else
|
|
||||||
newlength = thisrect.height - lineheight;
|
|
||||||
if (newlength % (lineheight * 3) <= 2)
|
|
||||||
newlength -= lineheight;
|
|
||||||
let remainder = newlength % lineheight;
|
|
||||||
newlength = newlength - remainder;
|
|
||||||
|
|
||||||
// Line clamping for Marginals
|
|
||||||
if (element.classList.contains("ha-marginalbox")) {
|
|
||||||
let marginals = element.querySelectorAll(".ha-marginal");
|
|
||||||
let h = 0;
|
|
||||||
for (let m of marginals) {
|
|
||||||
let cr = m.getBoundingClientRect();
|
|
||||||
let eh = cr.bottom - cr.top;
|
|
||||||
h += eh;
|
|
||||||
if (h >= newlength) {
|
|
||||||
let lines = Math.floor(eh / lineheight);
|
|
||||||
let cutoff = Math.floor((h - newlength) / lineheight);
|
|
||||||
m.style.cssText += "-webkit-line-clamp: " + (lines - cutoff) + ";";
|
|
||||||
m.style.cssText += "line-clamp: " + (lines - cutoff) + ";";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
requestAnimationFrame(() => {
|
|
||||||
collapsedboxes.push(element);
|
|
||||||
collapsebox(element, newlength, lineheight);
|
|
||||||
addbuttoncaollapsebox(element, newlength, hoverfunction, topmove);
|
|
||||||
});
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const marginalboxwidthset = function (containerid, classes) {
|
|
||||||
let lt = document.getElementById(containerid);
|
|
||||||
if (lt !== null) {
|
|
||||||
let mg = lt.querySelectorAll(classes);
|
|
||||||
if (mg.length > 0) {
|
|
||||||
let ltbcr = lt.getBoundingClientRect();
|
|
||||||
let mgbcr = mg[0].getBoundingClientRect();
|
|
||||||
let nw = ltbcr.right - mgbcr.left - 20;
|
|
||||||
|
|
||||||
for (let element of mg) {
|
|
||||||
element.style.width = nw + "px";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const clearcollapsedboxes = function () {
|
|
||||||
let elements = document.querySelectorAll(".ha-marginalbox");
|
|
||||||
elements.forEach(element => {
|
|
||||||
element.removeAttribute("style");
|
|
||||||
});
|
|
||||||
collapsedboxes.forEach(element => {
|
|
||||||
element.classList.remove("ha-expanded-box");
|
|
||||||
element.classList.remove("ha-collapsed-box");
|
|
||||||
element.outerHTML = element.outerHTML;
|
|
||||||
});
|
|
||||||
collapsedboxes = [];
|
|
||||||
elements = document.querySelectorAll(".ha-btn-collapsed-box");
|
|
||||||
elements.forEach(element => {
|
|
||||||
element.remove();
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const resetall = function () {
|
|
||||||
clearcollapsedboxes();
|
|
||||||
marginalboxwidthset("ha-letterbody", ".ha-marginalbox");
|
|
||||||
marginalboxwidthset("ha-register-body", ".ha-letlinks");
|
|
||||||
startup_marginals();
|
|
||||||
};
|
|
||||||
|
|
||||||
const collapseboxes = function () {
|
|
||||||
overlappingcollapsebox(".ha-letlinks", true, "ha-register-body");
|
|
||||||
overlappingcollapsebox(".ha-marginalbox", true, "ha-letterbody");
|
|
||||||
};
|
|
||||||
|
|
||||||
window.addEventListener("resize", function () {
|
|
||||||
clearTimeout(debounce_resize);
|
|
||||||
debounce_resize = setTimeout(resetall, 17);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (hoverfunction) {
|
||||||
|
let timer = null;
|
||||||
|
|
||||||
|
element.addEventListener("mouseenter", function (ev) {
|
||||||
|
ev.stopPropagation();
|
||||||
|
if (timer != null) {
|
||||||
|
clearTimeout(timer);
|
||||||
|
}
|
||||||
|
if (element.classList.contains("ha-collapsed-box")) {
|
||||||
|
uncollapsebox(element, topmove);
|
||||||
|
btn.classList.add("ha-close-btn-collapsed-box");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
element.addEventListener("mouseleave", function (ev) {
|
||||||
|
ev.stopPropagation();
|
||||||
|
timer = setTimeout(function () {
|
||||||
|
if (
|
||||||
|
element.classList.contains("ha-expanded-box") &&
|
||||||
|
!btn.classList.contains("ha-collapsed-box-manually-toggled")
|
||||||
|
) {
|
||||||
|
collapsebox(element, height, 0);
|
||||||
|
btn.classList.remove("ha-close-btn-collapsed-box");
|
||||||
|
}
|
||||||
|
}, 200);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
element.parentNode.insertBefore(btn, element);
|
||||||
|
};
|
||||||
|
|
||||||
|
const overlappingcollapsebox = function (
|
||||||
|
selector,
|
||||||
|
hoverfunction,
|
||||||
|
containerid,
|
||||||
|
) {
|
||||||
|
let container = document.getElementById(containerid);
|
||||||
|
if (!container) return;
|
||||||
|
let containerrect = document
|
||||||
|
.getElementById(containerid)
|
||||||
|
.getBoundingClientRect();
|
||||||
|
let boxes = document.querySelectorAll(selector);
|
||||||
|
let lineheight = 1;
|
||||||
|
|
||||||
|
if (boxes.length >= 1) {
|
||||||
|
lineheight = getLineHeight(boxes[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < boxes.length; i++) {
|
||||||
|
let element = boxes[i];
|
||||||
|
let thisrect = element.getBoundingClientRect();
|
||||||
|
let overlap = -2;
|
||||||
|
let topmove = false;
|
||||||
|
// Check if this works
|
||||||
|
|
||||||
|
if (i < boxes.length - 1) {
|
||||||
|
let nextrect = boxes[i + 1].getBoundingClientRect();
|
||||||
|
overlap = thisrect.bottom - nextrect.top;
|
||||||
|
} else if (thisrect.bottom > containerrect.bottom) {
|
||||||
|
overlap = thisrect.bottom - containerrect.bottom;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (thisrect.bottom > containerrect.bottom) {
|
||||||
|
topmove = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
// -1 for catching lines that perfectly close up on each other
|
||||||
|
overlap >= -1 &&
|
||||||
|
!(window.getComputedStyle(element).display === "none")
|
||||||
|
) {
|
||||||
|
let newlength = 0;
|
||||||
|
if (overlap >= 0) newlength = thisrect.height - overlap;
|
||||||
|
else newlength = thisrect.height - lineheight;
|
||||||
|
if (newlength % (lineheight * 3) <= 2) newlength -= lineheight;
|
||||||
|
let remainder = newlength % lineheight;
|
||||||
|
newlength = newlength - remainder;
|
||||||
|
|
||||||
|
// Line clamping for Marginals
|
||||||
|
if (element.classList.contains("ha-marginalbox")) {
|
||||||
|
let marginals = element.querySelectorAll(".ha-marginal");
|
||||||
|
let h = 0;
|
||||||
|
for (let m of marginals) {
|
||||||
|
let cr = m.getBoundingClientRect();
|
||||||
|
let eh = cr.bottom - cr.top;
|
||||||
|
h += eh;
|
||||||
|
if (h >= newlength) {
|
||||||
|
let lines = Math.floor(eh / lineheight);
|
||||||
|
let cutoff = Math.floor((h - newlength) / lineheight);
|
||||||
|
m.style.cssText +=
|
||||||
|
"-webkit-line-clamp: " + (lines - cutoff) + ";";
|
||||||
|
m.style.cssText += "line-clamp: " + (lines - cutoff) + ";";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
collapsedboxes.push(element);
|
||||||
|
collapsebox(element, newlength, lineheight);
|
||||||
|
addbuttoncaollapsebox(element, newlength, hoverfunction, topmove);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const marginalboxwidthset = function (containerid, classes) {
|
||||||
|
let lt = document.getElementById(containerid);
|
||||||
|
if (lt !== null) {
|
||||||
|
let mg = lt.querySelectorAll(classes);
|
||||||
|
if (mg.length > 0) {
|
||||||
|
let ltbcr = lt.getBoundingClientRect();
|
||||||
|
let mgbcr = mg[0].getBoundingClientRect();
|
||||||
|
let nw = ltbcr.right - mgbcr.left - 20;
|
||||||
|
|
||||||
|
for (let element of mg) {
|
||||||
|
element.style.width = nw + "px";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const clearcollapsedboxes = function () {
|
||||||
|
let elements = document.querySelectorAll(".ha-marginalbox");
|
||||||
|
elements.forEach((element) => {
|
||||||
|
element.removeAttribute("style");
|
||||||
|
});
|
||||||
|
collapsedboxes.forEach((element) => {
|
||||||
|
element.classList.remove("ha-expanded-box");
|
||||||
|
element.classList.remove("ha-collapsed-box");
|
||||||
|
element.outerHTML = element.outerHTML;
|
||||||
|
});
|
||||||
|
collapsedboxes = [];
|
||||||
|
elements = document.querySelectorAll(".ha-btn-collapsed-box");
|
||||||
|
elements.forEach((element) => {
|
||||||
|
element.remove();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const resetall = function () {
|
||||||
|
clearcollapsedboxes();
|
||||||
marginalboxwidthset("ha-letterbody", ".ha-marginalbox");
|
marginalboxwidthset("ha-letterbody", ".ha-marginalbox");
|
||||||
marginalboxwidthset("ha-register-body", ".ha-letlinks");
|
marginalboxwidthset("ha-register-body", ".ha-letlinks");
|
||||||
collapseboxes();
|
startup_marginals();
|
||||||
|
};
|
||||||
|
|
||||||
|
const collapseboxes = function () {
|
||||||
|
overlappingcollapsebox(".ha-letlinks", true, "ha-register-body");
|
||||||
|
overlappingcollapsebox(".ha-marginalbox", true, "ha-letterbody");
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener("resize", function () {
|
||||||
|
clearTimeout(debounce_resize);
|
||||||
|
debounce_resize = setTimeout(resetall, 17);
|
||||||
|
});
|
||||||
|
|
||||||
|
marginalboxwidthset("ha-letterbody", ".ha-marginalbox");
|
||||||
|
marginalboxwidthset("ha-register-body", ".ha-letlinks");
|
||||||
|
collapseboxes();
|
||||||
};
|
};
|
||||||
|
|
||||||
export { startup_marginals };
|
export { startup_marginals };
|
||||||
|
|||||||
Reference in New Issue
Block a user