diff --git a/Other/account-options-template.js b/Other/account-options-template.js new file mode 100644 index 0000000..37e9e4c --- /dev/null +++ b/Other/account-options-template.js @@ -0,0 +1,84 @@ +"use strict"; +class AccountOptions extends HTMLElement { + constructor() { + super() + this.attachShadow({ mode: "open" }) + } + + connectedCallback() { + this.shadowRoot.innerHTML = html` + + + ` + const style = document.createElement("style") + style.innerHTML = css` + .account-options { + display: flex; + } + + .login-button { + height: 100%; + } + + .account-image { + height: 48px; + aspect-ratio: 1/1; + object-fit: cover; + border-radius: 4px; + } + + .account-link { + display: flex; + height: 100%; + justify-content: center; + width: 100px; + column-gap: 8px; + align-items: center; + }` + this.shadowRoot.appendChild(style) + defineAndInject(this, this.shadowRoot) + + this.loginButton.addEventListener("click", () => { + let loginSignup = document.querySelector("login-signup"); + if (loginSignup) { + loginSignup.remove() + } + loginSignup = createFromData("login-signup"); + document.body.appendChild(loginSignup) + loginSignup.addEventListener("finished", (d, e) => { + this.loginCallback() + }) + loginSignup.open() + }) + + isLoggedIn().then((loggedIn) => { + if (loggedIn) { + this.loginCallback() + } + }) + } + + trySignout() { + if (typeof signout === "undefined") { + signout() + } + } + + async loginCallback() { + this.loginButton.style.display = "none" + this.accountOptions.style.display = "flex" + + const accountData = await getAccountData() + if (accountData) { + + } + } +} + +customElements.define("account-options", AccountOptions) diff --git a/Other/component-registrar.js b/Other/component-registrar.js index c75c44b..8376523 100644 --- a/Other/component-registrar.js +++ b/Other/component-registrar.js @@ -1,3 +1,4 @@ +"use strict"; /** * Will instance the given element, returning it so that it can be inserted into the DOM. * @param {HTMLElement} name Element to be instanced @@ -8,7 +9,9 @@ function createFromData(name, data = null) { let element = document.createElement(name) if (data) { for (const [key, value] of Object.entries(data)) { - element.setAttribute(key, value.toString()) + if (value !== undefined) { + element.setAttribute("data-" + key, value.toString()) + } } } // TODO: This causes issues with some nodes wanting to be IN the dom before methods can be used on them diff --git a/Other/content-warning-template.js b/Other/content-warning-template.js index ac63a8a..36376f8 100644 --- a/Other/content-warning-template.js +++ b/Other/content-warning-template.js @@ -1,3 +1,5 @@ +"use strict"; + class ContentWarning extends HTMLElement { constructor() { super() diff --git a/Other/header-template.js b/Other/header-template.js index 65b59d0..3ffdd3c 100644 --- a/Other/header-template.js +++ b/Other/header-template.js @@ -1,3 +1,4 @@ +"use strict"; class SubliminalHeader extends HTMLElement { constructor() { super() @@ -8,7 +9,9 @@ class SubliminalHeader extends HTMLElement { this.shadowRoot.innerHTML = html`
- Dog + + +

Subliminal


` @@ -64,8 +58,8 @@ class SubliminalHeader extends HTMLElement { line-height: 64px; white-space: nowrap; } - - a { + + nav > a { align-self: center; white-space: nowrap; /*Click hitboxes*/ @@ -73,6 +67,11 @@ class SubliminalHeader extends HTMLElement { padding-bottom: 24px; } + nav > a[current] { + background-color: #0074d90d; + border-bottom: 4px solid var(--input-hilight); + } + span, div { font-family: Arial, Helvetica, sans-serif; } @@ -80,24 +79,23 @@ class SubliminalHeader extends HTMLElement { p, a { font-family: Arial, Helvetica, sans-serif; font-size: 110%; - } + } - a[current] { - background-color: #0074d90d; - border-bottom: 4px solid var(--input-hilight); + .logo-button { + display: flex; + cursor: pointer; } - img { + .logo { align-self: center; - cursor: pointer; transition: .2s transform; } - - img:hover { + + .logo:hover { transform: rotate(10deg) scale(1.5); } - - img:active { + + .logo:active { transform: rotate(8deg) scale(1.1); } @@ -113,7 +111,7 @@ class SubliminalHeader extends HTMLElement { column-gap: 8px; padding: 8px; } - + hr { border: none; border-top: 1px solid gray; @@ -129,12 +127,12 @@ class SubliminalHeader extends HTMLElement { display: none; } - img { + .logo { width: 48px; height: 48px; } - a { + nav > a { font-size: 3.4vw; flex: 1 1 auto; white-space: nowrap; @@ -163,7 +161,7 @@ class SubliminalHeader extends HTMLElement { } @media(prefers-color-scheme: dark) { - img { + .logo { filter: invert(1); } @@ -181,19 +179,12 @@ class SubliminalHeader extends HTMLElement { } } - ;(async function(_this){ - if (_this.getAttribute("nologin") || typeof isLoggedIn === "undefined" || typeof isLoggedIn !== "function") { - // TODO: Potentially change right to just display 'The coolest crowdsourced anthology on the web' (delete buttons) - _this.right.style.display = 'none' - console.warn("WARNING: Page has not imported account.js or is requesting nologin. Login UI disabled") - return - } - - if (await isLoggedIn()) { - _this.loginButton.style.display = "none" - _this.logoutButton.style.display = "block" - } - })(this) + if (this.getAttribute("nologin") || typeof isLoggedIn === "undefined" || typeof isLoggedIn !== "function") { + // TODO: Potentially change right to just display 'The coolest crowdsourced anthology on the web' (delete buttons) + this.right.style.display = 'none' + console.warn("WARNING: Page has not imported account.js or is requesting nologin. Login UI disabled") + return + } } } diff --git a/Other/login-template.js b/Other/login-template.js index d39825a..134a60a 100644 --- a/Other/login-template.js +++ b/Other/login-template.js @@ -1,3 +1,4 @@ +"use strict"; class LoginSignup extends HTMLElement { #nocancel = false @@ -203,13 +204,13 @@ class LoginSignup extends HTMLElement { } } - //Impossible to log in without code, GUID can be retrieved though async signin(username, email) { + let signinData = null try { const signinResponse = await fetch(serverBaseAddress + "/auth/signin", { method: "POST", headers: { 'Content-Type': 'application/json' }, - body: JSON.stringify({ Username: username, Email: email }) + body: JSON.stringify({ username, email }) }) if (!signinResponse.ok) { @@ -217,19 +218,16 @@ class LoginSignup extends HTMLElement { return } - const signinData = await signinResponse.json() - - localStorage.accountUsername = username - localStorage.accountEmail = email - //localStorage.accountGuid = signinData.guid - - showMyAccount() + signinData = await signinResponse.json() + //sessionStorage.accountId = signinData.id + //sessionStorage.accountToken = signinData.token + this.login.close() } catch (error) { this.confirmFail(error) } - const finishEvent = new CustomEvent("finished", { type: "signin" }) + const finishEvent = new CustomEvent("finished", { type: "signin", data: signinData }) this.dispatchEvent(finishEvent) } diff --git a/Other/notification-template.js b/Other/notification-template.js index 13ce307..5e269db 100644 --- a/Other/notification-template.js +++ b/Other/notification-template.js @@ -1,3 +1,4 @@ +"use strict"; class SubliminalNotification extends HTMLElement { constructor() { super() diff --git a/Other/poem-canvas-editor-template.js b/Other/poem-canvas-editor-template.js new file mode 100644 index 0000000..02e107c --- /dev/null +++ b/Other/poem-canvas-editor-template.js @@ -0,0 +1,359 @@ +"use strict"; +class PoemCanvasEditor extends HTMLElement { + document = null + canvasScale = 1.5 + cursor = true + + constructor() { + super() + this.attachShadow({ mode: "open" }) + } + + connectedCallback() { + this.shadowRoot.innerHTML = html` + + + + + + +
+ + + + + + + +
` + const style = document.createElement("style") + style.innerHTML = css` + #editorCanvas { + display: block; + width: 100%; + height: 100%; + cursor: text; + } + #editorCanvas:focus { + outline: none; + } + #editorInput { + position: absolute; + opacity: 0; + pointer-events: none; + } + #editorContext { + display: flex; + visibility: hidden; + transition: left 0.1s ease 0s; + position: fixed; + width: 200px; + background-color: var(--button-transparent); + backdrop-filter: blur(5px); + border-radius: 4px; + box-shadow: 0px 0px 10px #656565; + border: 1px solid darkgray; + flex-direction: column; + padding: 4px; + row-gap: 4px; + } + #editorContext > button { + height: 32px; + background-color: transparent; + border: 1px solid var(--button-opaque); + display: flex; + align-items: center; + column-gap: 8px; + transition: .05 transform; + user-select: none; + } + #editorContext > button:hover { + background-color: var(--button-opaque-hover); + } + #editorContext > button:active { + transform: scale(0.98); + } + #editorContext > button > svg { + opacity: 0.6; + fill: var(--text-colour); + } + #editorContext > button > span { + font-weight: bold; + opacity: 0.4; + flex-grow: 1; + text-align: end; + } + .suggestions { + width: 192px; + position: absolute; + background: #bdbdbd; + z-index: 3; + border-radius: 4px; + top: 0px; + display: flex; + background-color: var(--button-transparent); + flex-direction: column; + border: 1px solid gray; + overflow: clip; + overflow-y: scroll; + } + .suggestions-item { + padding: 8px; + display: flex; + } + .suggestions-item > span:nth-child(1) { + flex-grow: 1; + } + .suggestions-item > span:nth-child(2) { + font-size: 10px; + opacity: 0.6; + }` + this.shadowRoot.appendChild(style) + defineAndInject(this, this.shadowRoot) + + // Apply attributes + this.document = new EditorDocument(1.0, 18) + this.canvasScale = Number(this.getAttribute("scale") || 1.5) + this.cursor = !!this.getAttribute("cursor") + this.tabIndex = "0" + + // Event listeners + this.editorInput.addEventListener("keydown", (event) => { + if (!(event.ctrlKey && event.key.toLowerCase() === 'v')) { + this.editorCanvas.dispatchEvent(new event.constructor(event.type, event)) + } + // Uses clipboard API if possible + else if (navigator.clipboard.readText) { + navigator.clipboard.readText() + .then((paste) => { + this.document.insertText(paste) + this.document.renderCanvasData(this.editorCanvas) + }) + .catch((error) => { + alert('Could not paste text. Please give the site access to the clipboard') + }) + } + }) + this.editorInput.addEventListener("paste", (event) => { + // WORKAROUND: For browsers not supporting the navigator clipboard readText API + // we let the paste (usually handled from) keydown event trickle through to an onpaste + // event, catch the text, and then pass it to the editor + if (navigator.clipboard.readText) return + const paste = (event.clipboardData || window.clipboardData).getData('text') + this.document.insertText(paste) + this.document.renderCanvasData(this.editorCanvas) + }) + this.editorCanvas.addEventListener("contextmenu", (event) => { + this.editorContext.style.left = (event.clientX) + 'px' + this.editorContext.style.top = (event.clientY) + 'px' + this.editorContext.style.visibility = 'visible' + return event.preventDefault() + }) + this.editorCanvas.addEventListener("mousedown", (event) => { + if (event.button === 2) return + event.preventDefault() + // TODO: handle virtkeyboard geometry in scrolldown + this.editorInput.focus() + navigator.virtualKeyboard?.show() + this.editorCanvas['pressed'] = true + this.document.clearSelection() + this.document.position = this.document.realToTextPosition(event.offsetX, event.offsetY, this.editorCanvas) + this.document.renderCanvasData(this.editorCanvas, this.cursor) + this.editorContext.style.visibility = 'hidden' + }) + this.editorCanvas.addEventListener("mouseup", (event) => { + this.editorCanvas['pressed'] = false + }) + this.editorCanvas.addEventListener("mousemove", (event) => { + if (this.editorCanvas['pressed']) { + let endPosition = this.document.realToTextPosition(event.offsetX, event.offsetY, this.editorCanvas) + this.document.selection.start = this.document.position > endPosition ? endPosition : this.document.position + this.document.selection.end = this.document.position > endPosition ? this.document.position : endPosition + this.document.renderCanvasData(this.editorCanvas, this.cursor) + } + }) + this.editorCanvas.addEventListener("dblclick", (event) => { + // TODO: Select word on mobile handles + }) + this.editorCanvas.addEventListener("keydown", (event) => { + if (event.key === 'Backspace') { + this.document.deleteText() + } + else if (event.key === 'Delete') { + this.document.deleteText(-1) + } + else if (event.key === 'Enter') { + this.document.addNewLine() + // TODO: only scroll down if cursor becomes off the screen + setTimeout(() => window.scrollTo(0, 1e5), 10) + } + else if (event.key === 'ArrowLeft' || event.key === 'ArrowRight' || event.key === 'ArrowUp' || event.key === 'ArrowDown') { + this.document.movePosition( + event.key == 'ArrowLeft' + ? positionMovements.left + : event.key == 'ArrowRight' + ? positionMovements.right + : event.key == 'ArrowUp' + ? positionMovements.up + : positionMovements.down, event.shiftKey) + } + else if (event.key == 'Home') { + throw new Error('\'Home\' control key is not implemented') + } + else if (event.key == 'End') { + throw new Error('\'End\' control key is not implemented') + } + else if (event.key == 'Tab') { + this.document.insertText('\t') + } + else if (event.key === 'Shift' || event.key.length > 1) { + return + } + else if (event.ctrlKey) { + switch (event.key.toLowerCase()) { + case 'a': { + this.document.selectAll() + break + } + case 'x': { + navigator.clipboard.writeText(this.document.getSelectionText()) + this.document.deleteSelection() + break + } + case 'c': { + navigator.clipboard.writeText(this.document.getSelectionText()) + break + } + case 'v': { + if (navigator.clipboard.readText) { // HACK: See handler in input catcher + navigator.clipboard.readText().then((clipText) => this.document.insertText(clipText)) + } + break + } + } + } + else { + this.document.insertText(event.key) + } + this.document.renderCanvasData(this.editorCanvas,this.cursor) + }) + this.editorCanvas.addEventListener("blur", (event) => { + this.document.renderCanvasData(this.editorCanvas, false) + }) + this.editorCanvas.addEventListener("focus", (event) => { + this.document.renderCanvasData(this.editorCanvas, this.cursor) + }) + + // External handlers + this.addEventListener("blur", () => { + this.editorContext.style.visibility = 'hidden' + }) + window.addEventListener("resize", () => { + this.updateCanvas() + }) + this.updateCanvas() + this.syncInputCatcher() + } + + useDocument(document) { + this.document = document + this.updateCanvas() + this.syncInputCatcher() + } + + syncInputCatcher() { + // Ideally input catcher is a 100% plaintext synced version of the canvas render + // output, manual intervention is required for slight discrepancies between how + // the real input catcher and emulated canvas text editor work + this.editorInput.textContent = this.document.getText() + this.editorInput.style.width = this.editorCanvas.offsetWidth + "px" + this.editorInput.style.height = this.editorCanvas.offsetHeight + "px" + } + + updateCanvas() { + this.editorCanvas.width = this.editorCanvas.offsetWidth * this.canvasScale + this.editorCanvas.height = this.editorCanvas.offsetHeight * this.canvasScale + this.document.renderCanvasData(this.editorCanvas, this.cursor) + } +} + +customElements.define("poem-canvas-editor", PoemCanvasEditor) \ No newline at end of file diff --git a/Other/poem-ghost-template.js b/Other/poem-ghost-template.js index eaa383f..1e8c91a 100644 --- a/Other/poem-ghost-template.js +++ b/Other/poem-ghost-template.js @@ -1,3 +1,4 @@ +"use strict"; class PoemGhost extends HTMLElement { constructor() { super() diff --git a/Other/poem-load-cover-template.js b/Other/poem-load-cover-template.js index 17bdfae..1668587 100644 --- a/Other/poem-load-cover-template.js +++ b/Other/poem-load-cover-template.js @@ -1,3 +1,4 @@ +"use strict"; class PoemLoadCover extends HTMLElement { constructor() { super() diff --git a/Other/purgatory-entry-template.js b/Other/purgatory-entry-template.js index 7610518..eba1190 100644 --- a/Other/purgatory-entry-template.js +++ b/Other/purgatory-entry-template.js @@ -1,3 +1,4 @@ +"use strict"; class PurgatoryEntry extends HTMLElement { constructor() { super() @@ -8,23 +9,23 @@ class PurgatoryEntry extends HTMLElement { // TODO: Clear up slightly XSS prone string interpolations this.shadowRoot.innerHTML = html`
- ${this.getAttribute('notification') + ${this.getAttribute('data-notification') ? html`

- ${this.getAttribute('notification')} + ${this.getAttribute('data-notification')}

` : ''} - ${this.getAttribute('tooltip') + ${this.getAttribute('data-tooltip') ? html`

` : ''}

- By ${this.getAttribute('author')} -
+ By ${this.getAttribute('data-author')} +
-
+
@@ -86,6 +87,12 @@ class PurgatoryEntry extends HTMLElement { flex-direction: row; margin-left: 8px; margin-right: 8px; + column-gap: 8px; + } + + .vote-container { + display: flex; + flex: 1; } svg { @@ -106,14 +113,14 @@ class PurgatoryEntry extends HTMLElement { defineAndInject(this, this.shadowRoot) this.setAttribute("tabindex", "0") - if (this.getAttribute("tooltip")) { + if (this.getAttribute("data-tooltip")) { } - this.poemPreview.textContent = this.getAttribute("preview") - this.poemName.textContent = this.getAttribute("name") - this.approves.textContent = this.getAttribute("approves") - this.vetoes.textContent = this.getAttribute("vetoes") + this.poemPreview.textContent = this.getAttribute("data-preview") + this.poemName.textContent = this.getAttribute("data-name") + this.approves.textContent = this.getAttribute("data-approves") + this.vetoes.textContent = this.getAttribute("data-vetoes") if (this.getAttribute("new")) { setTimeout(() => { @@ -132,12 +139,11 @@ class PurgatoryEntry extends HTMLElement { this.classList.remove("entry-new") }, 1600) } - this.onclick = function() { - const guid = this.getAttribute("guid") - if (guid) { - window.location.href = "./purgatory-poem?guid=" + guid + this.addEventListener("click", function() { + if (this.dataset.id) { + window.location.href = "./purgatory-poem?id=" + this.dataset.id } - } + }) } } diff --git a/Other/report-template.js b/Other/report-template.js index 0b0790f..3cfb34a 100644 --- a/Other/report-template.js +++ b/Other/report-template.js @@ -1,3 +1,4 @@ +"use strict"; class ReportPopup extends HTMLElement { constructor() { super() diff --git a/Other/select-template.js b/Other/select-template.js index 550367a..abaef8d 100644 --- a/Other/select-template.js +++ b/Other/select-template.js @@ -1,3 +1,4 @@ +"use strict"; class SubliminalSelect extends HTMLElement { constructor() { super() diff --git a/Resources/AbbstraktDog.ico b/Resources/AbbstraktDog.ico new file mode 100644 index 0000000..e337369 Binary files /dev/null and b/Resources/AbbstraktDog.ico differ diff --git a/Resources/AbbstrakDog.png b/Resources/AbbstraktDog.png similarity index 100% rename from Resources/AbbstrakDog.png rename to Resources/AbbstraktDog.png diff --git a/SubliminalServer/.config/dotnet-tools.json b/SubliminalServer/.config/dotnet-tools.json index d9d129c..071b4ee 100644 --- a/SubliminalServer/.config/dotnet-tools.json +++ b/SubliminalServer/.config/dotnet-tools.json @@ -3,7 +3,7 @@ "isRoot": true, "tools": { "dotnet-ef": { - "version": "8.0.3", + "version": "8.0.8", "commands": [ "dotnet-ef" ] diff --git a/SubliminalServer/ApiModel/AvatarRequest.cs b/SubliminalServer/ApiModel/AvatarRequest.cs new file mode 100644 index 0000000..0c4bb2c --- /dev/null +++ b/SubliminalServer/ApiModel/AvatarRequest.cs @@ -0,0 +1,3 @@ +namespace SubliminalServer.ApiModel; + +public record AvatarRequest(string MimeType, string Data); \ No newline at end of file diff --git a/SubliminalServer/AuthorizationMiddleware.cs b/SubliminalServer/AuthorizationMiddleware.cs deleted file mode 100644 index d2e99a3..0000000 --- a/SubliminalServer/AuthorizationMiddleware.cs +++ /dev/null @@ -1,29 +0,0 @@ -using Microsoft.EntityFrameworkCore; - -namespace SubliminalServer; - -public class AuthorizationMiddleware -{ - private readonly RequestDelegate nextRequest; - private readonly DatabaseContext databaseContext; - - public AuthorizationMiddleware(RequestDelegate nextReq, DatabaseContext db) - { - nextRequest = nextReq; - databaseContext = db; - } - - public async Task Invoke(HttpContext context) - { - var accountToken = context.Request.Cookies["Token"]; - - if (accountToken != null) - { - var account = await databaseContext.Accounts - .SingleOrDefaultAsync(account => account.Token == accountToken); - context.Items["Account"] = account; - } - - await nextRequest(context); - } -} \ No newline at end of file diff --git a/SubliminalServer/DataModel/Account/AccountAddress.cs b/SubliminalServer/DataModel/Account/AccountClient.cs similarity index 53% rename from SubliminalServer/DataModel/Account/AccountAddress.cs rename to SubliminalServer/DataModel/Account/AccountClient.cs index c94172d..24d6e45 100644 --- a/SubliminalServer/DataModel/Account/AccountAddress.cs +++ b/SubliminalServer/DataModel/Account/AccountClient.cs @@ -5,18 +5,28 @@ namespace SubliminalServer.DataModel.Account; [PrimaryKey(nameof(Id))] -public class AccountAddress +public class AccountClient { - // Unique, Primary key [Required] [DatabaseGenerated(DatabaseGeneratedOption.Identity)] public int Id { get; set; } + public string IpAddress { get; set; } + public string UserAgent { get; set; } public DateTime LastUsed { get; set; } - // Foreign key AccountData - public int AccountKey { get; set; } - + public int AccountId { get; set; } // Navigation property to the parent AccounData public AccountData Account { get; set; } + + // EFCore constructor - Don't use + public AccountClient() { } + + public AccountClient(string ipAddress, string userAgent, int accountId, DateTime? lastUsed = null) + { + IpAddress = ipAddress; + UserAgent = userAgent; + AccountId = accountId; + LastUsed = lastUsed ?? DateTime.Now; + } } \ No newline at end of file diff --git a/SubliminalServer/DataModel/Account/AccountData.cs b/SubliminalServer/DataModel/Account/AccountData.cs index b22caf0..601229c 100644 --- a/SubliminalServer/DataModel/Account/AccountData.cs +++ b/SubliminalServer/DataModel/Account/AccountData.cs @@ -20,7 +20,7 @@ public class AccountData : AccountProfile // Navigation property [JsonIgnore] - public List KnownIPs { get; set; } + public List KnownIPs { get; set; } // Navigation property [JsonIgnore] public List Drafts { get; set; } @@ -35,7 +35,7 @@ public AccountData(string username, string email, DateTime joinDate, string toke { Token = token; Email = email; - KnownIPs = new List(); + KnownIPs = new List(); Drafts = new List(); Blocked = new List(); LikedPoems = new List(); diff --git a/SubliminalServer/DataModel/Configurations/AccountAddressConfiguration.cs b/SubliminalServer/DataModel/Configurations/AccountAddressConfiguration.cs deleted file mode 100644 index f061dc5..0000000 --- a/SubliminalServer/DataModel/Configurations/AccountAddressConfiguration.cs +++ /dev/null @@ -1,20 +0,0 @@ -using Microsoft.EntityFrameworkCore; -using Microsoft.EntityFrameworkCore.Metadata.Builders; -using SubliminalServer.DataModel.Account; - -namespace SubliminalServer.DataModel.Configurations; - -public class AccountAddressConfiguration : IEntityTypeConfiguration -{ - public void Configure(EntityTypeBuilder builder) - { - builder.HasKey(address => address.Id); - builder.Property(address => address.Id) - .ValueGeneratedOnAdd(); - - // One to many (AccountData) - builder.HasOne(address => address.Account) - .WithMany(account => account.KnownIPs) - .HasForeignKey(address => address.Id); - } -} diff --git a/SubliminalServer/DataModel/Configurations/AccountClientConfiguration.cs b/SubliminalServer/DataModel/Configurations/AccountClientConfiguration.cs new file mode 100644 index 0000000..4912a5a --- /dev/null +++ b/SubliminalServer/DataModel/Configurations/AccountClientConfiguration.cs @@ -0,0 +1,16 @@ +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Metadata.Builders; +using SubliminalServer.DataModel.Account; + +namespace SubliminalServer.DataModel.Configurations; + +public class AccountClientConfiguration : IEntityTypeConfiguration +{ + public void Configure(EntityTypeBuilder builder) + { + // One to many (AccountData) + builder.HasOne(client => client.Account) + .WithMany(account => account.KnownIPs) + .HasForeignKey(client => client.AccountId); + } +} diff --git a/SubliminalServer/DataModel/Configurations/AccountDataConfiguration.cs b/SubliminalServer/DataModel/Configurations/AccountDataConfiguration.cs index 579f347..bd5cc21 100644 --- a/SubliminalServer/DataModel/Configurations/AccountDataConfiguration.cs +++ b/SubliminalServer/DataModel/Configurations/AccountDataConfiguration.cs @@ -34,7 +34,7 @@ public void Configure(EntityTypeBuilder builder) // One to many IPs (AccountBadge) builder.HasMany(mainEntity => mainEntity.KnownIPs) .WithOne(address => address.Account) - .HasForeignKey(address => address.AccountKey); + .HasForeignKey(address => address.AccountId); // Many to many AccountData (Blocked), AccountData (BlockedBy) builder.HasMany(account => account.Blocked) diff --git a/SubliminalServer/DataModel/Purgatory/PurgatoryEntry.cs b/SubliminalServer/DataModel/Purgatory/PurgatoryEntry.cs index fedd3cc..4eaeb93 100644 --- a/SubliminalServer/DataModel/Purgatory/PurgatoryEntry.cs +++ b/SubliminalServer/DataModel/Purgatory/PurgatoryEntry.cs @@ -56,5 +56,6 @@ public class PurgatoryEntry : IDatabasePoem public int Approves { get; set; } public int Vetoes { get; set; } public DateTime DateCreated { get; set; } + // TODO: Make pick object instead to track who assigned what pick public bool Pick { get; set; } } \ No newline at end of file diff --git a/SubliminalServer/DataModel/Report/AccountReport.cs b/SubliminalServer/DataModel/Report/AccountReport.cs index c52ecf4..b78678a 100644 --- a/SubliminalServer/DataModel/Report/AccountReport.cs +++ b/SubliminalServer/DataModel/Report/AccountReport.cs @@ -9,6 +9,7 @@ public class AccountReport : Report [Required] [ForeignKey(nameof(Account))] public int AccountId { get; set; } + // Navigation property to reported account public AccountData Account { get; set; } } \ No newline at end of file diff --git a/SubliminalServer/DataModel/Report/PurgatoryReport.cs b/SubliminalServer/DataModel/Report/PurgatoryReport.cs index 916b65e..d9cf2be 100644 --- a/SubliminalServer/DataModel/Report/PurgatoryReport.cs +++ b/SubliminalServer/DataModel/Report/PurgatoryReport.cs @@ -9,5 +9,6 @@ public class PurgatoryReport : Report [Required] [ForeignKey(nameof(Poem))] public int PoemId { get; set; } + // Navigation property to reporteed poem public PurgatoryEntry Poem { get; set; } } \ No newline at end of file diff --git a/SubliminalServer/DataModel/Report/Report.cs b/SubliminalServer/DataModel/Report/Report.cs index 141cb05..3bf93fb 100644 --- a/SubliminalServer/DataModel/Report/Report.cs +++ b/SubliminalServer/DataModel/Report/Report.cs @@ -16,7 +16,7 @@ public class Report // Foreign key Account Data [Required] [ForeignKey(nameof(Reporter))] - public int ReporterKey { get; set; } + public int ReporterId { get; set; } public AccountData Reporter { get; set; } [MaxLength(300)] diff --git a/SubliminalServer/DatabaseContext.cs b/SubliminalServer/DatabaseContext.cs index 7e6160b..56a2a41 100644 --- a/SubliminalServer/DatabaseContext.cs +++ b/SubliminalServer/DatabaseContext.cs @@ -11,7 +11,7 @@ public class DatabaseContext : DbContext { public DbSet Accounts { get; set; } public DbSet AccountBadges { get; set; } - public DbSet AccountAddressInfos { get; set; } + public DbSet AccountClients { get; set; } public DbSet PurgatoryDrafts { get; set; } public DbSet PurgatoryEntries { get; set; } @@ -32,7 +32,7 @@ public DatabaseContext(DbContextOptions options) : base(options protected override void OnModelCreating(ModelBuilder modelBuilder) { modelBuilder.ApplyConfiguration(new AccountDataConfiguration()); - modelBuilder.ApplyConfiguration(new AccountAddressConfiguration()); + modelBuilder.ApplyConfiguration(new AccountClientConfiguration()); modelBuilder.ApplyConfiguration(new AccountBadgeConfiguration()); modelBuilder.ApplyConfiguration(new PurgatoryEntryConfiguration()); modelBuilder.ApplyConfiguration(new PurgatoryTagConfiguration()); diff --git a/SubliminalServer/EnsureAuthorizationMiddleware.cs b/SubliminalServer/EnsureAuthorizationMiddleware.cs deleted file mode 100644 index b613820..0000000 --- a/SubliminalServer/EnsureAuthorizationMiddleware.cs +++ /dev/null @@ -1,25 +0,0 @@ -using SubliminalServer.DataModel.Account; - -namespace SubliminalServer; - -public class EnsureAuthorizationMiddleware -{ - private readonly RequestDelegate nextRequest; - - public EnsureAuthorizationMiddleware(RequestDelegate nextReq) - { - nextRequest = nextReq; - } - - public async Task Invoke(HttpContext context) - { - var account = context.Items["Account"]; - if (account is not AccountData) - { - context.Response.StatusCode = StatusCodes.Status401Unauthorized; - await context.Response.WriteAsync("This endpoint requires account authorisation."); - } - - await nextRequest(context); - } -} \ No newline at end of file diff --git a/SubliminalServer/EnsuredAuthorizationMiddleware.cs b/SubliminalServer/EnsuredAuthorizationMiddleware.cs new file mode 100644 index 0000000..ff0b82f --- /dev/null +++ b/SubliminalServer/EnsuredAuthorizationMiddleware.cs @@ -0,0 +1,41 @@ +using Microsoft.EntityFrameworkCore; +using SubliminalServer.DataModel.Account; + +namespace SubliminalServer; + +public class EnsuredAuthorizationMiddleware +{ + private readonly RequestDelegate nextRequest; + + public EnsuredAuthorizationMiddleware(RequestDelegate nextRequest) + { + this.nextRequest = nextRequest; + } + + public async Task InvokeAsync(HttpContext context, DatabaseContext database) + { + var token = GetRequestToken(context); + if (token is null || await GetTokenAccount(token, database) is not { } account) + { + context.Response.StatusCode = StatusCodes.Status401Unauthorized; + await context.Response.WriteAsJsonAsync(new { Message = "Invalid token provided in auth header" }); + return; + } + + context.Items["Account"] = account; + await nextRequest(context); + } + + private static async Task GetTokenAccount(string token, DatabaseContext database) + { + var account = await database.Accounts.FirstOrDefaultAsync(account => account.Token == token); + return account; + } + + private static string? GetRequestToken(HttpContext context) + { + var token = context.Request.Headers.Authorization.FirstOrDefault() + ?? context.Request.Cookies["Token"] ?? context.Request.Query["token"].FirstOrDefault(); + return token; + } +} \ No newline at end of file diff --git a/SubliminalServer/Extensions.cs b/SubliminalServer/Extensions.cs new file mode 100644 index 0000000..e500cd2 --- /dev/null +++ b/SubliminalServer/Extensions.cs @@ -0,0 +1,17 @@ +namespace SubliminalServer; + +using System.Text.RegularExpressions; + +public static class Extensions +{ + public delegate Regex PathMatchRegex(); + public static RouteHandlerBuilder UseMiddleware(this RouteHandlerBuilder builder, WebApplication app, PathMatchRegex predicate, params object?[] args) + { + app.UseWhen + ( + context => predicate().IsMatch(context.Request.Path), + appBuilder => appBuilder.UseMiddleware(args) + ); + return builder; + } +} \ No newline at end of file diff --git a/SubliminalServer/Program.Accounts.cs b/SubliminalServer/Program.Accounts.cs index 8685100..c5ea658 100644 --- a/SubliminalServer/Program.Accounts.cs +++ b/SubliminalServer/Program.Accounts.cs @@ -1,3 +1,4 @@ +using System.Text.RegularExpressions; using Microsoft.AspNetCore.Mvc; using SubliminalServer.ApiModel; using SubliminalServer.DataModel.Account; @@ -6,8 +7,12 @@ namespace SubliminalServer; -public static partial class Program +internal static partial class Program { + // /accounts/{id:int} + [GeneratedRegex(@"^\/accounts\/\d+\/report\/*$")] + private static partial Regex AccountReportEndpointRegex(); + private static void AddAccountEndpoints() { // Account action endpoints @@ -26,7 +31,7 @@ private static void AddAccountEndpoints() throw new NotImplementedException(); }); - httpServer.MapPost("/accounts/{accountId}/report", (int accountId, [FromBody] UploadableReport reportUpload, [FromServices] DatabaseContext database, HttpContext context) => + var reportEndpoint = httpServer.MapPost("/accounts/{accountId}/report", (int accountId, [FromBody] UploadableReport reportUpload, [FromServices] DatabaseContext database, HttpContext context) => { var validationIssues = new Dictionary(); if (reportUpload.Reason.Length > 300) @@ -53,60 +58,118 @@ private static void AddAccountEndpoints() } var account = (AccountData) context.Items["Account"]!; - /*var report = new Report() + var report = new AccountReport() { - ReporterKey = account.Id, - TargetKey = reportUpload.TargetKey, + ReporterId = account.Id, + AccountId = reportUpload.TargetKey, Reason = reportUpload.Reason, ReportType = reportUpload.ReportType, ReportTargetType = reportUpload.TargetType, DateCreated = DateTime.UtcNow }; - database.Reports.Add(report); - database.SaveChanges();*/ + database.AccountReports.Add(report); + database.SaveChanges(); return Results.Ok(); - }); - // TODO: Reimplement this - //rateLimitEndpoints.Add("/accounts/{accountId}/report", (1, TimeSpan.FromSeconds(60))); - //authRequiredEndpoints.Add("/accounts/{accountId}/report"); + }).UseMiddleware(httpServer, AccountReportEndpointRegex); + if (!httpServer.Environment.IsDevelopment()) + { + httpServer.UseWhen + ( + context => AccountReportEndpointRegex().IsMatch(context.Request.Path), + appBuilder => appBuilder.UseMiddleware(1, TimeSpan.FromSeconds(60)) + ); + } httpServer.MapPost("/accounts/{accountId}/unfollow", ([FromBody] string userId, HttpContext context) => { throw new NotImplementedException(); }); - // Personal account endpoints - httpServer.MapPost("/accounts/me/email", ([FromBody] string email, HttpContext context) => + httpServer.MapGet("/accounts/me", (HttpContext context, DatabaseContext dbContext) => { - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + return Results.Ok(account); }); - httpServer.MapPost("/accounts/me/pen-name", ([FromBody] string penName, HttpContext context) => + httpServer.MapPost("/accounts/me/email", async ([FromBody] string email, HttpContext context, DatabaseContext dbContext) => { - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + account!.Email = email; + await dbContext.SaveChangesAsync(); + return Results.Ok(); }); - httpServer.MapPost("/accounts/me/biography", ([FromBody] string biography, HttpContext context) => + httpServer.MapPost("/accounts/me/pen-name", async ([FromBody] string penName, HttpContext context, DatabaseContext dbContext) => { - - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + account!.PenName = penName; + await dbContext.SaveChangesAsync(); + return Results.Ok(); }); - httpServer.MapPost("/accounts/me/location", ([FromBody] string location, HttpContext context) => + httpServer.MapPost("/accounts/me/biography", async ([FromBody] string biography, HttpContext context, DatabaseContext dbContext) => { - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + account!.Biography = biography; + await dbContext.SaveChangesAsync(); + return Results.Ok(); }); - httpServer.MapPost("/accounts/me/role", ([FromBody] string role, HttpContext context) => + httpServer.MapPost("/accounts/me/location", async ([FromBody] string location, HttpContext context, DatabaseContext dbContext) => { - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + account!.Location = location; + await dbContext.SaveChangesAsync(); + return Results.Ok(); }); - httpServer.MapPost("/accounts/me/avatar", ([FromBody] string avatarUrl, HttpContext context) => + httpServer.MapPost("/accounts/me/role", async ([FromBody] string role, HttpContext context, DatabaseContext dbContext) => { - throw new NotImplementedException(); + var account = context.Items["Account"] as AccountData; + account!.Role = role; + await dbContext.SaveChangesAsync(); + return Results.Ok(); }); + + var allowedAvatarMimes = new Dictionary + { + { "image/png", ".png" }, + { "image/jpeg", ".jpg" }, + { "image/webp", ".webp" }, + { "image/gif", ".gif" } + }; + httpServer.MapPost("/accounts/me/avatar", async ([FromBody] AvatarRequest pictureRequest, HttpContext context, DatabaseContext dbContext) => + { + // Only allow user to access their own account + var account = (AccountData)context.Items["Account"]!; + + if (!allowedAvatarMimes.TryGetValue(pictureRequest.MimeType, out var fileExtension)) + { + return Results.BadRequest(new { Message = "Supplied image was not of a valid format" }); + } + var fileName = account.Id + fileExtension; + var savePath = Path.Combine(profileImagesDir.FullName, fileName); + var fileData = Convert.FromBase64String(pictureRequest.Data); + if (fileData.Length > 25e5) + { + return Results.BadRequest(new { Message = "Supplied image can not be more than 2.5MB" }); + } + + if (account.AvatarUrl is not null) + { + var previousFile = Path.Combine(profileImagesDir.FullName, account.AvatarUrl.Split("/").Last()); + File.Delete(previousFile); + } + + await File.WriteAllBytesAsync(savePath, fileData); + account.AvatarUrl = $"/profiles/avatars/{fileName}"; + await dbContext.SaveChangesAsync(); + return Results.Ok(); + }); + + // Protect personal endpoints with the auth required middleware to ensure they can not be called otherwise + authRequiredEndpoints.Add("/accounts/me"); } } \ No newline at end of file diff --git a/SubliminalServer/Program.Auth.cs b/SubliminalServer/Program.Auth.cs index 14db4d3..4d66d37 100644 --- a/SubliminalServer/Program.Auth.cs +++ b/SubliminalServer/Program.Auth.cs @@ -1,16 +1,17 @@ using Microsoft.AspNetCore.Mvc; using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Storage; using SubliminalServer.ApiModel; using SubliminalServer.DataModel.Account; namespace SubliminalServer; -public static partial class Program +internal static partial class Program { private static void AddAuthEndpoints() { //Creates a new account with a provided pen name, and then gives the client the credentials for their created account - httpServer.MapPost("/auth/signup", static ([FromBody] LoginDetails details, [FromServices] DatabaseContext database, HttpContext context) => + httpServer.MapPost("/auth/signup", async ([FromBody] LoginDetails details, [FromServices] DatabaseContext database, HttpContext context) => { if (!PermissibleUsernameRegex().IsMatch(details.Username)) { @@ -28,29 +29,25 @@ private static void AddAuthEndpoints() // TODO: Email validation, this will all be moved elsewhere var tokenString = GenerateToken(); var account = new AccountData(details.Username, details.Email, DateTime.UtcNow, tokenString); + database.Accounts.Add(account); + await database.SaveChangesAsync(); // Rate limit middleware should have passed us a nicely sanitised IP. Otherwise we will just fallback - var requestIp = context.Items["RealIp"] as string ?? context.Connection.RemoteIpAddress?.ToString(); + var requestIp = GetRequestIp(context); + var userAgent = context.Request.Headers.UserAgent.ToString(); if (requestIp is null) { return Results.Forbid(); } - account.KnownIPs.Add(new AccountAddress() - { - IpAddress = requestIp - }); - database.Accounts.Add(account); - database.SaveChanges(); + await UpdateAccountClients(requestIp, userAgent, account, database); - context.Response.Cookies.Append("Token", tokenString, new CookieOptions() - { - HttpOnly = true, - SameSite = SameSiteMode.Strict, - Expires = DateTimeOffset.UtcNow.AddMonths(1) - }); + + // Add auth cookie for subsequent requests + AppendTokenCookie(account.Token, httpServer, context); + // If for some reason they can not persist the cookie, we also send them the token so that they may save it somwehere // secure on certain platforms, such as a third party non-web client. - return Results.Ok(account.Token); + return Results.Ok(new { account.Token, account.Id}); }); if (!httpServer.Environment.IsDevelopment()) { @@ -58,8 +55,8 @@ private static void AddAuthEndpoints() sizeLimitEndpoints.Add("/auth/signup", PayloadSize.FromKilobytes(5)); } - // Allows a user to signin and receive account data - httpServer.MapPost("/auth/signin/token", ([FromBody] string? token, [FromServices] DatabaseContext database, HttpContext context) => + // Allows a user to signin and receive account data + httpServer.MapPost("/auth/signin/token", async ([FromBody] string? token, [FromServices] DatabaseContext database, HttpContext context) => { if (string.IsNullOrEmpty(token)) { @@ -73,7 +70,7 @@ private static void AddAuthEndpoints() } // Completely invalid token - Reject - var expiryString = token.Split(";").Last(); + var expiryString = token.Split("_").Last(); if (!long.TryParse(expiryString, out var expiry)) { return Results.Unauthorized(); @@ -93,18 +90,18 @@ private static void AddAuthEndpoints() } // Rate limit middleware should have passed us a nicely sanitised IP. Otherwise we will just fallback - var requestIp = context.Items["RealIp"] as string ?? context.Connection.RemoteIpAddress?.ToString(); + var requestIp = GetRequestIp(context); + var userAgent = context.Request.Headers.UserAgent.ToString(); if (requestIp is null) { return Results.Forbid(); } - account.KnownIPs.Add(new AccountAddress() - { - IpAddress = requestIp - }); - database.SaveChanges(); + await UpdateAccountClients(requestIp, userAgent, account, database); + + // Add auth cookie for subsequent requests + AppendTokenCookie(account.Token, httpServer, context); - return Results.Json(account); + return Results.Ok(new { account.Token, account.Id}); }); if (!httpServer.Environment.IsDevelopment()) { @@ -123,46 +120,84 @@ private static void AddAuthEndpoints() // If the current account token is expired, we will generate a new one, // we will also give them the token cookie regardless - var expiryString = account.Token.Split(";").Last(); + var expiryString = account.Token.Split("_").Last(); if (long.Parse(expiryString) < DateTimeOffset.UtcNow.ToUnixTimeSeconds()) { var tokenString = GenerateToken(); account.Token = tokenString; - database.SaveChanges(); + await database.SaveChangesAsync(); } // Rate limit middleware should have passed us a nicely sanitised IP. Otherwise we will just fallback - var requestIp = context.Items["RealIp"] as string ?? context.Connection.RemoteIpAddress?.ToString(); + var requestIp = GetRequestIp(context); + var userAgent = context.Request.Headers.UserAgent.ToString(); if (requestIp is null) { return Results.Forbid(); } - var addressInfo = await database.AccountAddressInfos - .SingleOrDefaultAsync(info => info.IpAddress == requestIp); - if (addressInfo is null) + await UpdateAccountClients(requestIp, userAgent, account, database); + + // Add auth cookie for subsequent requests + AppendTokenCookie(account.Token, httpServer, context); + + return Results.Ok(new { account.Token, account.Id}); + }); + if (!httpServer.Environment.IsDevelopment()) + { + rateLimitEndpoints.Add("/auth/signin", (1, TimeSpan.FromSeconds(1))); + sizeLimitEndpoints.Add("/auth/signin", PayloadSize.FromKilobytes(5)); + } + + httpServer.MapPost("/auth/signout", (HttpContext context) => + { + if (context.Request.Cookies["Token"] != null) { - account.KnownIPs.Add(new AccountAddress() + context.Response.Cookies.Delete("Token", new CookieOptions() { - IpAddress = requestIp, - LastUsed = DateTime.Now + HttpOnly = true }); } - else - { - addressInfo.LastUsed = DateTime.Now; - } - await database.SaveChangesAsync(); - context.Response.Cookies.Append("Token", account.Token, new CookieOptions() - { - HttpOnly = true, - SameSite = SameSiteMode.Strict, - Expires = DateTimeOffset.UtcNow.AddMonths(1) - }); - - return Results.Json(account); + return Results.Ok(); }); - rateLimitEndpoints.Add("/auth/signin", (1, TimeSpan.FromSeconds(1))); - sizeLimitEndpoints.Add("/auth/signin", PayloadSize.FromKilobytes(5)); + } + + private static string? GetRequestIp(HttpContext context) + { + return context.Items["RealIp"] as string ?? context.Connection.RemoteIpAddress?.ToString(); + } + + private static async Task UpdateAccountClients(string requestIp, string userAgent, AccountData account, DatabaseContext database) + { + var accountClient = await database.AccountClients + .SingleOrDefaultAsync(info => + info.IpAddress == requestIp + && info.AccountId == account.Id + && info.UserAgent == userAgent); + if (accountClient is null) + { + var newClient = new AccountClient(requestIp, userAgent, account.Id); + database.AccountClients.Add(newClient); + } + else + { + accountClient.LastUsed = DateTime.Now; + } + await database.SaveChangesAsync(); + } + + private static void AppendTokenCookie(string token, WebApplication app, HttpContext context) + { + var cookieOptions = new CookieOptions + { + HttpOnly = true, + Secure = true, + SameSite = config!.CorsCookie ? SameSiteMode.Lax : SameSiteMode.None, + Expires = DateTimeOffset.UtcNow.AddMonths(1), + Domain = context.Request.Host.Host, + Path = "/" + }; + + context.Response.Cookies.Append("Token", token, cookieOptions); } } \ No newline at end of file diff --git a/SubliminalServer/Program.Profiles.cs b/SubliminalServer/Program.Profiles.cs index c2d0da5..59a4b47 100644 --- a/SubliminalServer/Program.Profiles.cs +++ b/SubliminalServer/Program.Profiles.cs @@ -3,7 +3,7 @@ namespace SubliminalServer; -public static partial class Program +internal static partial class Program { private static void AddProfileEndpoints() { @@ -19,8 +19,7 @@ private static void AddProfileEndpoints() } var profile = new UploadableProfile(account); - return Results.Json(profile); + return Results.Ok(profile); }); - rateLimitEndpoints.Add("/Profiles", (1, TimeSpan.FromMilliseconds(500))); } } \ No newline at end of file diff --git a/SubliminalServer/Program.Purgatory.cs b/SubliminalServer/Program.Purgatory.cs index c67dcec..76700b8 100644 --- a/SubliminalServer/Program.Purgatory.cs +++ b/SubliminalServer/Program.Purgatory.cs @@ -1,65 +1,37 @@ +using System.Text.RegularExpressions; using Microsoft.AspNetCore.Mvc; +using Microsoft.EntityFrameworkCore; using SubliminalServer.ApiModel; using SubliminalServer.DataModel.Account; using SubliminalServer.DataModel.Purgatory; +using SubliminalServer.DataModel.Report; namespace SubliminalServer; -public static partial class Program +internal static partial class Program { + // /accounts/{id:int} + [GeneratedRegex(@"^\/purgatory\/\d+\/report\/*$")] + private static partial Regex PurgatoryReportEndpointRegex(); + [GeneratedRegex(@"^\/purgatory\/\d+\/report\/*$")] + private static partial Regex PurgatoryEndpointRegex(); + + private static void AddPurgatoryEndpoints() { - httpServer.MapGet("/purgatory/{poemId}/report", (string poemId) => - { - Console.ForegroundColor = ConsoleColor.Yellow; - Console.WriteLine($"WIP: Report function - Poem {poemId} has been reported, please investigate!"); - Console.ResetColor(); - return Results.Ok(); - }); - // TODO: Reimplement this - //authRequiredEndpoints.Add("/purgatory/{poemId}/report"); - //rateLimitEndpoints.Add("/purgatory/{poemId}/report", (1, TimeSpan.FromSeconds(5))); - //sizeLimitEndpoints.Add("/purgatory/{poemId}/report", PayloadSize.FromKilobytes(100)); - - httpServer.MapGet("/purgatory/picks", ([FromServices] DatabaseContext database) => + httpServer.MapGet("/purgatory/{poemId:int}", async (int poemId, DatabaseContext dbContext) => { - var entries = database.PurgatoryEntries - .Where(entry => entry.Pick == true) - .Select(entry => entry.Id); - return Results.Json(entries); - }); - rateLimitEndpoints.Add("/purgatory/picks", (1, TimeSpan.FromSeconds(2))); - - httpServer.MapGet("/purgatory/after", ([FromQuery(Name = "date")] DateTime date, [FromQuery(Name = "count")] int count, [FromServices] DatabaseContext database) => - { - var poemIds = database.PurgatoryEntries - .Where(entry => entry.DateCreated > date) - .Take(Math.Clamp(count, 1, 50)) - .Select(poem => poem.Id); - return Results.Json(poemIds); - }); - rateLimitEndpoints.Add("/purgatory/after", (1, TimeSpan.FromSeconds(2))); - + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } - httpServer.MapGet("/purgatory/before", ([FromQuery(Name = "date")] DateTime date, [FromQuery(Name = "count")] int count, [FromServices] DatabaseContext database) => - { - var poemIds = database.PurgatoryEntries - .Where(entry => entry.DateCreated < date) - .Take(Math.Clamp(count, 1, 50)) - .Select(poem => poem.Id); - return Results.Json(poemIds); + return Results.Ok(poem); }); - rateLimitEndpoints.Add("/purgatory/before", (1, TimeSpan.FromSeconds(2))); - // Take into account genres that they have liked, accounts they have blocked, new poems and interactions when reccomending - httpServer.MapGet("/purgatory/recommended", () => - { - return Results.Problem(); - }); - authRequiredEndpoints.Add("/purgatory/recommended"); - rateLimitEndpoints.Add("/purgatory/recommended", (1, TimeSpan.FromSeconds(5))); - httpServer.MapPost("/purgatory", ([FromBody] UploadableEntry entryUpload, [FromServices] DatabaseContext database, HttpContext context) => + httpServer.MapPost("/purgatory/upload", ([FromBody] UploadableEntry entryUpload, [FromServices] DatabaseContext database, HttpContext context) => { var validationIssues = new Dictionary(); var tags = new List(); @@ -129,36 +101,162 @@ private static void AddPurgatoryEndpoints() return Results.Ok(entry.Id); }); - authRequiredEndpoints.Add("/purgatory"); + authRequiredEndpoints.Add("/purgatory/upload"); if (!httpServer.Environment.IsDevelopment()) { rateLimitEndpoints.Add("/purgatory", (1, TimeSpan.FromSeconds(60))); sizeLimitEndpoints.Add("/purgatory", PayloadSize.FromMegabytes(5)); } - - httpServer.MapPost("/purgatory/{id}/like", ([FromBody] int poemId, HttpContext context) => + + httpServer.MapGet("/purgatory/picks", async ([FromServices] DatabaseContext database) => { - throw new NotImplementedException(); + var entriesQuery = database.PurgatoryEntries + .Where(entry => entry.Pick == true) + .Select(entry => entry.Id); + var entries = await entriesQuery.ToListAsync(); + return Results.Ok(entries); }); + rateLimitEndpoints.Add("/purgatory/picks", (1, TimeSpan.FromSeconds(2))); - httpServer.MapPost("/purgatory/{id}/unlike", ([FromBody] int poemId, HttpContext context) => + httpServer.MapGet("/purgatory/after", ([FromQuery(Name = "date")] DateTime date, [FromQuery(Name = "count")] int count, [FromServices] DatabaseContext database) => { - throw new NotImplementedException(); + var poemIds = database.PurgatoryEntries + .Where(entry => entry.DateCreated > date) + .Take(Math.Clamp(count, 1, 50)) + .Select(poem => poem.Id); + return Results.Ok(poemIds); }); + rateLimitEndpoints.Add("/purgatory/after", (1, TimeSpan.FromSeconds(2))); - httpServer.MapPost("/purgatory/{id}/pin", ([FromBody] int poemId, HttpContext context) => + + httpServer.MapGet("/purgatory/before", ([FromQuery(Name = "date")] DateTime date, [FromQuery(Name = "count")] int count, [FromServices] DatabaseContext database) => { - throw new NotImplementedException(); + var poemIds = database.PurgatoryEntries + .Where(entry => entry.DateCreated < date) + .Take(Math.Clamp(count, 1, 50)) + .Select(poem => poem.Id); + return Results.Ok(poemIds); }); + rateLimitEndpoints.Add("/purgatory/before", (1, TimeSpan.FromSeconds(2))); - httpServer.MapPost("/purgatory/{id}/unpin", ([FromBody] int poemId, HttpContext context) => + // Take into account genres that they have liked, accounts they have blocked, new poems and interactions when reccomending + httpServer.MapGet("/purgatory/recommended", () => { - throw new NotImplementedException(); + return Results.Problem(); + }); + authRequiredEndpoints.Add("/purgatory/recommended"); + rateLimitEndpoints.Add("/purgatory/recommended", (1, TimeSpan.FromSeconds(5))); + + httpServer.MapPost("/purgatory/{poemId:int}/like", async (int poemId, HttpContext context, DatabaseContext dbContext) => + { + var account = (AccountData)context.Items["Account"]!; + + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } + + if (!account.LikedPoems.Contains(poem)) + { + account.LikedPoems.Add(poem); + await dbContext.SaveChangesAsync(); + } + + return Results.Ok(); + }); + + httpServer.MapPost("/purgatory/{poemId:int}/unlike", async (int poemId, HttpContext context, DatabaseContext dbContext) => + { + var account = (AccountData)context.Items["Account"]!; + + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } + + if (account.LikedPoems.Contains(poem)) + { + account.LikedPoems.Remove(poem); + await dbContext.SaveChangesAsync(); + } + + return Results.Ok(); + }); + + httpServer.MapPost("/purgatory/{poemId:int}/pin", async (int poemId, HttpContext context, DatabaseContext dbContext) => + { + var account = (AccountData)context.Items["Account"]!; + + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } + + if (!account.PinnedPoems.Contains(poem)) + { + account.PinnedPoems.Add(poem); + await dbContext.SaveChangesAsync(); + } + + return Results.Ok(); + }); + + httpServer.MapPost("/purgatory/{poemId:int}/unpin", async (int poemId, HttpContext context, DatabaseContext dbContext) => + { + var account = (AccountData)context.Items["Account"]!; + + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } + + if (account.PinnedPoems.Contains(poem)) + { + account.PinnedPoems.Remove(poem); + await dbContext.SaveChangesAsync(); + } + + return Results.Ok(); }); - httpServer.MapPost("/purgatory/{id}/rate", ([FromBody] UploadableRating ratingUpload, HttpContext context) => + httpServer.MapPost("/purgatory/{poemId:int}/rate", async (int poemId, [FromBody] UploadableRating ratingUpload, HttpContext context, DatabaseContext dbContext) => { throw new NotImplementedException(); }); + + httpServer.MapPost("/purgatory/{poemId:int}/report", async (int poemId, DatabaseContext dbContext) => + { + throw new NotImplementedException(); + + var poem = await dbContext.PurgatoryEntries.FindAsync(poemId); + if (poem is null) + { + return Results.NotFound(); + } + + var report = new PurgatoryReport + { + PoemId = poemId, + Poem = poem, + // Add any other necessary report details + }; + + dbContext.PurgatoryReports.Add(report); + await dbContext.SaveChangesAsync(); + + return Results.Ok(); + }).UseMiddleware(httpServer, PurgatoryReportEndpointRegex); + if (!httpServer.Environment.IsDevelopment()) + { + httpServer.UseWhen + ( + context => PurgatoryReportEndpointRegex().IsMatch(context.Request.Path), + appBuilder => appBuilder.UseMiddleware(1, TimeSpan.FromSeconds(5)) + ); + } } } \ No newline at end of file diff --git a/SubliminalServer/Program.cs b/SubliminalServer/Program.cs index 3b98df4..c31dd16 100644 --- a/SubliminalServer/Program.cs +++ b/SubliminalServer/Program.cs @@ -17,13 +17,16 @@ namespace SubliminalServer; // Prerelease .NET 9 may require "dotnet tool install --global dotnet-ef --prerelease" // to update from a non-prerelease, do "dotnet tool update --global dotnet-ef --prerelease" -public static partial class Program +internal static partial class Program { private static WebApplication httpServer; private static List authRequiredEndpoints; private static Dictionary rateLimitEndpoints; // Should only really be needed on POST endpoint private static Dictionary sizeLimitEndpoints; + + private static DirectoryInfo profileImagesDir; + private static ServerConfig? config; [GeneratedRegex("^[a-z][a-z0-9_-]{0,23}$")] private static partial Regex PermissibleTagRegex(); @@ -33,19 +36,29 @@ public static partial class Program public static async Task Main(string[] args) { var dataDir = new DirectoryInfo("Data"); - var profileImageDir = new DirectoryInfo(Path.Join(dataDir.FullName, "ProfileImages")); + var profilesDir = new DirectoryInfo(Path.Combine(dataDir.FullName, "Profiles", "Avatars")); + profileImagesDir = new DirectoryInfo(Path.Combine(dataDir.FullName, "Profiles", "Avatars")); var soundsDir = new DirectoryInfo(Path.Join(dataDir.FullName, "Sounds")); var configFile = new FileInfo("config.json"); var dbPath = Path.Join(dataDir.FullName, "subliminal.db"); - - ServerConfig? config = null; if (File.Exists(configFile.Name)) { - var configText = File.ReadAllText(configFile.Name); + var configText = await File.ReadAllTextAsync(configFile.Name); config = JsonSerializer.Deserialize(configText); } + if (config?.Version < ServerConfig.LatestVersion) + { + var configMoveLocation = configFile.FullName.Replace(".json", $".version{config.Version}.old.json"); + Console.ForegroundColor = ConsoleColor.Yellow; + Console.WriteLine("[WARN]: Current config of version {0} older than current version {1}." + + "Outdated config file will be moved to {2}.", + config.Version, ServerConfig.LatestVersion, configMoveLocation); + Console.ResetColor(); + File.Move(configFile.FullName, configMoveLocation); + config = null; + } if (config is null) { await using var stream = File.OpenWrite(configFile.Name); @@ -55,13 +68,13 @@ public static async Task Main(string[] args) }); await stream.FlushAsync(); Console.ForegroundColor = ConsoleColor.Green; - Console.WriteLine("[LOG]: Config created! Please edit {0} and run this program again!", configFile); + Console.WriteLine("[LOG]: New config created! Please edit {0} and run this program again!", configFile); Console.ResetColor(); Environment.Exit(0); } Console.ForegroundColor = ConsoleColor.Yellow; - foreach (var dirPath in new[] { dataDir, profileImageDir }) + foreach (var dirPath in new[] { dataDir, profilesDir, profileImagesDir, soundsDir }) { if (!Directory.Exists(dirPath.FullName)) { @@ -81,11 +94,13 @@ public static async Task Main(string[] args) { options.AddDefaultPolicy(policy => { - policy.WithOrigins("https://poemanthology.org", "*") - .WithOrigins("https://zekiah-a.github.io/", "*"); + policy.WithOrigins("http://localhost:80", "http://localhost:1234", "https://poemanthology.org") + .AllowAnyHeader() + .AllowAnyMethod() + .AllowCredentials(); }); }); - + builder.Services.Configure(options => { options.SerializerOptions.PropertyNamingPolicy = JsonNamingPolicy.CamelCase; @@ -113,21 +128,16 @@ public static async Task Main(string[] args) httpServer.UseStaticFiles(new StaticFileOptions { - FileProvider = new PhysicalFileProvider(profileImageDir.FullName), - RequestPath = "/ProfileImage" + FileProvider = new PhysicalFileProvider(profileImagesDir.FullName), + RequestPath = "/profiles/avatars" }); - + if (httpServer.Environment.IsDevelopment()) { httpServer.UseSwagger(); httpServer.UseSwaggerUI(); } - // This is some straightup weirdness to force inject the DB, it seems to work for out current use though - var scope = httpServer.Services.CreateScope(); - var serviceProvider = scope.ServiceProvider; - httpServer.UseMiddleware(serviceProvider.GetRequiredService()); - authRequiredEndpoints = new List(); rateLimitEndpoints = new Dictionary(); sizeLimitEndpoints = new Dictionary(); // Should only really be needed on POST endpoints @@ -170,7 +180,7 @@ public static async Task Main(string[] args) context => context.Request.Path.StartsWithSegments(endpoint), appBuilder => { - appBuilder.UseMiddleware(); + appBuilder.UseMiddleware(); } ); } @@ -182,7 +192,7 @@ private static string GenerateToken() { var tokenBytes = RandomNumberGenerator.GetBytes(32); var tokenString = Convert.ToBase64String(tokenBytes) - + ";" + DateTimeOffset.UtcNow.AddMonths(1).ToUnixTimeSeconds(); + + "_" + DateTimeOffset.UtcNow.AddMonths(1).ToUnixTimeSeconds(); return tokenString; } } \ No newline at end of file diff --git a/SubliminalServer/ServerConfig.cs b/SubliminalServer/ServerConfig.cs index d7f54c2..e81b100 100644 --- a/SubliminalServer/ServerConfig.cs +++ b/SubliminalServer/ServerConfig.cs @@ -2,10 +2,11 @@ namespace SubliminalServer; public class ServerConfig { - public const int LatestVersion = 0; - public int Version = 0; + public const int LatestVersion = 1; + public int Version = 1; public string? Certificate = null; public string? Key = null; public int Port = 1234; public bool UseHttps = false; + public bool CorsCookie = false; } \ No newline at end of file diff --git a/SubliminalServer/SubliminalServer.csproj b/SubliminalServer/SubliminalServer.csproj index 715a2e9..3d9a37a 100644 --- a/SubliminalServer/SubliminalServer.csproj +++ b/SubliminalServer/SubliminalServer.csproj @@ -1,11 +1,9 @@  - net8.0 + net9.0 enable enable - false - false @@ -14,7 +12,7 @@ all - + diff --git a/SubliminalServer/config.json b/SubliminalServer/config.json index 884b47a..21b5420 100644 --- a/SubliminalServer/config.json +++ b/SubliminalServer/config.json @@ -3,5 +3,6 @@ "Certificate": null, "Key": null, "Port": 1234, - "UseHttps": false + "UseHttps": false, + "CorsCookie": false } \ No newline at end of file diff --git a/Volume-2/Idiots/modern-day-brockelhurst.json b/Volume-2/Idiots/modern-day-brockelhurst.json index 875869c..d0df413 100644 --- a/Volume-2/Idiots/modern-day-brockelhurst.json +++ b/Volume-2/Idiots/modern-day-brockelhurst.json @@ -1 +1 @@ -{"summary":"","tags":"","cWarning":false,"cWarningAdditions":"","poemName":"Modern Day Brockelhurst","poemAuthor":"Zekiah","poemContent":{"type":"fragment","styles":[],"children":[{"type":"text","content":"So this Sunday, treat all others with love kindness and compassion. "},{"type":"newline","count":1},{"type":"text","content":"You SLIMY SINNERS, you freaking scum, "},{"type":"newline","count":1},{"type":"text","content":"Can't you see my godly duty's done? "},{"type":"newline","count":1},{"type":"text","content":" \"I'm sorry priest, I must confess\" "},{"type":"newline","count":1},{"type":"text","content":"I couldn't give a fuck if you're damned or blessed! "},{"type":"newline","count":2},{"type":"text","content":"My sheep are starving, my basement's dark, "},{"type":"newline","count":1},{"type":"text","content":"screw this family business. Hark! "},{"type":"newline","count":1},{"type":"text","content":"the herald angel screams and sings, "},{"type":"newline","count":1},{"type":"text","content":"I'm mad with power, god made me king. "},{"type":"newline","count":2},{"type":"text","content":"So now with this service, we gently proceed, "},{"type":"newline","count":1},{"type":"text","content":"open the bibles, turn to page 3, and then piss on it "},{"type":"newline","count":1},{"type":"text","content":"'til you make a hole, this stupid book won't "},{"type":"newline","count":1},{"type":"text","content":"save your soul! "},{"type":"newline","count":2},{"type":"text","content":"My way is the ONLY way to be religious, "},{"type":"newline","count":1},{"type":"text","content":"that old man's work is just sacrilegious "},{"type":"newline","count":1},{"type":"text","content":"Punish the body to spare the soul "},{"type":"newline","count":1},{"type":"text","content":"even if you have to end your life. "},{"type":"newline","count":1},{"type":"text","content":"Punish the soul to save the body "},{"type":"newline","count":1},{"type":"text","content":"And spare those trees and foxes lives. "},{"type":"newline","count":2},{"type":"text","content":"You may have priority on the road, "},{"type":"newline","count":1},{"type":"text","content":"but you're breaching heaven's code "},{"type":"newline","count":1},{"type":"text","content":"Fuck your wellbeing, screw your health "},{"type":"newline","count":1},{"type":"text","content":"build a cross and kill yourself! "},{"type":"newline","count":2},{"type":"text","content":"Deny thy doctors, deny thy teachers. "},{"type":"newline","count":1},{"type":"text","content":"All the little french boys know, "},{"type":"newline","count":1},{"type":"text","content":"to accept, "},{"type":"newline","count":1},{"type":"text","content":"thy, "},{"type":"newline","count":1},{"type":"text","content":"PREACHER! "},{"type":"newline","count":2},{"type":"text","content":"(I'll be posting some pron tomorrow "},{"type":"newline","count":1},{"type":"text","content":"so keep in touch kids! I mean, it is "},{"type":"newline","count":1},{"type":"text","content":"the grand Methodist church afterall) "},{"type":"newline","count":3}]},"pageStyle":"centre","pageBackground":""} \ No newline at end of file +{"summary":"","tags":"","cWarning":false,"cWarningAdditions":"","poemName":"Modern Day Brockelhurst","poemAuthor":"Zekiah","poemContent":{"type":"fragment","styles":[],"children":[{"type":"text","content":"So this Sunday, treat all others with love kindness and compassion. "},{"type":"newline","count":1},{"type":"text","content":"You SLIMY SINNERS, you freaking scum, "},{"type":"newline","count":1},{"type":"text","content":"Can't you see my godly duty's done? "},{"type":"newline","count":1},{"type":"fragment","styles":[{"type":"italic"}],"children":[{"type":"text","content":"\"I'm sorry priest, I must confess\""}]},{"type":"newline","count":1},{"type":"text","content":"I couldn't give a fuck if you're damned or blessed! "},{"type":"newline","count":2},{"type":"text","content":"My sheep are starving, my basement's dark, "},{"type":"newline","count":1},{"type":"text","content":"screw this family business. Hark! "},{"type":"newline","count":1},{"type":"text","content":"the herald angel screams and sings, "},{"type":"newline","count":1},{"type":"text","content":"I'm mad with power, god made "},{"type":"fragment","styles":[{"type":"bold"},{"type":"underline"}],"children":[{"type":"text","content":"me"}]},{"type":"text","content":" king."},{"type":"newline","count":2},{"type":"text","content":"So now with this service, we gently proceed, "},{"type":"newline","count":1},{"type":"text","content":"open the bibles, turn to page 3, and then piss on it "},{"type":"newline","count":1},{"type":"text","content":"'til you make a hole, this stupid book won't "},{"type":"newline","count":1},{"type":"text","content":"save your soul! "},{"type":"newline","count":2},{"type":"text","content":"My way is the "},{"type":"fragment","styles":[{"type":"bold"},{"type":"italic"}],"children":[{"type":"text","content":"ONLY"}]},{"type":"text","content":" way to be religious, "},{"type":"newline","count":1},{"type":"text","content":"that old man's work is just sacrilegious "},{"type":"newline","count":1},{"type":"text","content":"Punish the body to spare the soul "},{"type":"newline","count":1},{"type":"text","content":"even if you have to end your life. "},{"type":"newline","count":1},{"type":"text","content":"Punish the soul to save the body "},{"type":"newline","count":1},{"type":"text","content":"And spare those trees and foxes lives. "},{"type":"newline","count":2},{"type":"text","content":"You may have priority on the road, "},{"type":"newline","count":1},{"type":"text","content":"but you're breaching heaven's code "},{"type":"newline","count":1},{"type":"text","content":"Fuck your wellbeing, screw your health "},{"type":"newline","count":1},{"type":"text","content":"build a cross and kill yourself!"},{"type":"newline","count":2},{"type":"text","content":"Deny thy doctors, deny thy teachers."},{"type":"newline","count":1},{"type":"text","content":"All the little french boys know, "},{"type":"newline","count":1},{"type":"text","content":"to accept, "},{"type":"newline","count":1},{"type":"text","content":"thy, "},{"type":"newline","count":1},{"type":"text","content":"PREACHER! "},{"type":"newline","count":2},{"type":"fragment","styles":[{"type": "subscript"}],"children":[{"type":"text","content":"(I'll be posting some pron tomorrow "},{"type":"newline","count":1},{"type":"text","content":"so keep in touch kids! I mean, it "},{"type":"fragment","styles":[{"type":"italic"}],"children":[{"type":"text","content":"is"}]},{"type":"newline","count":1},{"type":"text","content":"the grand Methodist church afterall)"}]}]},"pageStyle":"centre","pageBackground":""} \ No newline at end of file diff --git a/Volume-2/Tales-And-Tomes/bull-and-bush.json b/Volume-2/Tales-And-Tomes/bull-and-bush.json index 7046b8c..9af9a0c 100644 --- a/Volume-2/Tales-And-Tomes/bull-and-bush.json +++ b/Volume-2/Tales-And-Tomes/bull-and-bush.json @@ -1 +1 @@ -{"summary":"","tags":"","cWarning":false,"cWarningAdditions":"","poemName":"Bull and Bush","poemAuthor":"Angelos","poemContent":{"type":"fragment","styles":[],"children":[{"type":"text","content":"Oben den Hügel, liegt es den alten Bull & Bush"},{"type":"newline","count":1},{"type":"text","content":"A Bush and a Bull, can’t see them around,"},{"type":"newline","count":1},{"type":"text","content":"Just in gothic writing, on the north facing pub."},{"type":"newline","count":2},{"type":"text","content":"So where are they hidden - "},{"type":"newline","count":1},{"type":"text","content":"The Bush and the Bull? "},{"type":"newline","count":1},{"type":"text","content":"The toilets, the kitchen, or under the Müll? "},{"type":"newline","count":1},{"type":"text","content":"The white building? Unlikely"},{"type":"newline","count":1},{"type":"text","content":"For beasts that mighty. "},{"type":"newline","count":1},{"type":"text","content":"But maybe I’m wrong. "},{"type":"newline","count":2},{"type":"text","content":"The bull was me, for it climbed the bush, "},{"type":"newline","count":1},{"type":"text","content":"In an electrical substation; eternal damnation, "},{"type":"newline","count":1},{"type":"text","content":"Danger of death! 415 volts. "},{"type":"newline","count":1},{"type":"text","content":"But you won’t die; it's between screws and bolts. "},{"type":"newline","count":2},{"type":"text","content":"For it seems like a tube station"},{"type":"newline","count":1},{"type":"text","content":"Rather than a substation,"},{"type":"newline","count":1},{"type":"text","content":"Clickety clack, knew t’was that! "},{"type":"newline","count":1},{"type":"text","content":"Artificial crescendo; where are they off to? "},{"type":"newline","count":1},{"type":"text","content":"Air turbulence shows, it blows to the north. "},{"type":"newline","count":2},{"type":"text","content":"This dust has a familiar smell; "},{"type":"newline","count":1},{"type":"text","content":"The smell of decay, the stench of rocks. "},{"type":"newline","count":1},{"type":"text","content":"It’s appeared on the headlines countless times "},{"type":"newline","count":1},{"type":"text","content":"For its particulates are known for scarring lungs. "},{"type":"newline","count":1},{"type":"text","content":"Yet who can resist this landmark of London? "},{"type":"newline","count":2},{"type":"text","content":"Turning right towards the circular shaft, "},{"type":"newline","count":1},{"type":"text","content":"Hier liegt ein Sign der sagt: "},{"type":"newline","count":1},{"type":"text","content":"Depth of shaft: 38 metres "},{"type":"newline","count":1},{"type":"text","content":"My wheel touches the first stair of the shaft. "},{"type":"newline","count":1},{"type":"text","content":"Spiralling down in a spiral of decay, "},{"type":"newline","count":1},{"type":"text","content":"Like Mr Statin’s might; I picture its plight. "},{"type":"newline","count":2},{"type":"text","content":"Slamming the breaks, bike "},{"type":"newline","count":1},{"type":"text","content":"Crashed on the landing, "},{"type":"newline","count":1},{"type":"text","content":"Just like the empire of OzyAndy. "},{"type":"newline","count":2},{"type":"text","content":"Rushing down to platform level, "},{"type":"newline","count":1},{"type":"text","content":"A train whizzed south from north; a furious rebel."},{"type":"newline","count":1},{"type":"text","content":"Its destination read ‘‘express to history’’ "},{"type":"newline","count":1},{"type":"text","content":"of the Hill’s largest port. "},{"type":"newline","count":1},{"type":"text","content":"Once a trading spot; largest depot, "},{"type":"newline","count":1},{"type":"text","content":"At the world’s poshest school, "},{"type":"newline","count":1},{"type":"text","content":"now among the level sands. "},{"type":"newline","count":2},{"type":"text","content":"Like a dead old bull decaying in a bush, "},{"type":"newline","count":1},{"type":"text","content":"OzyAndy’s statue crumbles, "},{"type":"newline","count":1},{"type":"text","content":"On what once was a platform. "},{"type":"newline","count":1},{"type":"text","content":"Reduced to rubbles."},{"type":"newline","count":3}]},"pageStyle":"poem-centre","pageBackground":"url(\"\")"} \ No newline at end of file +{"summary":"","tags":"","cWarning":false,"cWarningAdditions":"","poemName":"Bull and Bush","poemAuthor":"Angelos","poemContent":{"type":"fragment","styles":[],"children":[{"type":"text","content":"Oben den Hügel, liegt es den alten Bull & Bush"},{"type":"newline","count":1},{"type":"text","content":"A Bush and a Bull, can’t see them around,"},{"type":"newline","count":1},{"type":"text","content":"Just in gothic writing, on the north facing pub."},{"type":"newline","count":2},{"type":"text","content":"So where are they hidden -"},{"type":"newline","count":1},{"type":"text","content":"The Bush and the Bull?"},{"type":"newline","count":1},{"type":"text","content":"The toilets, the kitchen, or under the Müll?"},{"type":"newline","count":1},{"type":"text","content":"The white building? Unlikely"},{"type":"newline","count":1},{"type":"text","content":"For beasts that mighty."},{"type":"newline","count":1},{"type":"text","content":"But maybe I’m wrong."},{"type":"newline","count":2},{"type":"text","content":"The bull was me, for it climbed the bush,"},{"type":"newline","count":1},{"type":"text","content":"In an electrical substation; eternal damnation,"},{"type":"newline","count":1},{"type":"text","content":"Danger of death! 415 volts."},{"type":"newline","count":1},{"type":"text","content":"But you won’t die; it's between screws and bolts."},{"type":"newline","count":2},{"type":"text","content":"For it seems like a tube station"},{"type":"newline","count":1},{"type":"text","content":"Rather than a substation,"},{"type":"newline","count":1},{"type":"text","content":"Clickety clack, knew t’was that!"},{"type":"newline","count":1},{"type":"text","content":"Artificial crescendo; where are they off to?"},{"type":"newline","count":1},{"type":"text","content":"Air turbulence shows, it blows to the north."},{"type":"newline","count":2},{"type":"text","content":"This dust has a familiar smell;"},{"type":"newline","count":1},{"type":"text","content":"The smell of decay, the stench of rocks."},{"type":"newline","count":1},{"type":"text","content":"It’s appeared on the headlines countless times"},{"type":"newline","count":1},{"type":"text","content":"For its particulates are known for scarring lungs."},{"type":"newline","count":1},{"type":"text","content":"Yet who can resist this landmark of London?"},{"type":"newline","count":2},{"type":"text","content":"Turning right towards the circular shaft,"},{"type":"newline","count":1},{"type":"text","content":"Hier liegt ein Sign der sagt:"},{"type":"newline","count":1},{"type":"text","content":"Depth of shaft: 38 metres "},{"type":"newline","count":1},{"type":"text","content":"My wheel touches the first stair of the shaft."},{"type":"newline","count":1},{"type":"text","content":"Spiralling down in a spiral of decay,"},{"type":"newline","count":1},{"type":"text","content":"Like Mr Statin’s might; I picture its plight."},{"type":"newline","count":2},{"type":"text","content":"Slamming the breaks, bike"},{"type":"newline","count":1},{"type":"text","content":"Crashed on the landing, "},{"type":"newline","count":1},{"type":"text","content":"Just like the empire of OzyAndy."},{"type":"newline","count":2},{"type":"text","content":"Rushing down to platform level,"},{"type":"newline","count":1},{"type":"text","content":"A train whizzed south from north; a furious rebel."},{"type":"newline","count":1},{"type":"text","content":"Its destination read ‘‘express to history’’"},{"type":"newline","count":1},{"type":"text","content":"of the Hill’s largest port."},{"type":"newline","count":1},{"type":"text","content":"Once a trading spot; largest depot,"},{"type":"newline","count":1},{"type":"text","content":"At the world’s poshest school,"},{"type":"newline","count":1},{"type":"text","content":"now among the level sands."},{"type":"newline","count":2},{"type":"text","content":"Like a dead old bull decaying in a bush,"},{"type":"newline","count":1},{"type":"text","content":"OzyAndy’s statue crumbles,"},{"type":"newline","count":1},{"type":"text","content":"On what once was a platform."},{"type":"newline","count":1},{"type":"text","content":"Reduced to rubbles."},{"type":"newline","count":3}]},"pageStyle":"poem-centre","pageBackground":""} \ No newline at end of file diff --git a/account.html b/account.html index 70536b2..aa86db8 100644 --- a/account.html +++ b/account.html @@ -1,483 +1,563 @@ - - - - Subliminal - - - - - - - - - - - - - - -

Your profile role:

-
- - - - - - - - - - - - -
-
- -
- -

These are for vanity purposes only, setting a certain role won't limit you from doing certain actions!

-
- -

Enter your email:

- -
- -
-
-
-
- - - -
-
-
-

I'm

- -

,

-

a writer

-
- @username - -
-
🌍 
-
📌 Poem1, Poem2, Poem3, Poem4, Poem5
-
-
-
-
-

- -

Recent works:

-
- -

Following:

-
- -

Private account info:

-
-

This stuff won't appear on your profile, it is for your eyes only!

- - -

-

Draft poems:

-
-
-

London

-

I walk through each chartered street, near where the chartered thames do flow. And marks on every face I meet, marks of weakness, marks of woe.

-
-
- -

-
-

Account agreement +

- -
- <- Back - - - + + + + Subliminal + + + + + + + + + + + + + + + + +

Your profile role:

+
+ + + + + + + + + + + + +
+
+ +
+ +

These are for vanity purposes only, setting a certain role won't limit you from doing certain actions!

+
+ +

Enter your email:

+ +
+ +
+
+
+
+ + + +
+
+
+

I'm

+ +

,

+

a writer

+
+ @username + +
+
🌍 
+
📌 Poem1, Poem2, Poem3, Poem4, Poem5
+
+
+
+
+

+ +

Recent works:

+
+ +

Following:

+
+ +

Private account info:

+
+

This stuff won't appear on your profile, it is for your eyes only!

+ + +

+

Draft poems:

+
+
+

London

+

I walk through each chartered street, near where the chartered thames do flow. And marks on every face I meet, marks of weakness, marks of woe.

+
+
+ +

+
+

Account agreement +

+ +
+ <- Back + + + diff --git a/account.js b/account.js index 26010e6..71f3d02 100644 --- a/account.js +++ b/account.js @@ -1,31 +1,4 @@ -const actionType = { - // General account actions - BlockUser: 0, - UnblockUser: 1, - FollowUser: 2, - UnfollowUser: 3, - LikePoem: 4, - UnlikePoem: 5, - RatePoem: 6, - UploadDraft: 7, - DeleteDraft: 8, - GetDraft: 9, - Report: 10, - - // Location - Account data - UpdateEmail: 11, - UpdateNumber: 12, - - //Location - Account profile - UpdatePenName: 13, - UpdateBiography: 14, - UpdateLocation: 15, - UpdateRole: 16, - UpdateAvatar: 17, - PinPoem: 18, - UnpinPoem: 19 -} - +"use strict"; const badgeType = { Admin: 0, Based: 1, @@ -59,17 +32,15 @@ function getBadgeInfo(badge) { } async function getAccountData() { - let data = null - - await fetch(serverBaseAddress + "/auth/signin", { + const res = await fetch(serverBaseAddress + "/accounts/me", { method: "POST", - headers: { 'Content-Type': 'application/json' }, - body: '"' + localStorage.accountCode + '"' + headers: { 'Content-Type': 'application/json' } }) - .then((res) => res.json()) - .then((dataObject) => data = dataObject) - - return data + if (!res.ok) { + return null + } + const accountData = await res.json() + return accountData } async function getPublicProfile(accountId) { @@ -86,14 +57,20 @@ async function getPublicProfile(accountId) { } async function isLoggedIn() { - if (!localStorage.accountCode) return false + if (sessionStorage.accountToken) { + return true + } let response = await fetch(serverBaseAddress + "/auth/signin/token", { method: "POST", - headers: { 'Content-Type': 'application/json' }, body: '"' + localStorage.accountCode + '"'} - ) + headers: {'Content-Type': 'application/json'} + }) return response.ok } -console.log("%cPrivate account data may be held in browser local storage, a thing that can be acessed by putting code in this browser console! If someone tells you to put something here, there is a high chance that you may get hacked!", "background: red; color: yellow; font-size: large") +async function signout() { + window.location.reload(true) +} + +console.log("%cAccount credentials may be held in browser storage, a thing that can be acessed by putting code in this console! If someone tells you to put something here, there is a high chance that you may get hacked!", "background: red; color: yellow; font-size: large") console.log("%cTL;DR: Never put anything you do not understand here. Uncool things may happen.", "color: blue; font-size: x-large") \ No newline at end of file diff --git a/contents.css b/contents.css new file mode 100644 index 0000000..6f6de23 --- /dev/null +++ b/contents.css @@ -0,0 +1,324 @@ +.section-span { + cursor: pointer; + max-width: 20%; + width: 20%; + margin: 20px; + transition: .2s font-weight, .2s margin, .2s max-width; +} + +.section-body { + /*border-radius: 10px; + border-left: 2px solid grey; + padding-left: 5px;*/ + transition: transform .5s, opacity .5s; +} + +.section-collapsed { + position: absolute; + top: 34px; + transform: translate(-50%, -50%); + right: 16px; + transform-origin: center; + transition: .2s transform; + border: 1px solid lightgray; + border-radius: 100%; + width: 36px; + height: 36px; + background-color: var(--button-opaque); + z-index: 2; + cursor: pointer; +} + +.section-collapsed > svg { + fill: var(--text-colour); + padding: 8px; + box-sizing: border-box; +} + +.grid-container { + display: grid; + grid-template-columns: auto auto; + padding: 10px; + grid-gap: 10px; +} + +.grid-sub-container { + display: flex; + flex-direction: row; + border-radius: 8px; + margin: 8px; + padding: 8px; + background-color: var(--panel-transparent); + position: relative; + transition: max-height .2s; + max-height: 1000px; + overflow: hidden; +} + +.grid-sub-container[collapsed] { + height: 64px; + max-height: 64px; + cursor: pointer; +} + +.grid-sub-container[collapsed] > .section-span { + max-width: 100%; + width: 100%; + font-weight: bold; + position: absolute; + margin: 14px; + overflow: clip; +} + +.grid-sub-container[collapsed] > .section-body { + transform: translateX(110%); + opacity: 0; +} + +.grid-sub-container[collapsed] > .section-collapsed { + transform: translate(-50%, -50%) rotate(-90deg); +} + +.seminars-centre { + border-radius: 8px; + height: 400px; + background-color: var(--button-opaque); + display: flex; + flex-direction: column; + padding: 16px; + background: linear-gradient(-90deg, #ee6352, #d16e8d); + flex-grow: 1; + flex-shrink: 0; + transition: .2s box-shadow; + margin: 0px; +} + +.seminars-centre:hover { + box-shadow: 0px 0px 6px 2px lightgrey; +} + +.seminars-parent { + margin-top: 16px; + width: 100%; + display: flex; + column-gap: 16px; +} + +.seminars-side { + border-radius: 8px; + height: 400px; + background-color: var(--button-opaque); + display: flex; + flex-direction: column; + padding: 16px; + flex-shrink: 2; + flex-basis: 20%; + margin: 0px; +} + +.seminars-parent-mobile { + margin-top: 16px; + width: 100%; + display: none; + column-gap: 16px; + flex-direction: column; + box-sizing: border-box; +} + +.seminars-side-parent-mobile { + display: flex; + column-gap: 8px; + padding-top: 8px; + box-sizing: border-box; +} + +.seminars-side-mobile { + border-radius: 8px; + height: 280px; + background-color: var(--button-opaque); + display: flex; + flex-direction: column; + padding: 16px; + flex: 50%; +} + +.poem-preview { + display: none; + position: fixed; + left: 20px; + top: 200px; + width: 250px; + height: 400px; + border: 2px solid grey; + border-radius: 2px; + background: white; +} + +.poem-preview > iframe { + width: 100%; + height: 100%; + border: none; +} + +/*Purgatory flex CSS*/ +#purgatoryFlex { + display: flex; + max-width: 100%; + padding: 8px; + flex-direction: row; + column-gap: 16px; + overflow-x: auto; + user-select: none; + overflow-y: hidden; + height: min-content; + max-height: 80vh; + box-sizing: border-box; +} + +#purgatoryGrid { + display: grid; + grid-template-columns: auto auto auto auto auto; + grid-gap: 16px; + overflow: hidden; + overflow-y: auto; + position: relative; + height: min-content; + max-height: 80vh; + box-sizing: border-box; +} + +.purgatory-warning { + width: 100%; + display: flex; + justify-content: center; + margin-top: 16px; + margin-bottom: 16px; +} + +.purgatory-warning > span { + align-self: center; + background: var(--button-transparent); + padding: 8px; + border-radius: 8px; + border: 1px solid lightgray; +} + +.purgatory-actions-bottom { + display: flex; +} + +#filtersBar { + display: flex; + justify-content: center; + column-gap: 4px; + margin: 8px; +} + +#filtersBar > button { + border-radius: 32px; + padding: 1px; + position: relative; +} + +#filtersBar > button > div { + padding-left: 12px; + padding-right: 12px; + padding-top: 4px; + padding-bottom: 4px; + background-color: var(--button-transparent); + border-radius: 32px; + user-select: none; + pointer-events: none; +} + +.signup-code-hidden { + background: gray; + border-radius: 4px; + color: gray; + cursor: pointer; + transition: .2s all; +} + +.searchbar { + color: var(--text-colour); + background-color: #ffffff33; + backdrop-filter: blur(10px); + z-index: 1; + border: 1px solid gray; + border-radius: 4px; + padding-left: 8px; +} + +.classics-header { + top: 78px; + display: flex; + column-gap: 8px; + position: sticky; + line-height: 40px; + z-index: 3; + margin: 0; + padding: 0px; + transition: .2s background-color, .2s margin, .2s padding, .2s border; +} + +.classics-header.stuck { + border: 1px solid gray; + padding: 8px; + margin: -8px; + border-radius: 8px; + backdrop-filter: blur(10px); +} + +.classics-header.stuck > .searchbar { + border-radius: 8px 0 0 8px; +} + +@media screen and (orientation:portrait) { + .grid-container { + grid-template-columns: auto; + } + + .searchbar { + height: 48px; + flex-grow: 1; + } + + .classics-header { + top: 64px; + } + + .grid-sub-container { + flex-direction: column; + } + + .section-span { + max-width: 100%; + width: 100%; + } + + #purgatoryGrid { + grid-template-columns: auto auto; + } + + .purgatory-actions-bottom { + justify-content: end; + } + + .seminars-parent-mobile { + display: flex; + } + + .seminars-parent { + display: none; + } +} + +@media screen and (orientation:landscape) { + .searchbar { + width: 50%; + line-height: 300%; + top: 2px; + z-index: 1; + flex-grow: 1; + } +} diff --git a/contents.html b/contents.html index f893457..c2669bb 100644 --- a/contents.html +++ b/contents.html @@ -1,347 +1,24 @@ - + Subliminal + - + + - - + + + -
@@ -672,6 +349,7 @@

Other stuff:

<- Back - - - + Subliminal + + + + + + + + + -
diff --git a/editor-document.js b/editor-document.js index acd25af..abc0a9a 100644 --- a/editor-document.js +++ b/editor-document.js @@ -1,5 +1,5 @@ -// Code that handles logic for complex poem editor interactions "use strict"; +// Code that handles logic for complex poem editor interactions const positionMovements = { up: 0, down: 1, diff --git a/for-teachers.html b/for-teachers.html index 7159d8a..f26df31 100644 --- a/for-teachers.html +++ b/for-teachers.html @@ -4,8 +4,10 @@ Subliminal + - + + diff --git a/index.css b/index.css new file mode 100644 index 0000000..df3956d --- /dev/null +++ b/index.css @@ -0,0 +1,243 @@ +body { + overflow-x: clip; + overflow-y: scroll; +} + +.fixed { + vertical-align: middle; + text-align: center; + height: calc(100% - 40px); + width: 100%; + overflow: scroll; + margin-bottom: 5px; + scroll-behavior: smooth; + -ms-overflow-style: none; + scrollbar-width: none; + border-radius: 2px; + transition: width .5s; +} + +.fixed:has(a > #advert3:target) { + width: 128%; +} + +.fixed::-webkit-scrollbar { + display: none; +} + +.fixed-container { + position: fixed; + top: 30%; + left: 65%; + height: 46vh; + width: 18vh; +} + +.slide-link { + text-decoration: none; + padding: 2px 6px 2px 6px; + color: rgb(50, 50, 50); + background-color: #eeeeee; + border: 1px solid rgb(50, 50, 50); + border-radius: 2px; +} + +.scroll-in { + opacity: 1 !important; + transform: unset !important; +} + +.scroll-unseen { + opacity: 0; + transform: translateY(64px) scale(1.1); + transition: .5s opacity, .5s transform; +} + +.main-logo { + min-width: 70%; + min-height: 70%; + aspect-ratio: 1/1; +} + +/* Sections, below main style */ +.section { + display: flex; + flex-direction: row; + border-radius: 8px; + background-color: var(--button-transparent); + padding: 8px; + margin-left: 5%; + margin-right: 5%; + align-self: center; + column-gap: 16px; +} + +/*Special bottom section code*/ +.section-double { + flex-direction: row; + background: transparent; +} + +.section-double > div { + /*max-height: 256px; + height: 256px;*/ + margin-left: 0px; + margin-right: 0px; +} + +.section-double img, .section-double video { + width: 180px; + height: 320px; + object-fit: cover; + border-radius: 4px; +} + +/* Relatable-s messages */ +.section-info { + margin-left: 5%; + margin-right: 5%; + margin-top: 32px; + margin-bottom: 128px; +} + +.presenter-container { + display: flex; + flex-direction: column; + width: 60%; + margin: 0px; +} + +.presenter { + position: relative; + width: 100%; + border-radius: 4px; + overflow: hidden; + user-select: none; + flex-grow: 1; +} + +.presenter img { + width: 100%; + height: 100%; +} + +.presenter p { + position: absolute; + top: 0px; + left: 50%; + transform: rotateX(42deg) rotateY(-12deg) rotateZ(24deg) translate(-50%); + white-space: nowrap; + color: white; + opacity: 0.6; + font-family: 'Comic Sans MS', 'Comic Sans', cursive; + line-height: 1; + font-weight: bolder; + font-size: 1.8vw; +} + +.presenter-caption { + opacity: 0.6; + margin-top: 8px; + font-style: italic; +} + +#aboutInfo.scroll-unseen { + transform: translateX(-90%); +} + +.info-list { + padding-left: 32px; +} + +.info-list > li { + margin: 18px 18px 18px 12px; + list-style-type: none; +} + +#poemProcessSubInfo { + display: flex; + column-gap: 64px; + transition: .5s column-gap, .2s background-color; + overflow-x: auto; + -ms-overflow-style: none; + scrollbar-width: none; + transition: .5s column-gap, .2s background-color; +} + +#poemProcessSubInfo > li { + background-color: var(--panel-transparent); + padding: 8px; + border-radius: 8px; + min-width: 156px; + user-select: none; + list-style-type: none; +} + +#poemProcessSubInfo > li > p { + opacity: 0.6; + transition: .2s opacity; +} + +#poemProcessSubInfo > li:hover { + background-color: var(--button-opaque-hover); +} + +#poemProcessSubInfo > li:hover > p { + opacity: 1; +} + +#poemProcessSubInfo.scroll-in { + column-gap: 4px; +} + +#topBackground { + position: absolute; + z-index: -1; + left: 0px; + top: 0px; +} + +#topBackground > img { + width: 100%; +} + +#topBackground::after { + position: absolute; + left: 0px; + top: 0px; + width: 100%; + height: 100%; + background: linear-gradient(to bottom, transparent, var(--background-opaque) 90%); + content: ""; +} + +@media(prefers-color-scheme: dark) { + .index-image { + filter: invert(1); + } +} + +@media screen and (orientation:portrait) { + .fixed-container { + left: auto; + right: 32px; + } + + .section { + flex-direction: column-reverse; + } + + .section-double { + row-gap: 16px; + } + + .presenter p { + font-size: 4.9vw; + } + + .presenter-container { + display: flex; + flex-direction: column; + width: 100%; + } +} diff --git a/index.html b/index.html index 60abae8..d4ac463 100644 --- a/index.html +++ b/index.html @@ -1,264 +1,36 @@ - + Subliminal - + - + + + + - - -
+
Subliminal scenery background

Subliminal

“The methallyne blue analogy of stuff.

Probably the weirdest poetry anthology to ever exist.”

- Dog +

Beware: C o o l s t u f f a h e a d .

About Subliminal

-
+

Subliminal is an open source project to create the largest repository of poems in one place, where anyone can contribute their own work, no matter how chaotic, critical, or hilarious it may be!

Here we value fairness and freedom of expression, because everyone should have the right to be creative.

@@ -278,26 +50,26 @@ A vulture, flew overhead,
then dropped down dead! Beating
my back like a brick of lead. Yet I
+ Shouted 'horay', meat for days!
+ On my way to Al'mogus.
...

A long road to Al'mogus, a poem about human desire, by Zekiah.
-
- - - +

Get started

-
+
@@ -310,52 +82,49 @@

Simply navigate to the "poems" page to see all of the contents of the anthology. Here you can also set up an account, or start writing your own poem with the "+add a poem" button whenever you like!

If you ever have any questions, check the dictionary and disclaimer pages for more info on what you may find here...

-
- - - +

Other stuff

-
-
+
+
A fun game to play in your free time -
+

Rplace.live: a spinoff of the reddit place event!

A game by Zekiah-A and BlobKat, featuring a massive multiplayer canvas, place pixels, collaborate or start pixel wars, to create the most epic canvas arts possible.

-
-
- -
+ + +
-
+

The best waste of money since buying an NFT!

Help fund development of subliminal, and support me with my other projects by donating, and getting an absolute monstrosity of literature in return, it's almost a good deal!

-
-
-
+ + +

@@ -363,6 +132,7 @@
+ diff --git a/poem-editor.css b/poem-editor.css index 71c49ce..3314a57 100644 --- a/poem-editor.css +++ b/poem-editor.css @@ -4,56 +4,6 @@ border-radius: 4px; background-color: var(--background-opaque); } -#editorCanvas { - display: block; - width: 100%; - height: 100%; - cursor: text; -} -#editorCanvas:focus { - outline: none; -} -#editorContext { - display: flex; - visibility: hidden; - transition: left 0.1s ease 0s; - position: fixed; - width: 200px; - background-color: var(--button-transparent); - backdrop-filter: blur(5px); - border-radius: 4px; - box-shadow: 0px 0px 10px #656565; - border: 1px solid darkgray; - flex-direction: column; - padding: 4px; - row-gap: 4px; -} -#editorContext > button { - height: 32px; - background-color: transparent; - border: 1px solid var(--button-opaque); - display: flex; - align-items: center; - column-gap: 8px; - transition: .05 transform; - user-select: none; -} -#editorContext > button:hover { - background-color: var(--button-opaque-hover); -} -#editorContext > button:active { - transform: scale(0.98); -} -#editorContext > button > svg { - opacity: 0.6; - fill: var(--text-colour); -} -#editorContext > button > span { - font-weight: bold; - opacity: 0.4; - flex-grow: 1; - text-align: end; -} .tool { margin: 1px; height: 24px; @@ -281,7 +231,6 @@ display: none; #tagContainer > button { border-radius: 4px; height: 64px; - background: lightgrey; padding: 4px; text-align: center; position: relative; @@ -351,31 +300,6 @@ color: white; padding: 8px; overflow-y: auto; } -.suggestions { - width: 192px; - position: absolute; - background: #bdbdbd; - z-index: 3; - border-radius: 4px; - top: 0px; - display: flex; - background-color: var(--button-transparent); - flex-direction: column; - border: 1px solid gray; - overflow: clip; - overflow-y: scroll; -} -.suggestions-item { - padding: 8px; - display: flex; -} -.suggestions-item > span:nth-child(1) { - flex-grow: 1; -} -.suggestions-item > span:nth-child(2) { - font-size: 10px; - opacity: 0.6; -} .online-editors { position: fixed; left: 16px; diff --git a/poem-editor.html b/poem-editor.html index 694d3fa..9bf4b36 100644 --- a/poem-editor.html +++ b/poem-editor.html @@ -1,1030 +1,820 @@ - - - Subliminal - - - - - - - - - - - - - -

You're almost there!

- + + + Subliminal + + + + + + + + + + + + + + + +

You're almost there!

+ +

+ When you click submit, your poem will be uploaded publicly to the poem + purgatory. For more information, see the license agreement below. +

+
+

+ License agreement + +

+
- -

Built-in backgrounds:

- These backgrounds are free for you to use to jazz up your poem's look! - Drawn and shot by members of the subliminal team. + By uploading your work to this site, you give consent for your poem to + be licensed under Creative Commons + Attribution-NonCommercial-ShareAlike 4.0.

-
-
- -
Taken somewhere in the Chilterns, UK.
-
+

+ You also agree to allow us permission to store and process your work. + We can also accept no responsibility, if you write something that gets + you in trouble with people you know, an institution, or local + government, you agree that you WILLINGLY chose to upload it here. We + accept no responsibility how your work is interpreted, and what + consequences may or may not come as a result of it. +

+

+ You will always be the lone copyright owner of your work, we will + never take ownership of what you made from you, and we will always + comply if you ask for your poem to be modified, and or taken down from + this site. +

+
+ +
+ +

Built-in backgrounds:

+

+ These backgrounds are free for you to use to jazz up your poem's look! + Drawn and shot by members of the subliminal team. +

+
+
+ +
Taken somewhere in the Chilterns, UK.
+
+
+
- -
- Taken at an intersection somewhere between Uusikaupunki and - Helsinki, Finland. -
+ Taken at an intersection somewhere between Uusikaupunki and + Helsinki, Finland.
+
+
+
- -
- A fish that mysteriously disappeared the next day, Island near - Uusikaupunki, Finland. -
+ A fish that mysteriously disappeared the next day, Island near + Uusikaupunki, Finland.
+
+
+ +
A bridge to the Tate, London, UK.
+
+
+ +
A day in the city, near Farringdon, London, UK.
+
+
+
- -
A bridge to the Tate, London, UK.
+ It took standing on a rock in the middle of the sea to get this, + Portheleven, UK.
+
+
+
- -
A day in the city, near Farringdon, London, UK.
+ A really old shell cave, in Gyllyngdune Gardens, Falmouth, UK.
+
+
+ +
A long path to mount Snowdon (maybe), Wales, UK.
+
+
+
- -
- It took standing on a rock in the middle of the sea to get this, - Portheleven, UK. -
+ A spooky snap of quarry taken from a moving vehicle, Snowdonia, + Wales, UK.
+
+
+ +
A sunnier view of the massive quarry, Snowdonia, Wales, UK.
+
+
+
- -
- A really old shell cave, in Gyllyngdune Gardens, Falmouth, UK. -
+ An atmospheric pee(a)k (punny) of a quarry, Snowdonia, Wales, UK.
+
+
+
- -
A long path to mount Snowdon (maybe), Wales, UK.
+ This boat certainly has more than 5 horsepower, Island near + Uusikaupunki, Finland.
+
+
+ +
A long, long road, past field and moor, Yorkshire, UK.
+
+
+
- -
- A spooky snap of quarry taken from a moving vehicle, Snowdonia, - Wales, UK. -
+ Chaos, for those who want to have a bit of zazz around their stuff.
+
+
+
- -
A sunnier view of the massive quarry, Snowdonia, Wales, UK.
+ Is it a bird? Dog? A secret message of text? All we know is that + this mascot is iconic.
-
- -
- An atmospheric pee(a)k (punny) of a quarry, Snowdonia, Wales, UK. +
+
+
+ +

Load poem:

+ +
+ + +
+
+
+

+ Click On Me to Start Editing +

+

 - By 

+

+ Click On Me to Start Editing +

+
+
+
+
- -

Load poem:

- -
-
+
- - -
-
-

- Click On Me to Start Editing -

-

 - By 

-

- Click On Me to Start Editing -

-
-
-
- - - - - - - - - - -
- + + + + + + + +
+ + - - - -
-
-
- -

Find someone to invite

- -
-
-
- Online editors: -
- -
- You
-
- + + +
-
- - - - -
- - - - - - - + You
-
-
-
-
-

Poem summary:

-
- + +
+
+ +
+
+
+
+
+

Poem summary:

-
-
-

Poem tags:

-
-
- -
+ +
+
+
+

Poem tags:

-
-
-

Content Warning:

-
-
- - - -
+
+
-
-
-

Poem Rhyme finder:

- -

- Use * to match any word ending, or ?? to match anything within a word, - for example pai* (paint, pain), p???t (paint, print) -

- - - - - - - - - - - - - -
- +
+
+

Content Warning:

-

- Credits to datamuse for their fantastic API. https://www.datamuse.com/ -

-
-
-

Poem AI coauthor:

- Use the subliminal poem writing AI to help cowrite your work! - - - - - -
-
-
- - - + + +
+
- <- Back - - + } + + function appendRhyme(event) { + // TODO: Rhyme + console.log(event) + } + + async function editMode() { + loadPoemPopup.close() + let currentJson = await fetchPathJson(edit) + currentJson.edits = edit + load(currentJson) + } + + if (amend) + amendMode(); //enter poem amendment mode + else if (edit) editMode(); + else fetchStorage(); //Load saved poems from local storage + + formattingToolbar.addEventListener("mousemove", (event) => { + for (let button of formattingToolbar.children) { + if (button.className == "separator") continue; + button.style.background = + "radial-gradient(at left " + + (event.screenX - button.offsetLeft) + + "px top " + + (event.screenY - button.offsetTop) + + "px, darkgray, var(--background-opaque)" + } + }); + + formattingToolbar.addEventListener("mouseleave", (event) => { + for (let button of formattingToolbar.children) { + if (button.className == "separator") continue + button.style.background = "none" + } + }); + + + \ No newline at end of file diff --git a/poem.html b/poem.html index 2f90db9..0eedede 100644 --- a/poem.html +++ b/poem.html @@ -3,6 +3,7 @@ Subliminal + @@ -24,7 +25,7 @@

Poem - By Author

- + + <- Back -
+
@@ -58,24 +59,24 @@

Edit poem:

} try { const poemData = await response.json() - //Set up title and url bar for vanity + // Set up title and url bar for vanity window.history.replaceState(null, "Title", path) document.title = "Subliminal - " + poemData.poemName - //Display content warning with additions if needed + // Display content warning with additions if needed if (poemData.cWarning === true) { document.body.insertBefore( createFromData("content-warning", { addition: poemData.cWarningAdditions }), back) } - //Place poem data into the DOM + // Place poem data into the DOM poemTitle.innerText = poemData.poemName + " - By " + poemData.poemAuthor const poemDocument = new EditorDocument(poemData.poemContent) poemDocument.renderHtmlData(poemContent) poemMain.classList.add(poemData.pageStyle) document.body.style.background = poemData.pageBackground - //Probably useless since pages are procedurally generated... + // Probably useless since pages are procedurally generated... document.querySelector('meta[name="description"]').setAttribute("content", poemData.summary) document.querySelector('meta[name="keywords"]').setAttribute("content", poemData.tags) } diff --git a/polyfill.js b/polyfill.js index 194ced5..43ad70e 100644 --- a/polyfill.js +++ b/polyfill.js @@ -7,7 +7,6 @@ */ (() => { "use strict" - Path2D.prototype.roundRect ??= roundRect; if (globalThis.CanvasRenderingContext2D) { globalThis.CanvasRenderingContext2D.prototype.roundRect ??= roundRect diff --git a/purgatory-poem.html b/purgatory-poem.html index af90e1e..29c718b 100644 --- a/purgatory-poem.html +++ b/purgatory-poem.html @@ -2,17 +2,19 @@ - Subliminal - Loading... + Subliminal + - + +