Quellcode durchsuchen

i love cloudflare upload limits

Sv443 vor 1 Jahr
Ursprung
Commit
bdf9b9ebac
6 geänderte Dateien mit 4336 neuen und 0 gelöschten Zeilen
  1. 68 0
      docs/raw/errorPage.css
  2. 54 0
      docs/raw/errorPage.html
  3. 55 0
      docs/raw/errorPage.js
  4. 1152 0
      docs/raw/index.css
  5. 1762 0
      docs/raw/index.html
  6. 1245 0
      docs/raw/index.js

+ 68 - 0
docs/raw/errorPage.css

@@ -0,0 +1,68 @@
+@import url("https://fonts.googleapis.com/css?family=Roboto+Mono&display=swap");
+
+:root {
+    --bg-color: #444;
+    --bg-accent-color: #333;
+    --bg-color-sidenav-opened: #777;
+    --header-bg-color: #101747;
+    --header-bg-color-sidenav-opened: #101747;
+
+    --accent-color: #8a2be2;
+    --accent-color-light: #b05ffc;
+    --accent-color-sidenav-opened: #a779d3;
+}
+
+body {
+    margin: 0;
+    background-color: var(--bg-color);
+    color: #fff;
+    font-family: "Roboto", "Segoe UI", "Arial", sans-serif;
+    font-size: 15px;
+}
+
+header {
+    margin-top: 30px;
+    text-align: center;
+    display: block;
+}
+
+#errorContainer {
+    display: inline-block;
+    padding: 15px;
+    border: 2px solid #f00;
+    border-radius: 10px;
+    background-color: var(--bg-accent-color);
+}
+
+#errorContainer h1, #errorContainer h3 {
+    margin: 0;
+    padding: 0;
+}
+
+#errorContainer h1 {
+    margin-bottom: 10px;
+}
+
+.noa {
+    color: inherit;
+}
+
+.noul, .mimica {
+    cursor: pointer;
+}
+
+a:not(.noul), .mimica:not(.noul) {
+    color: #8f9aff;
+    text-decoration: none;
+}
+
+a:not(.noul):hover, .mimica:not(.noul):hover {
+    color: #bcc2ff;
+    text-decoration: underline;
+}
+
+#errSubText {
+    text-align: center;
+    display: inline-block;
+    width: 60%;
+}

+ 54 - 0
docs/raw/errorPage.html

@@ -0,0 +1,54 @@
+<!DOCTYPE html>
+<html>
+    <head>
+        <meta charset="UTF-8">
+        <meta name="viewport" content="width=device-width, initial-scale=1.0">
+        <meta name="pagename" content="JokeAPI - Error">
+
+        <meta charset="UTF-8">
+        <meta name="pagename" content="JokeAPI - Error">
+        <meta name="viewport" content="width=device-width, initial-scale=1.0">
+        <meta name="HandheldFriendly" content="true">
+        <meta name="description" content="A RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta name="twitter:card" content="A RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta name="subject" content="API">
+        <meta name="copyright" content="Sv443 - Licensed under MIT license (https://sv443.net/LICENSE)">
+        <meta name="author" content="Sv443, [email protected]">
+        <meta property="og:title" content="JokeAPI">
+        <meta property="og:url" content="https://jokeapi.dev/">
+        <meta property="og:description" content="A RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta property="og:image" content="https://jokeapi.dev/favicon.ico">
+        <meta property="og:type" content="profile">
+        <meta property="og:locale" content="en_US">
+        <meta property="og:locale:alternate" content="en_GB">
+        <meta property="og:locale:alternate" content="en_AU">
+        <meta name="application-name" content="JokeAPI - Error">
+        <meta name="theme-color" content="#101747">
+
+        <title>JokeAPI - Error</title>
+
+        <!-- JokeAPI ErrorPage Stylesheet: -->
+        <link rel="stylesheet" href="./static/errorPage.css">
+
+        <!-- JSLib: -->
+        <script src="https://sv443.net/cdn/jsl/1.5.0.js"></script>
+
+        <!-- JSCookie script -->
+        <script src="https://cdn.jsdelivr.net/npm/js-cookie@beta/dist/js.cookie.min.js"></script>
+
+        <!-- JokeAPI-errorPage.js -->
+        <script src="./static/errorPage.js"></script>
+    </head>
+    <body>
+        <header>
+            <div id="errorContainer">
+                <h1 id="errCodeDisplay">(Loading...)</h1>
+                <h3 id="errDetailDisplay">(Loading...)</h3>
+
+                <br><br>
+                
+                <span id="errSubText"></span>
+            </div>
+        </header>
+    </body>
+</html>

+ 55 - 0
docs/raw/errorPage.js

@@ -0,0 +1,55 @@
+window.errorWrittenToPage = false;
+
+document.addEventListener("DOMContentLoaded", function() {
+    setTimeout(function() {
+        if(window.errorWrittenToPage != true)
+        {
+            setErrorDisp(500, "Internal Server Error", "Error while finding the error message - oh the irony");
+        }
+    }, 6000);
+
+    try
+    {
+        let errorInfo = JSON.parse(Cookies.get("errorInfo")); // eslint-disable-line no-undef
+
+        let statusCode = parseInt(errorInfo["API-Error-StatusCode"]);
+        let errorReasonMsg = "";
+        let errorSubtext = "";
+
+        switch(statusCode)
+        {
+            case 404:
+                errorReasonMsg = "Not Found";
+                errorSubtext = "<!--%#INSERT:NAME#%--> couldn't find a resource that corresponds to the URL you have entered.<br>Please make sure the URL is correct or <a href=\"<!--%#INSERT:DOCSURL#%-->\">visit the documentation by clicking here</a>.";
+            break;
+            case 500: default:
+                errorReasonMsg = "Internal Server Error";
+                errorSubtext = "<!--%#INSERT:NAME#%--> encountered an unexpected internal error.<br>If this error persists and error details were provided on this page, please <a href=\"<!--%#INSERT:AUTHORWEBSITEURL#%-->\">contact me</a> with the error details and I will try to fix it and/or help you.<br>Alternatively, <a href=\"<!--%#INSERT:DOCSURL#%-->\">visit the documentation by clicking here</a>.";
+            break;
+        }
+
+        setErrorDisp(parseInt(statusCode), errorReasonMsg, errorInfo["API-Error-Message"], errorSubtext);
+    }
+    catch(err)
+    {
+        setErrorDisp(500, "Internal Server Error", "Error while finding the error message - oh the irony");
+    }
+});
+
+/**
+ * Sets the error display of the page
+ * @param {Number} code 
+ * @param {String} summary
+ * @param {String} details 
+ * @param {String} subText
+ */
+function setErrorDisp(code = 500, summary = "Internal Server Error", details = "No details provided", subText = "")
+{
+    window.errorWrittenToPage = true;
+    document.title = ("<!--%#INSERT:NAME#%--> - Error " + code.toString());
+    document.getElementById("errCodeDisplay").innerHTML = (code.toString() + " - " + summary);
+    document.getElementById("errDetailDisplay").innerHTML = "Details: " + details;
+
+    if(subText)
+        document.getElementById("errSubText").innerHTML = subText;
+}

+ 1152 - 0
docs/raw/index.css

@@ -0,0 +1,1152 @@
+@import url("https://fonts.googleapis.com/css?family=Roboto&display=swap");
+@import url("https://fonts.googleapis.com/css?family=Roboto+Mono&display=swap");
+
+:root {
+    --bg-color: #2e323b;
+    --bg-accent-color: #1d2025;
+    --bg-accent-color-darker: #111316;
+    --bg-accent-color-sidenav-opened: #4e535c;
+    --bg-accent-color-darker-sidenav-opened: #5e646e;
+    --bg-color-sidenav-opened: #616774;
+    --header-bg-color: #0b1031;
+    --header-bg-color-sidenav-opened: #0b1031;
+
+    --doc-header-color: #5a83c0;
+
+    --header-height: 50px;
+
+    --accent-color: #8a2be2;
+    --accent-color-light: #b05ffc;
+    --accent-color-sidenav-opened: #a779d3;
+
+    --doc-header-font-size: 29px;
+    --doc-sub-header-font-size: 22px;
+
+    --sidenav-animation-speed: 0.15s;
+    --sidenav-colorblur-speed: 0.5s;
+    --targetblink-animation-delay: 0.5s;
+
+    --scrollbar-track-color: #222;
+    --scrollbar-thumb-color: rgba(175, 175, 175, .7);
+    --scrollbar-webkit-hover-color: rgba(65, 131, 196, .8);
+
+    --line-height: calc(100% + 4px);
+}
+
+@font-face { 
+    font-family: "Cascadia Code";
+    src: url("./cascadia-code.ttf"); /* URL can't have the "./static/" path since this CSS file is already loaded from "./static/" */
+}
+
+
+
+
+#submissions-hidden {
+    display: none;
+}
+
+/* #MARKER BEGIN */
+html, body, #main {
+    scroll-behavior: smooth;
+}
+
+body {
+    margin: 0;
+    background-color: var(--bg-color);
+    color: #fff;
+    font-family: "Roboto", "Segoe UI", "Arial", sans-serif;
+    font-size: 17px;
+    line-height: var(--line-height);
+    font-weight: 200;
+
+    overflow-y: hidden;
+
+    transition: background-color ease-out var(--sidenav-colorblur-speed);
+
+    /* Ah yes web development */
+    -webkit-touch-callout: inherit;
+    -webkit-user-select: inherit;
+    -khtml-user-select: inherit;
+    -moz-user-select: inherit;
+    -ms-user-select: inherit;
+    user-select: inherit;
+}
+
+.noselect {
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+}
+
+.noa {
+    color: inherit;
+}
+
+.noul, .mimica {
+    cursor: pointer;
+}
+
+a:not(.noul), .mimica:not(.noul) {
+    color: #8f9aff;
+    text-decoration: none;
+}
+
+a:not(.noul):hover, .mimica:not(.noul):hover {
+    color: #bcc2ff;
+    text-decoration: underline;
+}
+
+.rtext {
+    text-align: right;
+}
+
+.blackshadow {
+    text-shadow:
+        -1px -1px 0 #000,  
+         1px -1px 0 #000,
+        -1px  1px 0 #000,
+         1px  1px 0 #000;
+}
+
+.whiteshadow {
+    text-shadow:
+        -1px -1px 0 #fff,  
+         1px -1px 0 #fff,
+        -1px  1px 0 #fff,
+         1px  1px 0 #fff;
+}
+
+body[data-sidenav="opened"] {
+    background-color: var(--bg-color-sidenav-opened);
+
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+}
+
+body[data-sidenav="closed"] {
+    background-color: var(--bg-color);
+
+    -webkit-touch-callout: inherit;
+    -webkit-user-select: inherit;
+    -khtml-user-select: inherit;
+    -moz-user-select: inherit;
+    -ms-user-select: inherit;
+    user-select: inherit;
+}
+
+.lText {
+    font-size: 120%;
+}
+
+/*#MARKER scrollbar*/
+::-webkit-scrollbar {
+    max-width: 10px !important;
+    max-height: 10px !important;
+    background: var(--scrollbar-track-color) !important;
+}
+
+::-webkit-scrollbar-track, ::-webkit-scrollbar-corner {
+    background: var(--scrollbar-track-color) !important;
+}
+
+::-webkit-scrollbar-thumb {
+    background: var(--scrollbar-thumb-color) !important;
+}
+
+::-webkit-scrollbar-thumb:hover {
+    background: var(--scrollbar-webkit-hover-color) !important;
+}
+
+:root {
+    scrollbar-color: var(--scrollbar-thumb-color) var(--scrollbar-track-color) !important;
+    scrollbar-width: thin !important;
+}
+
+/*#MARKER header*/
+header {
+    /* position: -webkit-sticky;
+    position: sticky; */
+    position: relative;
+    top: 0;
+    left: 0;
+    background-color: var(--header-bg-color);
+    /* background: linear-gradient(0deg, rgba(255,213,0,1) 0%, rgba(255,213,0,1) 50%, rgba(0,91,187,1) 50%, rgba(0,91,187,1) 100%); */
+    color: white;
+    text-shadow: 
+        -1px -1px 0px #000,
+         0px -1px 0px #000,
+         1px -1px 0px #000,
+        -1px  0px 0px #000,
+         1px  0px 0px #000,
+        -1px  1px 0px #000,
+         0px  1px 0px #000,
+         1px  1px 0px #000;
+    z-index: 1000;
+
+    transition: filter ease-out var(--sidenav-colorblur-speed), width linear var(--sidenav-animation-speed);
+
+    display: flex;
+    flex-direction: row;
+    flex-wrap: nowrap;
+
+    width: 100%;
+    height: var(--header-height);
+
+    filter: grayscale(0%) drop-shadow(2px 2px 3px rgba(0, 0, 0, 0.6));
+
+    line-height: initial;
+}
+
+header[data-grayscaled="true"] {
+    /* filter: grayscale(100%) drop-shadow(2px 2px 3px rgba(0, 0, 0, 0.3)); */
+    /* background-color: var(--header-bg-color-sidenav-opened); */
+    width: 100%;
+}
+
+header .headeritem {
+    display: inline-block;
+    white-space: nowrap;
+    flex-grow: 0;
+    padding-top: 5px;
+    padding-bottom: 5px;
+    width: 100%;
+}
+
+header .headeritem.m {
+    flex-grow: 1;
+    text-align: center;
+}
+
+header .headeritem.r {
+    text-align: right;
+}
+
+header #docTitle {
+    cursor: pointer;
+    margin: 0;
+    font-size: 30px;
+    color: #fff;
+}
+
+#headerversion {
+    font-size: 18px;
+    display: inline-block;
+    padding-right: 10px;
+}
+
+#headerVersionNumber {
+    font-size: 15px;
+    padding-right: 6px;
+}
+
+#changelogLink {
+    font-size: 15px;
+}
+
+#headersidenavopenwrapper {
+    padding-left: 10px;
+}
+
+#docTitle {
+    cursor: pointer;
+    display: inline-block;
+    color: #fff;
+    white-space: nowrap;
+
+    transition: color 0.4s linear, filter 0.2s linear, transform 0.4s cubic-bezier(0,0,.27,1.77);
+
+    transform: rotate(0deg) scale(1.0);
+    filter: none;
+}
+
+@media (max-width: 750px) {
+    header #docTitle {
+        font-size: 20px;
+        white-space: normal;
+    }
+
+    .mobileHide {
+        display: none;
+    }
+}
+
+#docTitle:hover {
+    color: var(--accent-color-light);
+
+    transform: rotate(1deg) scale(1.15);
+    filter: drop-shadow(2px 2px 3px rgba(0, 0, 0, 0.8));
+}
+
+.noa {
+    cursor: default;
+}
+
+.noa, header #docTitle:hover {
+    text-decoration: none;
+}
+
+/*#MARKER main content*/
+label {
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+}
+
+.centered {
+    text-align: center;
+}
+
+.miniimg {
+    height: 1.1em;
+}
+
+.miniimg.invert {
+    filter: invert(100%);
+}
+
+.shadow {
+    filter: drop-shadow(1px 1px 3px rgba(0, 0, 0, 0.5));
+}
+
+img.badge {
+    /* badges slide up a few pixels for some reason */
+    height: 1.15em;
+    position: relative;
+    bottom: -2.5px;
+}
+
+#f_langHideContainer.hidden {
+    visibility: hidden;
+}
+
+input#f_customLang {
+    width: 3em;
+}
+
+#uptimeTable tr td:last-child {
+    padding-left: 5px;
+}
+
+.alternatingBgTable {
+    margin: 0;
+    margin-top: 10px;
+    padding: 0;
+    border-spacing: 0;
+}
+
+.alternatingBgTable tr:nth-child(2n+1) td {
+    background-color: rgba(0, 0, 0, 0.2);
+}
+
+.alternatingBgTable tr:nth-child(1) > th {
+    border: 1px solid #fff;
+    border-style: none none solid none;
+    text-align: left;
+}
+
+.alternatingBgTable tr > td:nth-child(1) {
+    padding-top: 3px;
+    padding-left: 4px;
+    padding-right: 10px;
+}
+
+.alternatingBgTable tr th:not(:first-child), .alternatingBgTable tr td:not(:first-child) {
+    padding: 2px 10px 2px 10px;
+}
+
+.alternatingBgTable tr th:first-child {
+    padding-right: 10px;
+    padding-bottom: 2px;
+}
+
+.alternatingBgTable tr td:not(:first-child) {
+    border: 1px solid #fff;
+    border-style: none none none solid;
+}
+
+table tr td.g, span.col.g {
+    color: #65b365;
+}
+
+table tr td.r, span.col.r {
+    color: #ffad41;
+}
+
+table tr td.dr, span.col.dr {
+    color: #aa2828;
+}
+
+.colB {
+    color: #a9a9ff;
+}
+
+.colY {
+    color: #ffffa9;
+}
+
+details summary {
+    font-size: 110%;
+    padding-top: 2px;
+    padding-bottom: 5px;
+    cursor: pointer;
+}
+
+details summary.big {
+    font-size: 110%;
+}
+
+#newsList li {
+    padding-top: 10px;
+}
+
+mark {
+    border: 1px solid #777;
+    border-radius: 2px;
+    background-color: var(--bg-accent-color);
+    display: inline-block;
+    padding: 1px;
+    font-family: "Cascadia Code", "Roboto Mono", "Courier New", monospace;
+    font-weight: normal;
+    font-variant-ligatures: none;
+    color: #ccc;
+    transition: background-color ease-out var(--sidenav-colorblur-speed);
+}
+
+body[data-sidenav="opened"] mark {
+    background-color: var(--bg-accent-color-sidenav-opened);
+}
+
+#bodyFlexContainer {
+    display: flex;
+    flex-direction: column;
+    flex-wrap: nowrap;
+    justify-content: flex-start;
+    height: 100vh;
+    scroll-behavior: smooth;
+}
+
+#bodyFlexContainer .bodyFlexItem {
+    flex-grow: 1;
+}
+
+.jsg_menu {
+    z-index: 1001;
+}
+
+#devStuff {
+    display: inline-block;
+    width: 95vw;
+    border: 2px solid #f40;
+    border-radius: 10px;
+    padding: 10px;
+    margin-bottom: 20px;
+    margin-top: 20px;
+}
+
+#devStuff::before {
+    content: "Dev Stuff:";
+    font-size: 20px;
+    text-decoration: underline;
+    margin-bottom: 15px;
+    color: #f40;
+}
+
+#content {
+    overflow-y: auto;
+    padding: 26px;
+    margin-left: 10px;
+
+    padding-top: 0;
+    margin-top: 0;
+
+    transition: margin-left var(--sidenav-animation-speed);
+}
+
+
+@media (max-width: 750px) {
+    #content {
+        padding: 10px;
+        margin-left: 5px;
+    }
+}
+
+body[data-sidenav="closed"] code, body code {
+    font-family: "Cascadia Code", "Inconsolata", "Courier New", "Consolas", monospace;
+    font-weight: normal;
+    font-variant-ligatures: none;
+    font-size: 15px;
+
+    white-space: pre;
+    -moz-tab-size: 4;
+    -o-tab-size: 4;
+    tab-size: 4;
+
+    background-color: #111;
+    display: block;
+
+    border-radius: 8px !important;
+    border-left: 8px solid var(--accent-color) !important;
+
+    filter: grayscale(0%);
+
+    transition: filter ease-out var(--sidenav-colorblur-speed);
+}
+
+body code > .actualCode {
+    padding: 0px 10px 18px 20px;
+    overflow-x: auto;
+    overflow-y: hidden;
+    line-height: 1.25;
+}
+
+body code .pln:first {
+    display: none;
+}
+
+body[data-sidenav="opened"] code {
+    background-color: #555;
+
+    filter: grayscale(50%);
+
+    transition: filter ease-out var(--sidenav-colorblur-speed);
+}
+
+body code .nocode.codeheader {
+    font-family: "Roboto Mono", monospace;
+    font-size: 19px;
+
+    display: block;
+    padding-top: 6px;
+    padding-bottom: 4px;
+    
+    border: 3px solid var(--accent-color);
+    border-style: none none solid none;
+}
+
+body code .nocode.codeheader::before {
+    font-size: 19px;
+    color: var(--accent-color-light);
+    content: "</>";
+    padding: 0px 15px;
+}
+
+kbd {
+    font-family: "Cascadia Code", "Inconsolata", "Roboto Mono", "Courier New", monospace;
+    font-weight: normal;
+    font-variant-ligatures: none;
+    font-size: 13px;
+    display: inline-block;
+    padding: 3.5px 4px;
+    line-height: 10px;
+    color: #eee;
+    vertical-align: middle;
+    background-color: #555;
+    border: 1px solid #333;
+    border-radius: 3px;
+    box-shadow: inset 0 -2px 0 #222;
+}
+
+.indented {
+    display: inline-block;
+    margin-left: 20px;
+}
+
+ul {
+    margin-top: 5px;
+    margin-bottom: 5px;
+}
+
+abbr {
+    cursor: help;
+}
+
+ul.lispacer li {
+    margin-bottom: 5px;
+}
+
+.antiBotE:not(.noul):not(.shown) {
+    cursor: pointer;
+    color: #8f9aff;
+    text-decoration: none;
+}
+
+.antiBotE:not(.noul):not(.shown):hover {
+    color: #bcc2ff;
+    text-decoration: underline;
+}
+
+.antiBotE.shown {
+    font-family: "Roboto Mono", "Cascadia Code", "Courier New", monospace;
+    font-weight: normal;
+}
+
+/*#SECTION Endpoints*/
+.requestURLwrapper {
+    border: 1px solid #777;
+    border-radius: 2px;
+    background-color: var(--bg-accent-color-darker);
+    display: inline-block;
+    font-size: 110%;
+    padding: 3px;
+    padding-left: 10px;
+    padding-right: 40px;
+    transition: background-color ease-out var(--sidenav-colorblur-speed);
+    white-space: nowrap;
+}
+
+body[data-sidenav="opened"] .requestURLwrapper {
+    background-color: var(--bg-accent-color-darker-sidenav-opened);
+    transition: background-color ease-out var(--sidenav-colorblur-speed);
+}
+
+.requestMethodGET, .requestMethodPUT {
+    display: inline-block;
+    margin-right: 10px;
+
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+}
+
+.requestMethodGET {
+    color: #7cfc00;
+}
+
+.requestMethodPUT {
+    color: #6495ed;
+}
+
+.requestURL {
+    color: #ccc;
+    font-family: "Roboto Mono", "Courier New", monospace;
+}
+
+.supportedParamsBox {
+    display: inline-block;
+    /* border: 1px solid #bbb;
+    border-radius: 2px;
+    background-color: var(--bg-accent-color);
+    padding: 6px; */
+    padding-right: 30px;
+}
+
+.placeholder {
+    color: #777;
+}
+
+footer#copyrightNotice {
+    display: block;
+    margin-bottom: 10px;
+    text-align: center;
+}
+
+footer#copyrightNotice span {
+    display: inline-block;
+}
+
+/*#SECTION Wrappers*/
+#wrapperTable {
+    border-spacing: 0;
+    font-size: 110%;
+}
+
+#wrapperTable tr th {
+    text-align: left;
+    padding: 2px;
+    padding-right: 20px;
+    border: 1px solid white;
+    border-style: none none solid none;
+}
+
+#wrapperTable tr th:first-child {
+    padding-right: 20px;
+}
+
+#wrapperTable tr td {
+    text-align: left;
+    padding: 2px;
+    padding-right: 20px;
+}
+
+#wrapperTable tr td.wrappericon {
+    padding-right: 5px;
+}
+
+/*#SECTION submit*/
+
+#submitContainer {
+    display: flex;
+    flex-direction: row;
+    flex-wrap: nowrap;
+
+    margin-left: 20px;
+}
+
+.submitFlexItem {
+    flex-grow: 1;
+    white-space: nowrap;
+}
+
+.submitFlexItem.l {
+    width: 100%;
+    margin-left: 30px;
+}
+
+@media (max-width: 1100px) {
+    #submitContainer {
+        flex-wrap: wrap;
+        flex-direction: column;
+    }
+
+    .submitFlexItem {
+        white-space: normal;
+    }
+
+    .submitFlexItem.l {
+        margin-top: 30px;
+        margin-left: 0;
+    }
+}
+
+#submitBtn {
+    font-size: 1.5em;
+}
+
+#submitBtn:disabled {
+    cursor: not-allowed;
+}
+
+textarea {
+    white-space: pre;
+    overflow: auto;
+    resize: vertical;
+
+    width: 100%;
+    max-height: 8em;
+}
+
+#propertyTable tr td {
+    vertical-align: top;
+    padding-right: 5px;
+}
+
+/* #SECTION contributors */
+
+#contributorsContainer .contributor {
+    margin-top: 30px;
+}
+
+#contributorsContainer .contributor .contributorName {
+    font-size: 115%;
+    font-weight: 125%;
+}
+
+#contributorsContainer .contributor .contributorContact {
+    display: inline;
+}
+
+#contributorsContainer .contributor .contributorContact:after {
+    content:"\a";
+    white-space: pre;
+}
+
+/*#MARKER sidenav*/
+#sidenav {
+    display: flex;
+    flex-direction: column;
+    justify-content: stretch;
+
+    height: 100%;
+    width: 0;
+    position: fixed;
+    z-index: 1;
+    top: 0;
+    left: 0;
+    background-color: #111;
+    overflow-x: hidden;
+    overflow-y: hidden;
+    transition: var(--sidenav-animation-speed);
+    z-index: 1002;
+
+    filter: drop-shadow(2px 2px 3px rgba(0, 0, 0, 0.6));
+
+    white-space: nowrap;
+    line-height: initial;
+}
+
+#sidenav a, .sidenav-mimica {
+    flex-grow: 1;
+
+    padding: 5px 8px 5px 32px;
+    text-decoration: none;
+    font-size: 20px;
+    color: #a5a5a5;
+    display: block;
+    transition: 0.3s;
+}
+
+#sidenav a:hover, .sidenav-mimica:hover {
+    cursor: pointer;
+    color: #f1f1f1;
+}
+
+#sidenav details > summary {
+    padding-left: 32px;
+}
+
+#sidenav details > summary:first-of-type {
+    list-style-type: none;
+}
+
+#sidenav details > summary:-webkit-details-marker {
+    display: none;
+}
+
+#sidenav details:not([open]) > summary::before {
+    flex-grow: 1;
+
+    padding-left: 0px;
+    padding-top: 0px;
+    content: "⯈";
+    transform: rotate(0deg);
+    display: inline-block;
+
+    transition: transform 0.1s ease-out;
+}
+
+#sidenav details[open] > summary::before {
+    flex-grow: 1;
+
+    padding-left: 0px;
+    padding-top: 0px;
+    content: "⯈";
+    transform: rotate(90deg);
+    display: inline-block;
+
+    transition: transform 0.1s ease-out;
+}
+
+#sidenav details > summary > span.sidenav-mimica {
+    display: inline-block;
+    width: 100%;
+    padding-left: 0px;
+}
+
+#sidenav details > summary {
+    cursor: pointer;
+    color: #a5a5a5;
+    transition: 0.3s;
+}
+
+#sidenav details > summary:hover {
+    color: #f1f1f1;
+    transition: 0.3s;
+}
+
+#sideNavOpen {
+    white-space: nowrap;
+    font-size: 30px;
+    cursor: pointer;
+}
+
+#sideNavOpen::after {
+    content: "Menu";
+}
+
+@media (max-width: 750px) {
+    #sideNavOpen::after {
+        content: "";
+    }
+
+    #changelogLink {
+        font-size: 10px;
+    }
+}
+
+#sidenav ul {
+    flex-grow: 1;
+
+    padding-top: 0;
+    margin-top: 0;
+    list-style-type: none;
+    margin-left: 25px;
+    padding-left: 0;
+}
+
+#sidenav ul > li > a {
+    font-size: 15px;
+}
+
+#sidenav #sidenavContent {
+    overflow-y: auto;
+    overflow-x: hidden;
+}
+
+#sidenav .bottomspacer {
+    margin-bottom: 50px;
+}
+
+#sidenav #sidenavHeader {
+    display: flex;
+    flex-direction: row;
+    flex-wrap: nowrap;
+    font-size: 36px;
+    width: 100%;
+}
+
+#sidenav #sidenavHeader .sidenavHeaderItem:not(.nopad) {
+    flex-grow: 1;
+    padding: 10px;
+}
+
+#sidenav #sidenavHeader .sidenavHeaderItem.nopad {
+    flex-grow: 1;
+    padding: 0px;
+    padding-right: 15px;
+}
+
+#sidenav #sidenavHeader .sidenavHeaderItem.spacer {
+    flex-grow: 5;
+}
+
+#sidenav .closebtn {
+    font-size: 36px;
+}
+
+div.sidenavSpacer {
+    display: block;
+    min-height: 18px;
+    max-height: 18px;
+}
+
+@media screen and (max-height: 450px) {
+    .sidenav {padding-top: 15px;}
+    .sidenav a {font-size: 18px;}
+}
+
+.docHeader .headerAnchorContainer {
+    cursor: pointer;
+    margin-right: 10px;
+    filter: opacity(30%);
+
+    transition: filter 0.075s linear;
+}
+
+.docHeader .headerAnchorContainer:hover {
+    filter: opacity(100%);
+    text-decoration: none !important;
+}
+
+.headerAnchorContainer {
+    display: inline-block;
+    margin-top: 20px;
+}
+
+.targetBlink:target {
+    animation: targetBlink 1s;
+    animation-iteration-count: 1;
+    animation-delay: var(--targetblink-animation-delay);
+}
+
+@keyframes targetBlink {
+    0% {
+        border-color: var(--doc-header-color);
+        color: #fff;
+
+        border-width: 2px;
+    }
+    10% {
+        border-width: 200%;
+    }
+    25% {
+        border-color: #f00;
+        color: #f00;
+    }
+    75% {
+        border-color: #f00;
+        color: #f00;
+    }
+    90% {
+        border-width: 200%;
+    }
+    100% {
+        border-color: var(--doc-header-color);
+        color: #fff;
+
+        border-width: 2px;
+    }
+}
+
+.docHeader {
+    font-size: var(--doc-header-font-size);
+    border: 2px solid var(--doc-header-color);
+    border-style: none none solid none;
+    padding-bottom: 2px;
+    margin-bottom: 10px;
+    margin-top: 70px;
+    line-height: initial;
+}
+
+.docHeader:not(:first)
+{
+    margin-top: 50px;
+}
+
+.subHeaderContainer {
+    display: block;
+    margin-top: 25px;
+    margin-bottom: 8px;
+    text-decoration: underline;
+    text-underline-offset: 3px; /* currently only supported in FF70+ and SF12.1+ (see https://caniuse.com/#search=text-underline-offset) */
+}
+
+.subHeaderContainer .subHeader {
+    font-size: var(--doc-sub-header-font-size);
+}
+
+.subHeaderContainer a {
+    vertical-align: super;
+    font-size: smaller;
+    text-decoration: none;
+    text-underline-offset: initial;
+
+    color: rgba(188, 194, 255, 0.4);
+    transition: color 0.075s linear;
+}
+
+.smallText {
+    font-size: small;
+}
+
+.subHeaderContainer a:hover {
+    color: rgba(188, 194, 255, 1.0);
+    text-decoration: inherit;
+}
+
+/*#SECTION usage terms */
+#usageTerms {
+    display: none;
+    background-color: var(--bg-accent-color);
+    border: 2px solid #fff;
+    border-radius: 10px;
+    padding: 13px;
+    padding-top: 20px;
+    padding-bottom: 20px;
+    margin-bottom: 20px;
+    margin-top: 20px;
+}
+
+#usageTerms[data-animate-border="true"] {
+    animation: targetBlink 1s;
+    animation-iteration-count: 1;
+}
+
+#usageTerms > h2 {
+    margin-top: 0;
+    color: orangered;
+}
+
+#usageTermsInnerLayout {
+    display: flex;
+    flex-direction: row;
+    flex-wrap: nowrap;
+}
+
+#usageTermsInnerLayout > .itm {
+    flex-grow: 1;
+}
+
+#usageTermsInnerLayout > .itm:last-of-type {
+    text-align: right;
+}
+
+/*#SECTION try it */
+#tryItContainer table tr td {
+    vertical-align: middle;
+}
+
+#tryItContainer table tr:not(:last-of-type) > td {
+    padding-bottom: 15px;
+}
+
+#tryItFormLatency {
+    font-size: 90%;
+}
+
+.highlightedContainer {
+    display: block;
+    background-color: var(--bg-accent-color);
+    border: 2px solid #fff;
+    border-radius: 10px;
+    padding: 10px;
+    padding-top: 20px;
+    padding-bottom: 20px;
+    margin-bottom: 20px;
+    margin-top: 20px;
+}
+
+body[data-sidenav="opened"] .highlightedContainer {
+    background-color: var(--bg-accent-color-sidenav-opened);
+}
+
+#tryItContainer table tr td {
+    padding-right: 20px;
+}
+
+#catSelectMulti {
+    padding-left: 20px;
+}
+
+.multiselect, #urlBuilderWrapper, #resultWrapper {
+    background-color: #222;
+    border: 1px solid #fff;
+    padding: 6px;
+    padding-right: 11px;
+    border-radius: 5px;
+}
+
+body[data-sidenav="opened"] .multiselect, body[data-sidenav="opened"] #urlBuilderWrapper, body[data-sidenav="opened"] #resultWrapper {
+    background-color: #555;
+}
+
+input[disabled] {
+    cursor: not-allowed;
+}
+
+input[disabled]+label {
+    color: #bbb;
+    cursor: not-allowed;
+}
+
+#searchStringInput {
+    width: 100%;
+}
+
+#urlBuilderWrapper {
+    display: inline-block;
+    margin-bottom: 12px;
+    font-size: 16px;
+}
+
+#urlBuilderWrapper button {
+    font-size: 17px;
+    padding: 3px 8px 3px 8px;
+}
+
+#urlBuilderUrl {
+    display: inline-block;
+    padding: 10px 5px 10px 8px;
+}

+ 1762 - 0
docs/raw/index.html

@@ -0,0 +1,1762 @@
+<!--
+    JokeAPI Documentation
+
+    By Sv443 (https://sv443.net or https://github.com/Sv443)
+    Licensed under the MIT license (https://sv443.net/LICENSE)
+-->
+
+
+<!DOCTYPE html>
+<html lang="en">
+    <head>
+        <!-- #MARKER head -->
+        <title><!--%#INSERT:NAME#%--> - Documentation</title>
+
+        <meta charset="UTF-8">
+        <meta name="viewport" content="width=device-width, initial-scale=1.0">
+        <meta name="pagename" content="<!--%#INSERT:NAME#%-->">
+
+        <meta charset="UTF-8">
+        <meta name="date" content="<!--%#INSERT:LASTMODIFIEDISO#%-->"> <!-- Date at which website was modified last or since <%#INSERT:NAME#%>'s latest startup -->
+        <meta name="pagename" content="<!--%#INSERT:NAME#%-->">
+        <meta name="viewport" content="width=device-width, initial-scale=1.0">
+        <meta name="HandheldFriendly" content="true">
+        <meta name="description" content="<!--%#INSERT:NAME#%--> is a RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta name="twitter:card" content="<!--%#INSERT:NAME#%--> is a RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta name="subject" content="API">
+        <meta name="copyright" content="Sv443 - Licensed under MIT license (https://sv443.net/LICENSE)">
+        <meta name="author" content="Sv443">
+        <meta property="og:title" content="<!--%#INSERT:NAME#%-->">
+        <meta property="og:url" content="<!--%#INSERT:DOCSURL#%-->">
+        <meta property="og:description" content="<!--%#INSERT:NAME#%--> is a RESTful API that serves jokes from many categories while also offering a lot of filtering methods">
+        <meta property="og:image" content="./favicon.ico">
+        <meta property="og:type" content="profile">
+        <meta property="og:locale" content="en_US">
+        <meta property="og:locale:alternate" content="en_GB">
+        <meta property="og:locale:alternate" content="en_AU">
+        <meta name="application-name" content="<!--%#INSERT:NAME#%-->">
+        <meta name="theme-color" content="#101747">
+
+        <link rel="icon" type="image/png" href="./favicon.ico">
+
+        <!-- IE detection -->
+        <script src="https://sv443.net/cdn/jokeapi/isIE.js"></script>
+
+        <!-- #MARKER dependencies -->
+        <!-- Have to do it with an absolute path because the page doesn't always have a trailing slash and that *somehow* makes it redirect to the page root: -->
+        <!-- <link rel="stylesheet" href="<%#INSERT:DOCSURL#%>/static/index.css">-->
+        <link rel="stylesheet" href="static/index.css">
+
+        <script src="https://cdn.jsdelivr.net/gh/Sv443/[email protected]/jslib.js"></script>
+
+        <!-- Have to do it with an absolute path because the page doesn't always have a trailing slash and that *somehow* makes it redirect to the page root: -->
+        <!-- <script src="<%#INSERT:DOCSURL#%>/static/index.js"></script>-->
+        <script src="static/index.js"></script>
+
+        <script src="https://cdn.jsdelivr.net/gh/google/code-prettify@master/loader/run_prettify.js?skin=sunburst"></script> <!-- Google Code-Prettify -->
+    </head>
+    <body onload="onLoad()">
+        <!-- #MARKER Main -->
+        <div id="bodyFlexContainer">
+            <!-- #MARKER Header -->
+            <header class="bodyFlexItem">
+                <span class="headeritem l">
+                    <span id="headersidenavopenwrapper" class="noselect">
+                        <a id="sideNavOpen" class="noul" title="Click to open">☰</a>
+                    </span>
+                </span>
+                <span class="headeritem m">
+                    <a id="docTitle" href="<!--%#INSERT:DOCSURL#%-->" class="noselect" title="Click to reload"><!--%#INSERT:NAME#%--> Documentation</a>
+                </span>
+                <span class="headeritem r">
+                    <span id="headerversion">
+                        <span id="headerVersionNumber">v<!--%#INSERT:VERSION#%--></span>
+                        <a id="changelogLink" href="https://github.com/Sv443/JokeAPI/blob/master/changelog.md#readme" target="_blank">(Changelog)</a><br>
+                        <a href="<!--%#INSERT:PROJGITHUBURL#%-->">GitHub</a>&nbsp;<span class="mobileHide">&bull;&nbsp;<a href="#legal">Legal&nbsp;Stuff</a>&nbsp;</span>&bull;&nbsp;<a href="<!--%#INSERT:AUTHORGITHUBURL#%-->">Author</a>
+                    </span>
+                </span>
+            </header>
+            <main id="content" class="bodyFlexItem">
+                <noscript>
+                    <h3 style="color: #ff7644;">You have disabled the execution of JavaScript code in your browser. Please note that this will severely impact your experience on this page.</h3>
+                    <span style="color: #ff7644;">
+                        If you didn't disable JavaScript code, the website might have loaded incorrectly. In this case, please press the <kbd>F5</kbd> key to try again.
+                    </span>
+                    <br><br>
+                </noscript>
+
+                <div id="usageTerms">
+                    <h2 style="color: #ff7644;">By using this website and API you are agreeing to the <a href="<!--%#INSERT:PRIVACYPOLICYURL#%-->">privacy policy</a></h2>
+                    <div id="usageTermsInnerLayout">
+                        <span class="itm">
+                            <span class="mimica" onclick="privPolMoreInfo()">(Click here to get more information on what data <!--%#INSERT:NAME#%--> collects)</span>
+                        </span>
+                        <span class="itm">
+                            <span class="mimica" onclick="hideUsageTerms()">
+                                <span class="lText">&#10004;</span> I agree
+                            </span>
+                        </span>
+                    </div>
+                </div>
+
+                <!-- #MARKER Information -->
+                <div class="docHeader targetBlink" id="info"><a class="headerAnchorContainer" href="#info" title="Click to link to this header">🔗</a><span class="docHeaderText">Information:</span></div>
+
+                <!--%#INSERT:NAME#%--> is a REST API that serves uniformly and well formatted jokes.<br>
+                It can be used <u>without any API token, membership, registration or payment.</u><br>
+                It supports a variety of filters that can be applied to get just the right jokes you need.<br>
+                The usage is very simple and similar to other RESTful APIs and requires only basic knowledge of HTTP requests and JSON, XML, YAML or plain text.<br>
+                <br>
+                If you come across a term you don't understand, you can try <span class="mimica" onclick="openNav()">opening the menu</span> and clicking on the section "Terminology".<br>
+                <!--%#INSERT:NAME#%--> currently serves <!--%#INSERT:TOTALJOKES#%--> jokes from <!--%#INSERT:JOKELANGCOUNT#%--> different languages. To submit a new joke, <a href="#submit">please click here.</a><br>
+                <br>
+                To report a bug or a grammar mistake or to suggest a feature, please <a href="<!--%#INSERT:PROJGITHUBURL#%-->/issues/new/choose" target="_blank">use the GitHub issue tracker.</a><br>
+                If you want to contribute to <!--%#INSERT:NAME#%-->, please read the <a href="<!--%#INSERT:PROJGITHUBURL#%-->/blob/master/.github/Contributing.md" target="_blank">Contributing Guide.</a><br>
+                <br>
+                <span style="color: #ff8b61;">If you are new to <!--%#INSERT:NAME#%-->, please start by reading the <a href="#getting-started">Getting Started Guide</a></span><br>
+
+                <br>
+
+                <!-- #SECTION Note -->
+                <div class="subHeaderContainer"><a href="#note" title="Click to link to this header">§</a> <span class="subHeader" id="note">Note:</span></div>
+                If you enjoy using <!--%#INSERT:NAME#%--> and want to support the development, please consider donating <a href="https://github.com/sponsors/Sv443" target="_blank">here ♥</a><br>
+                <br>
+                I can't speak all languages, so please note that I can't ensure format and validity of jokes that aren't in English or German.<br>
+                If you do come across an invalid joke or a joke that is formatted incorrectly, please consider <a href="<!--%#INSERT:PROJGITHUBURL#%-->/issues/new?assignees=Sv443&labels=reported+joke&template=3_report_a_joke.md&title=" target="_blank">opening an issue</a> or <a href="<!--%#INSERT:PROJGITHUBURL#%-->/blob/master/.github/Contributing.md#submitting-or-editing-jokes" target="_blank">submitting a pull request.</a><br>
+
+                <br>
+
+                <!-- #SECTION Uptime -->
+                <div class="subHeaderContainer"><a href="#uptime" title="Click to link to this header">§</a> <span class="subHeader" id="uptime">Uptime:</span></div>
+                <table id="uptimeTable">
+                    <tr>
+                        <td>Last day:</td>
+                        <td><img src="https://img.shields.io/uptimerobot/ratio/1/m784261094-bff76b959ebb8fc39f7eb2d0" class="badge"></td>
+                    </tr>
+                    <tr>
+                        <td>Last week:</td>
+                        <td><img src="https://img.shields.io/uptimerobot/ratio/7/m784261094-bff76b959ebb8fc39f7eb2d0" class="badge"></td>
+                    </tr>
+                    <tr>
+                        <td>Last month:</td>
+                        <td><img src="https://img.shields.io/uptimerobot/ratio/30/m784261094-bff76b959ebb8fc39f7eb2d0" class="badge"></td>
+                    </tr>
+                </table>
+                <a href="https://status.sv443.net/" target="_blank">(more stats)</a>
+                <br>
+
+                <br>
+
+                <!-- #SECTION Security -->
+                <div class="subHeaderContainer"><a href="#security" title="Click to link to this header">§</a> <span class="subHeader" id="security">Security:</span></div>
+                <!--%#INSERT:NAME#%--> has been a target of DoS attacks in the past, which is why there is now a limit of <!--%#INSERT:RATELIMITCOUNT#%--> requests per minute and why joke submissions are manually curated. For more information <a href="#rate-limiting">click here.</a><br>
+                <br>
+                All dependencies are tested for vulnerabilities by <a href="https://snyk.io/" target="_blank">Snyk</a>, you can find a summary of the known vulnerabilities <a href="https://snyk.io/test/github/Sv443/JokeAPI" target="_blank">here.</a><br>
+                Page security score (powered by Mozilla Observatory): <a href="https://observatory.mozilla.org/analyze/sv443.net" target="_blank"><img src="https://img.shields.io/mozilla-observatory/grade-score/sv443.net?publish" class="badge"></a><br>
+
+
+                <!-- #MARKER Information -->
+                <div class="docHeader targetBlink" id="news"><a class="headerAnchorContainer" href="#news" title="Click to link to this header">🔗</a><span class="docHeaderText">News:</span></div>
+                <ul id="newsList">
+                    <li>
+                        <b><!--%#INSERT:NAME#%--> now has its own domain!</b><br>
+                        You can now replace the base URL <mark>https://sv443.net/jokeapi/v2</mark> with <mark>https://v2.jokeapi.dev/</mark><br>
+                        Note: you should always use the <mark>v2</mark> subdomain to call the API but you can omit it if you just want to view the documentation.<br>
+                        For backwards compatibility the old URL will still be usable until version 2 will be deprecated.
+                    </li>
+                    <li>
+                        <b>Is <!--%#INSERT:NAME#%--> serving too many offensive jokes?</b><br>
+                        Try using version 2.3.0's new <a href="#safe-mode">safe mode</a> and the <a href="#blacklist-flags">explicit flag.</a>
+                    </li>
+                </ul>
+
+
+                <!-- #MARKER Try It -->
+                <div class="docHeader targetBlink" id="try-it"><a class="headerAnchorContainer" href="#try-it" title="Click to link to this header">🔗</a><span class="docHeaderText">Try it out here:</span></div>
+                
+                <div id="tryItContainer" class="highlightedContainer">
+                    <table>
+                        <tr>
+                            <td>
+                                Select <a href="#categories">category / categories</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect" id="categoryWrapper">
+                                    <div>
+                                        <input type="radio" name="catSelect" value="any" id="cat-radio1" onchange="reRender()"><label for="cat-radio1">Any</label>
+                                    </div>
+                                    <div>
+                                        <input type="radio" name="catSelect" value="multi" id="cat-radio2" onchange="reRender()"><label for="cat-radio2">Custom:</label>
+                                        <span id="catSelectMulti">
+                                            <input type="checkbox" id="cat-cb1" onchange="reRender()" disabled><label for="cat-cb1">Programming</label>
+                                            <input type="checkbox" id="cat-cb2" onchange="reRender()" disabled><label for="cat-cb2">Misc</label>
+                                            <input type="checkbox" id="cat-cb3" onchange="reRender()" disabled><label for="cat-cb3">Dark</label>
+                                            <input type="checkbox" id="cat-cb4" onchange="reRender()" disabled><label for="cat-cb4">Pun</label>
+                                            <input type="checkbox" id="cat-cb5" onchange="reRender()" disabled><label for="cat-cb5">Spooky</label>
+                                            <input type="checkbox" id="cat-cb6" onchange="reRender()" disabled><label for="cat-cb6">Christmas</label>
+                                        </span>
+                                    </div>
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Select <a href="#languages-endpoint">language</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect">
+                                    <select id="lcodeSelect" class="appendLangOpts"></select>
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Select <a href="#blacklist-flags">flags</a> to blacklist:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect">
+                                    (optional)&nbsp;&nbsp;&nbsp;
+                                    <input type="checkbox" id="blf-cb1" onchange="reRender()"><label for="blf-cb1">nsfw</label>
+                                    <input type="checkbox" id="blf-cb2" onchange="reRender()"><label for="blf-cb2">religious</label>
+                                    <input type="checkbox" id="blf-cb3" onchange="reRender()"><label for="blf-cb3">political</label>
+                                    <input type="checkbox" id="blf-cb4" onchange="reRender()"><label for="blf-cb4">racist</label>
+                                    <input type="checkbox" id="blf-cb5" onchange="reRender()"><label for="blf-cb5">sexist</label>
+                                    <input type="checkbox" id="blf-cb6" onchange="reRender()"><label for="blf-cb6">explicit</label>
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Select <a href="#response-formats">response format</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect">
+                                    <input type="radio" id="fmt-cb1" name="formatRadio" value="json" onchange="reRender()"><label for="fmt-cb1">default (json)</label>
+                                    <input type="radio" id="fmt-cb2" name="formatRadio" value="xml" onchange="reRender()"><label for="fmt-cb2">xml</label>
+                                    <input type="radio" id="fmt-cb3" name="formatRadio" value="yaml" onchange="reRender()"><label for="fmt-cb3">yaml</label>
+                                    <input type="radio" id="fmt-cb4" name="formatRadio" value="txt" onchange="reRender()"><label for="fmt-cb4">plain text</label>
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Select at least one <a href="#joke-type">joke type</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect" id="typeSelectWrapper">
+                                    <input type="checkbox" id="typ-cb1" onchange="reRender()"><label for="typ-cb1">single</label>
+                                    <input type="checkbox" id="typ-cb2" onchange="reRender()"><label for="typ-cb2">twopart</label>
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Search for a joke that<br>contains this <a href="#search-string">search string</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect">
+                                    <input type="text" id="searchStringInput" oninput="reRender()" placeholder="(optional)">
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                Search for a joke<br>in this <a href="#id-range">ID range</a>:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect" id="idRangeWrapper">
+                                    (optional)&nbsp;&nbsp;&nbsp;
+                                    From: <input type="number" id="idRangeInputFrom" value="0" min="0" step="1" oninput="reRender()" max="<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->">
+                                    To: <input type="number" id="idRangeInputTo" value="<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->" min="0" step="1" oninput="reRender()" max="<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->">
+                                </div>
+                            </td>
+                        </tr>
+                        <tr>
+                            <td>
+                                <a href="#amount">Amount</a> of jokes:
+                            </td>
+                            <td>
+                                <div class="multiselect noselect" id="jokeAmountWrapper">
+                                    <input type="number" id="jokesAmountInput" value="1" min="1" step="1" oninput="reRender()" max="<!--%#INSERT:MAXJOKEAMOUNT#%-->">
+                                </div>
+                            </td>
+                        </tr>
+                    </table>
+                    <br>
+                    <br>
+                    <div id="urlBuilderWrapper" class="flexItem">
+                        URL: <span id="urlBuilderUrl">(Loading...)</span><br>
+                        <button onclick="resetTryItForm(true)">Reset Form</button> <button onclick="sendTryItRequest()">Send Request &gt;</button>
+                    </div>
+<code id="urlBuilderPrettyprint" class="lang-json flexItem r"><span class="nocode codeheader">Result:</span>
+<div class="actualCode" id="tryItResult">(Set parameters and click "Send Request" above)</div>
+</code>
+                    <br>
+                    <span id="tryItFormLatency"></span>
+                </div>
+
+
+                <!-- #MARKER Getting Started -->
+                <div class="docHeader targetBlink" id="getting-started"><a class="headerAnchorContainer" href="#getting-started" title="Click to link to this header">🔗</a><span class="docHeaderText">Getting Started:</span></div>
+                <!--%#INSERT:NAME#%--> doesn't require any API token, authorization or payment to work, although there is a hard limit of <!--%#INSERT:RATELIMITCOUNT#%--> requests per minute due to some DoS attacks I have been getting.<br>
+                To start using <!--%#INSERT:NAME#%-->, you can do one of these two things:<br>
+                <ul class="lispacer">
+                    <li>
+                        You can see if your language has a wrapper for <!--%#INSERT:NAME#%--> (to see a list of officially supported wrappers, <a href="#wrappers">click here</a>)<br>
+                        Then, follow its documentation to communicate with the API (Disclaimer: I don't take any responsibility for their code).
+                    </li>
+                    <li>
+                        You can directly communicate with <!--%#INSERT:NAME#%--> with HTTP requests (XHR) from within your code.<br>
+                        Here's a few resources to get you started: (<a href="https://medium.com/better-programming/crash-course-in-http-requests-using-python-59246373a187" target="_blank">Python XHR Crash Course</a>, <a href="https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest" target="_blank">JavaScript XHR Documentation</a>, <a href="https://docs.microsoft.com/en-us/dotnet/api/system.net.http.httpclient?view=netcore-3.1" target="_blank">.NET HttpClient Class Documentation</a>)<br>
+                        Once you know how to send HTTP requests, please take a look at <a href="#endpoints">all the endpoints <!--%#INSERT:NAME#%--> has.</a><br>
+                        After you've finished reading that, you can start testing out the different endpoints with a client such as <a href="https://www.getpostman.com/" target="_blank">Postman</a> and look at some examples by <a href="#examples">going to the "Examples" section.</a>
+                    </li>
+                </ul>
+                <br>
+                <br>
+                If your project really can't contain any explicit jokes, consider enabling the <a href="#safe-mode">Safe Mode.</a><br>
+                <br>
+                If you know any good jokes and want to add them to <!--%#INSERT:NAME#%-->, please <a href="#submit">click here.</a><br>
+                If you need any help, feel free to <a href="https://sv443.net/discord" target="_blank">join my Discord server.</a><br>
+                <br>
+                I struggled at explaining some features so consult the provided examples if you don't understand them.<br>
+                
+
+                <!-- #MARKER Wrappers -->
+                <div class="docHeader targetBlink" id="wrappers"><a class="headerAnchorContainer" href="#wrappers" title="Click to link to this header">🔗</a><span class="docHeaderText">Official API Wrappers:</span></div>
+                A wrapper library translates some API into an interface that is easier to use for a certain language or environment.<br>
+                In the case of RESTful APIs like <!--%#INSERT:NAME#%-->, a wrapper library makes it so you can use your language's features to communicate with the API while the wrapper will do all the HTTP requests for you in the background.<br><br>
+                Disclaimer: I will not take any responsibility for the wrappers, they are community-made and I can't ensure security, validity and compatibility.<br>
+                Also, some of these might be outdated or not finished yet. Please consider supporting the creators or contributing to the libraries :)<br>
+                <br>
+                The following is a list of officially supported <!--%#INSERT:NAME#%--> wrappers:<br>
+
+                <br>
+
+                <table id="wrapperTable">
+                    <tr>
+                        <th></th>
+                        <th>Language</th>
+                        <th>Wrapper</th>
+                        <th>Author</th>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg invert" src="static/rust-icon"></td>
+                        <td>Rust</td>
+                        <td><a href="https://github.com/canarado/sv443_jokeapi_wrapper#readme" target="_blank">Sv443_JokeAPI Rust Crate</a></td>
+                        <td><a href="https://github.com/canarado" target="_blank">canarado</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/python-icon"></td>
+                        <td>Python</td>
+                        <td><a href="https://github.com/thenamesweretakenalready/Sv443s-JokeAPI-Python-Wrapper#readme" target="_blank">Sv443s JokeAPI Python Wrapper</a></td>
+                        <td><a href="https://github.com/thenamesweretakenalready" target="_blank">thenameswere<wbr>takenalready</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/nodejs-icon"></td>
+                        <td>Node.js</td>
+                        <td><a href="https://github.com/sahithyandev/sv443-joke-api-js-wrapper#readme" target="_blank">JokeAPI - JavaScript wrapper</a></td>
+                        <td><a href="https://github.com/sahithyandev" target="_blank">sahithyandev</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/dart-icon"></td>
+                        <td>Dart</td>
+                        <td><a href="https://github.com/MichaelDark/jokeapi#readme" target="_blank">JokeAPI - Dart</a></td>
+                        <td><a href="https://github.com/MichaelDark" target="_blank">MichaelDark</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/golang-icon"></td>
+                        <td>Go</td>
+                        <td><a href="https://github.com/Icelain/jokeapi#readme" target="_blank">jokeapi-go</a></td>
+                        <td><a href="https://github.com/Icelain" target="_blank">Icelain</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/java-icon"></td>
+                        <td>Java</td>
+                        <td><a href="https://github.com/the-codeboy/Jokes4J#readme" target="_blank">Jokes4J</a></td>
+                        <td><a href="https://github.com/the-codeboy" target="_blank">the-codeboy</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/java-icon"></td>
+                        <td>Java</td>
+                        <td><a href="https://github.com/EasyG0ing1/JokeAPI" target="_blank">JokeAPI - Java (com.simtechdata.jokeapi)</a></td>
+                        <td><a href="https://github.com/EasyG0ing1" target="_blank">EasyG0ing1</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/java-icon"></td>
+                        <td>Java</td>
+                        <td><a href="https://github.com/khurozov/jokeapi-java#readme" target="_blank">JokeAPI - Java (uz.khurozov.jokeapi-java)</a></td>
+                        <td><a href="https://github.com/khurozov" target="_blank">khurozov</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon">
+                            <img class="miniimg" src="static/kotlin-icon">
+                            <img class="miniimg" src="static/java-icon">
+                            <img class="miniimg" src="static/android-icon">
+                        </td>
+                        <td>Kotlin, Java, Android</td>
+                        <td><a href="https://github.com/ethauvin/jokeapi#readme" target="_blank">JokeAPI for Kotlin, Java and Android</a></td>
+                        <td><a href="https://github.com/ethauvin" target="_blank">ethauvin</a></td>
+                    </tr>
+                    <tr>
+                        <td class="wrappericon"><img class="miniimg" src="static/csharp-icon"></td>
+                        <td>C#</td>
+                        <td><a href="https://github.com/DanBuxton/JokeAPI-CS-Wrapper#readme" target="_blank">JokeAPI - C#</a></td>
+                        <td><a href="https://github.com/DanBuxton" target="_blank">DanBuxton</a></td>
+                    </tr>
+                </table>
+
+                <br>
+                <a class="smallText" href="#other-attributions">(Click here to see the icon attributions)</a>
+
+
+
+                <!-- #MARKER Submit Joke -->
+                <div class="docHeader targetBlink" id="submit"><a class="headerAnchorContainer" href="#submit" title="Click to link to this header">🔗</a><span class="docHeaderText">Submit a Joke:</span></div>
+
+		<div>
+			Joke submissions have been disabled for the forseeable future.<br>
+			Please see <a href="https://github.com/Sv443/JokeAPI/issues/489" target="_blank">this GitHub issue</a> for more info.
+		</div>
+
+                <div id="submissions-hidden"><div id="submitContainer" class="highlightedContainer">
+                    <div class="submitFlexItem">
+                        <table id="propertyTable">
+                            <tr>
+                                <td>Category:</td>
+                                <td>
+                                    <select id="f_category">
+                                        <option value="Misc" class="selectOnInit" selected>Misc</option>
+                                        <option value="Programming">Programming</option>
+                                        <option value="Dark">Dark</option>
+                                        <option value="Pun">Pun</option>
+                                        <option value="Spooky">Spooky</option>
+                                        <option value="Christmas">Christmas</option>
+                                    </select>
+                                </td>
+                            </tr>
+                            <tr>
+                                <td>Type:</td>
+                                <td>
+                                    <select id="f_type">
+                                        <option value="single" class="selectOnInit" selected>Single</option>
+                                        <option value="twopart">Two Part</option>
+                                    </select>
+                                </td>
+                            </tr>
+                            <tr>
+                                <td>Language:</td>
+                                <td>
+                                    <select id="f_language" class="appendLangOpts"></select>
+                                    <span id="f_langHideContainer" class="hidden">
+                                        <input type="text" id="f_customLang" placeholder="Language Code" minlength="2" maxlength="2" onchange="valChanged(this)" value="en">
+                                        <a href="<!--%#INSERT:DOCSURL#%-->/languages?format=txt" target="_blank" title="Click to view all possible ISO 639-1 language code values">(?)</a>
+                                    </span>
+                                </td>
+                            </tr>
+                            <tr>
+                                <td>Flags:</td>
+                                <td>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_nsfw" value="nsfw"><label for="f_flags_nsfw">NSFW</label>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_religious" value="religious"><label for="f_flags_religious">Religious</label>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_political" value="political"><label for="f_flags_political">Political</label>
+                                    <wbr>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_racist" value="racist"><label for="f_flags_racist">Racist</label>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_sexist" value="sexist"><label for="f_flags_sexist">Sexist</label>
+                                    <input type="checkbox" class="uncheckOnInit" id="f_flags_explicit" value="explicit"><label for="f_flags_explicit">Explicit</label>
+                                </td>
+                            </tr>
+                            <tr>
+                                <td>Joke:</td>
+                                <td>
+                                    <textarea class="clearOnInit" id="f_setup" placeholder="Joke"></textarea><br>
+                                    <textarea class="clearOnInit" id="f_delivery" placeholder="Delivery" style="display: none;"></textarea>
+                                </td>
+                            </tr>
+                        </table>
+        
+                        <br>
+        
+                        <div class="centered">
+                            <button id="submitBtn">Submit &gt;</button>
+                        </div>
+                    </div>
+                    <div class="submitFlexItem l">
+<code id="submissionCodeElement" class="prettyprint"><span class="nocode codeheader">Payload Preview:</span>
+<div class="actualCode" id="submissionDisplay"></div>
+</code>
+                    </div>
+                </div>
+
+                Note: Only Unicode characters in the range U+0000 to U+0FFF are allowed <a href="https://unicode-table.com/en" target="_blank">(more info)</a>
+                </div>
+
+
+                <!-- #MARKER Rate Limiting section-->
+                <div class="docHeader targetBlink" id="rate-limiting"><a class="headerAnchorContainer" href="#rate-limiting" title="Click to link to this header">🔗</a><span class="docHeaderText">Rate Limiting:</span></div>
+                Due to <!--%#INSERT:NAME#%--> being target of DoS attacks in the past, every client has a request budget of <!--%#INSERT:RATELIMITCOUNT#%--> requests they can send per minute.<br>
+                Sending any requests after this limit will make the API temporarily give you a HTTP 429 error until the time runs out.<br>
+                When submitting jokes, your budget is only <!--%#INSERT:SUBMISSIONRATELIMIT#%--> requests per minute.<br>
+                <br>
+                Almost everywhere when using <!--%#INSERT:NAME#%-->, you will be getting the following headers which tell you all you need to know about your current request budget:
+                <br>
+                <table id="rateLimitingTable" class="alternatingBgTable">
+                    <tr>
+                        <th>Name</th>
+                        <th>Description</th>
+                    </tr>
+                    <tr>
+                        <td>Retry-After</td>
+                        <td>Tells you in how many seconds your request budget gets reset back to <!--%#INSERT:RATELIMITCOUNT#%--></td>
+                    </tr>
+                    <tr>
+                        <td>RateLimit-Limit</td>
+                        <td>How many requests in total you can send to <!--%#INSERT:NAME#%--> per minute</td>
+                    </tr>
+                    <tr>
+                        <td>RateLimit-Remaining</td>
+                        <td>How many requests you have left in your budget</td>
+                    </tr>
+                    <tr>
+                        <td>RateLimit-Reset</td>
+                        <td>Contains an <a href="https://tools.ietf.org/html/rfc7231#section-7.1.1.1" target="_blank">IMF-fixdate</a> timestamp of when your request budget gets reset back to <!--%#INSERT:RATELIMITCOUNT#%--></td>
+                    </tr>
+                </table>
+
+
+
+                <!-- #MARKER URL parameters -->
+                <div class="docHeader targetBlink" id="url-parameters"><a class="headerAnchorContainer" href="#url-parameters" title="Click to link to this header">🔗</a><span class="docHeaderText">URL Parameters:</span></div>
+                URL parameters are like the settings or the configuration of an HTTP request.<br>
+                The parameters need to be prefixed by a single question mark (?) and separate key/value pairs need to be delimited from another by an ampersand (&amp;). Keys are separated from values with an equals sign (=).<br>
+                Example: <mark>https://example.org/some/page<span class="colB">?</span><span class="colY">key1=foo</span><span class="colB">&</span><span class="colY">key2=bar</span><span class="colB">&</span><span class="colY">key3=baz</span> ...</mark> (yellow = key/value pair)<br>
+                
+                <!--#SECTION Format-->
+                <div class="subHeaderContainer"><a href="#format-param" title="Click to link to this header">§</a> <span class="subHeader" id="format-param">Format:</span></div>
+                <mark>?format=format</mark><br><br>
+                This parameter is global - it can be used on every single endpoint of <!--%#INSERT:NAME#%--> (excluding the documentation and static content).<br>
+                It is used to change the response format from the default (JSON) to <span class="insFormatsS">(Loading...)</span>, depending on what you need.<br>
+                If the format parameter is invalid or not specified at all, the default value of JSON will be used.<br>
+                Available formats are <mark class="insFormats">(Loading...)</mark><br><br>
+                <details><summary>Example 1 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?format=xml</mark><br><br>
+                    <!-- Code blocks need to be at indentation level 0: -->
+<code class="prettyprint"><div class="actualCode">
+&lt;data&gt;
+	&lt;category&gt;Programming&lt;/category&gt;
+	&lt;type&gt;single&lt;/type&gt;
+	&lt;joke&gt;// This line doesn't actually do anything, but the code stops working when I delete it.&lt;/joke&gt;
+	&lt;flags&gt;
+		&lt;nsfw&gt;false&lt;/nsfw&gt;
+		&lt;religious&gt;false&lt;/religious&gt;
+		&lt;political&gt;false&lt;/political&gt;
+		&lt;racist&gt;false&lt;/racist&gt;
+		&lt;sexist&gt;false&lt;/sexist&gt;
+		&lt;explicit&gt;false&lt;/explicit&gt;
+	&lt;/flags&gt;
+    &lt;id&gt;12&lt;/id&gt;
+    &lt;lang&gt;en&lt;/lang&gt;
+&lt;/data&gt;
+</div></code>
+                </details><br>
+
+                <details><summary>Example 2 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?format=txt</mark><br><br>
+<code><div class="actualCode">
+How many programmers does it take to screw in a light bulb?
+
+None. It's a hardware problem.
+</div></code>
+                </details>
+
+
+                <br><br><!--#SECTION Blacklist Flags-->
+                <div class="subHeaderContainer"><a href="#flags-param" title="Click to link to this header">§</a> <span class="subHeader" id="flags-param">Blacklist Flags:</span></div>
+                <mark>?blacklistFlags=flag1[,flag2,...]</mark><br><br>
+                This parameter can only be used on the "joke" endpoint.<br>
+                If it is used, jokes that match the specified flag(s) will not be served (<a href="#blacklist-flags">what is a flag?</a>).<br>
+                If you want to use multiple flags, separate them with a comma (,) or a plus sign (+).<br>
+                These are all the available flags: <mark class="insFlags">(Loading...)</mark><br><br>
+
+                <details><summary>Example 1 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?blacklistFlags=nsfw</mark><br>
+                    Using the above URL makes sure you don't get any jokes that are not safe for work.
+                </details><br>
+
+                <details><summary>Example 2 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?blacklistFlags=nsfw,racist,sexist,explicit</mark><br>
+                    Using the above URL makes sure you don't get any jokes that are not safe for work, racist and sexist and that doesn't contain explicit language (swear words, etc.).
+                </details>
+
+
+                <br><br><!--#SECTION Language-->
+                <div class="subHeaderContainer"><a href="#lang" title="Click to link to this header">§</a> <span class="subHeader" id="lang">Language:</span></div>
+                <mark>?lang=langcode</mark><br><br>
+                This parameter can be used on every endpoint, but it is possible that no translations or jokes exist yet for your language.<br>
+                <br>
+                There are two types of languages; system languages and joke languages. Both are separate from each other.<br>
+                All system messages like errors can have a certain system language, while jokes can only have a joke language.<br>
+                It is possible, that system languages don't yet exist for your language while jokes already do.<br>
+                If no suitable system language is found, <!--%#INSERT:NAME#%--> will default to English.<br>
+                The language codes in <!--%#INSERT:NAME#%--> follow the <a href="https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes" target="_blank">ISO 639-1</a> / Alpha-2 specification.<br>
+                <br>
+                If you are interested in providing translations, please read the <a href="https://github.com/Sv443/JokeAPI/blob/master/.github/Contributing.md#submitting-translations" target="_blank">Contributing Guide</a>; it will explain it.<br>
+                You can also submit jokes in different languages. Please understand that I will not be able to ensure their proper formatting or legitimacy.<br>
+                <br>
+                Currently available system languages: <mark class="insSysLangs">(Loading...)</mark><br>
+                Currently available joke languages: <mark class="insJokeLangs">(Loading...)</mark><br><br>
+
+                <details><summary>Example 1 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?lang=ru</mark><br>
+                    Using the above URL will fetch only jokes that are Russian. If an error gets returned, it will also be in Russian, provided a translation was found.
+                </details><br>
+
+
+                <br><br><!--#SECTION Joke Type-->
+                <div class="subHeaderContainer"><a href="#type-param" title="Click to link to this header">§</a> <span class="subHeader" id="type-param">Joke Type:</span></div>
+                <mark>?type=type</mark><br><br>
+                This parameter can only be used on the "joke" endpoint.<br>
+                If it is set, you will only receive jokes with the specified joke type (<a href="#joke-type">what is a joke type?</a>).<br>
+                If it is not set or set to an invalid value, you will receive jokes of both types.<br>
+                Available types are: <mark>single, twopart</mark><br><br>
+
+                <details><summary>Example (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?type=twopart</mark><br><br>
+<code class="prettyprint"><div class="actualCode">
+{
+    "category": "Programming",
+    "type": "twopart",
+    "setup": "What is a dying programmer's last program?",
+    "delivery": "Goodbye, world!",
+    "flags": {
+        "nsfw": false,
+        "religious": false,
+        "political": false,
+        "racist": false,
+        "sexist": false,
+        "explicit": false
+    },
+    "id": 58,
+    "lang": "en"
+}
+</div></code><br>
+                    (no jokes with type "single" will be served)
+                </details>
+
+
+                <br><br><!--#SECTION Contains-->
+                <div class="subHeaderContainer"><a href="#contains-param" title="Click to link to this header">§</a> <span class="subHeader" id="contains-param">Contains:</span></div>
+                <mark>?contains=string</mark><br><br>
+                This parameter can only be used on the "joke" endpoint.<br>
+                If this parameter is specified, only jokes that contain the value of this parameter will be served (case insensitive).<br>
+                <span class="colY">IMPORTANT:</span> If the value of this parameter contains special characters, it needs to be <a href="https://en.wikipedia.org/wiki/Percent-encoding" target="_blank">percent-encoded</a> according to the <a href="https://tools.ietf.org/html/rfc3986#section-2.1" target="_blank">URI syntax standard</a>, otherwise you might get an error or a wrong response.<br><br>
+
+                <details><summary>Example (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?contains=C%23</mark> (%23 = pound character / hashtag / # - so the thing searched for here is "C#")<br><br>
+<code class="prettyprint"><div class="actualCode">
+{
+    "category": "Programming",
+    "type": "twopart",
+    "setup": "Why do programmers wear glasses?",
+    "delivery": "Because they need to C#",
+    "flags": {
+        "nsfw": false,
+        "religious": false,
+        "political": false,
+        "racist": false,
+        "sexist": false,
+        "explicit": false
+    },
+    "id": 51,
+    "lang": "en"
+}
+</div></code><br>
+                </details>
+
+
+                <br><br><!--#SECTION ID Range-->
+                <div class="subHeaderContainer"><a href="#idrange-param" title="Click to link to this header">§</a> <span class="subHeader" id="idrange-param">ID Range:</span></div>
+                <mark>?idRange=number[-number]</mark><br><br>
+                This parameter can only be used on the "joke" endpoint.<br>
+                If it is used, you will only get jokes that are inside the specified range of IDs.<br>
+                The lower and upper boundary of the ID range needs to be delimited with a minus (-), comma (,) or a plus sign (+).<br>
+                If you only want to get a single joke by its ID, you can also ignore the delimiter and just use the single ID as the value for this parameter.<br>
+                As there are currently <!--%#INSERT:TOTALJOKES#%--> jokes, the ID range can be any range from <mark>0 - <!--%#INSERT:TOTALJOKESZEROINDEXED#%--></mark><br><br>
+
+                <details><summary>Example 1 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?idRange=0-55</mark><br>
+                    Using the above URL will make <!--%#INSERT:NAME#%--> only serve you jokes that are inside the ID range of 0 to 55.
+                </details><br>
+
+                <details><summary>Example 2 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?idRange=33</mark><br>
+                    Using the above URL will make <!--%#INSERT:NAME#%--> only serve you the joke with the ID 33 (provided there are no other filters that are blacklisting that joke).
+                </details>
+
+
+
+                <br><br><!--#SECTION Joke Amount-->
+                <div class="subHeaderContainer"><a href="#amount-param" title="Click to link to this header">§</a> <span class="subHeader" id="amount-param">Joke Amount:</span></div>
+                <mark>?amount=number</mark><br><br>
+                This parameter can only be used on the "joke" endpoint.<br>
+                If it is used, you will get a certain number of jokes in a single request.<br>
+                If <!--%#INSERT:NAME#%--> has less jokes than demanded, it will not error out but instead send the maximum amount of jokes it can serve.<br>
+                Maximum possible number is <!--%#INSERT:MAXJOKEAMOUNT#%--> and using an invalid value will make <!--%#INSERT:NAME#%--> default to 1.<br>
+                The joke format will change from the default when you set the amount parameter to 2 or more.<br>
+                <br>
+
+                <details><summary>Click to view the format when fetching multiple jokes</summary>
+<code class="prettyprint lang-json">
+<div class="actualCode">{
+    "error": false,
+    "jokes": [
+        {
+            "category": "Misc",
+            "type": "single",
+            ...
+        },
+        ...
+    ],
+    "amount": 10
+}
+</div></code><br>
+                </details>
+
+                <br>
+
+                <details><summary>Example (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?amount=5</mark><br>
+                    Using the above URL will make <!--%#INSERT:NAME#%--> serve you five jokes at once.
+                </details><br>
+
+
+
+                <br><br><!--#SECTION Dry Run-->
+                <div class="subHeaderContainer"><a href="#dryrun-param" title="Click to link to this header">§</a> <span class="subHeader" id="dryrun-param">Submission Dry Run:</span></div>
+                <mark>?dry-run</mark><br><br>
+                This parameter can only be used on the "submit" endpoint. It doesn't need a value (part to the right of the equals character).<br>
+                <br>
+                If present, <!--%#INSERT:NAME#%--> will respond just like the usual "submit" endpoint except the joke isn't actually saved to the API.<br>
+                This functionality can be used to check if a joke is valid / what is wrong with it.<br>
+                <br>
+                A successful dry run will have its <mark>error</mark> property set to false.<br>
+                It will also have a <mark>message</mark> property with a short status message and a <mark>timestamp</mark> property with a 13-character UNIX timestamp.<br>
+
+                
+
+
+
+                <!-- #MARKER Safe Mode -->
+                <div class="docHeader targetBlink" id="safe-mode"><a class="headerAnchorContainer" href="#safe-mode" title="Click to link to this header">🔗</a><span class="docHeaderText">Safe Mode:</span></div>
+                Safe Mode was added in version 2.3.0. If enabled, <!--%#INSERT:NAME#%--> will try its best to serve only jokes that are considered safe for everyone.<br>
+                Unsafe jokes are those who can be considered explicit in any way, either through the used language, its references or its <a href="#blacklist-flags">blacklist flags.</a><br>
+                Jokes from the category <mark>Dark</mark> are also generally marked as unsafe.<br>
+                <br>
+                Note: this filter is really fine but that doesn't mean human error is out of the question.<br>
+                If you find any jokes that you consider unsafe but are served even when safe mode is activated, please <a href="https://github.com/Sv443/JokeAPI/issues/new/choose" target="_blank">submit an issue here.</a><br>
+                <br>
+                The Safe Mode can be enabled by adding the value-less URL parameter <mark>safe-mode</mark> to the URL.<br>
+                <br>
+                <details><summary>Example 1 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Any?safe-mode</mark><br>
+                    Using the above URL will make sure you won't get any unsafe jokes.
+                </details>
+                <br>
+                <details><summary>Example 2 (click to view)</summary>
+                    URL: <mark><!--%#INSERT:DOCSURL#%-->/joke/Misc,Programming?format=xml&safe-mode&type=single</mark><br>
+                    This URL will yield a safe joke from the Misc or Programming category in the file format XML that's also of type single.
+                </details><br>
+
+
+
+
+                <!-- #MARKER Errors -->
+                <div class="docHeader targetBlink" id="errors"><a class="headerAnchorContainer" href="#errors" title="Click to link to this header">🔗</a><span class="docHeaderText">Errors:</span></div>
+                In <!--%#INSERT:NAME#%--> all response data, no matter from which endpoint will have a boolean "error" parameter.<br>
+                Usually, it is set to false, which means there was no error in your request or on the server.<br>
+                If the "error" parameter is set to true though, there was an error and <!--%#INSERT:NAME#%--> will add an "internalError", "message", "causedBy", "additionalInfo" and "timestamp" parameter.<br>
+                <br>
+                The "internalError" parameter will be set to false, if the error was due to a malformed or failed request and it will be true, if the error is <!--%#INSERT:NAME#%-->'s fault.<br>
+                The "message" parameter will contain a short version of the error message and the "timestamp" parameter will contain a 13-character Unix timestamp.<br>
+                The "causedBy" parameter is an array of possible causes of this error and the "addidionalInfo" parameter contains a more descriptive error message.<br>
+                <br>
+                If possible, <!--%#INSERT:NAME#%--> will try to translate the error messages but keep in mind there might not exist a translation for your language yet or it is not 100% perfect.<br>
+                You can view an example of an error, where an invalid category was used, just below this paragraph.<br>
+                Together with the <a href="#status-codes">HTTP status code</a>, which will also be set according to the type of error, these parameters can be used to improve error handling massively from what was the error system in previous versions of <!--%#INSERT:NAME#%-->.<br><br><br>
+
+<code class="prettyprint"><span class="nocode codeheader">Example Error:</span>
+    <div class="actualCode">{
+    "error": true,
+    "internalError": false,
+    "code": 106,
+    "message": "No matching joke found",
+    "causedBy": [
+        "No jokes were found that match your provided filter(s)"
+    ],
+    "additionalInfo": "The specified category is invalid - Got: \"foo\" - Possible categories are: \"Any, Misc, Programming, Dark, Pun, Spooky, Christmas\" (case insensitive)",
+    "timestamp": 1579170794412
+}
+</div></code><br>
+                
+                <div class="subHeaderContainer"><a href="#status-codes" title="Click to link to this header">§</a> <span class="subHeader" id="status-codes">Status Codes:</span></div>
+                <table id="statusCodeTable" class="alternatingBgTable">
+                    <!-- Use CSS classes g (green), r (red) and dr (dark red) on the td's to color status codes -->
+                    <tr>
+                        <th>Code</th>
+                        <th>Name</th>
+                        <th>Description</th>
+                    </tr>
+                    <tr>
+                        <td class="g">200</td>
+                        <td>Ok</td>
+                        <td>The request was accepted and processed successfully and the response text contains the requested data or a status message</td>
+                    </tr>
+                    <tr>
+                        <td class="g">201</td>
+                        <td>Created</td>
+                        <td>The joke submission is formatted correctly and was accepted and successfully saved (unless <a href="#dryrun-param">?dry-run</a> was used)</td>
+                    </tr>
+                    <tr>
+                        <td class="r">400</td>
+                        <td>Bad Request</td>
+                        <td>The request you have sent to <!--%#INSERT:NAME#%--> is formatted incorrectly and cannot be processed</td>
+                    </tr>
+                    <tr>
+                        <td class="r">403</td>
+                        <td>Forbidden</td>
+                        <td>You have been added to the blacklist due to malicious behavior and are not allowed to send requests to <!--%#INSERT:NAME#%--> anymore</td>
+                    </tr>
+                    <tr>
+                        <td class="r">404</td>
+                        <td>Not Found</td>
+                        <td>The URL you have requested couldn't be found</td>
+                    </tr>
+                    <tr>
+                        <td class="r">413</td>
+                        <td>Payload Too Large</td>
+                        <td>The payload data sent to the server exceeds the maximum size of <!--%#INSERT:MAXPAYLOADSIZE#%--> bytes</td>
+                    </tr>
+                    <tr>
+                        <td class="r">414</td>
+                        <td>URI Too Long</td>
+                        <td>The URL exceeds the maximum length of <!--%#INSERT:MAXURLLENGTH#%--> characters</td>
+                    </tr>
+                    <tr>
+                        <td class="r">429</td>
+                        <td>Too Many Requests</td>
+                        <td>You have exceeded the limit of <!--%#INSERT:RATELIMITCOUNT#%--> requests per minute and have to wait a bit until you are allowed to send requests again</td>
+                    </tr>
+                    <tr>
+                        <td class="dr">500</td>
+                        <td>Internal Server Error</td>
+                        <td>There was a general internal error within <!--%#INSERT:NAME#%-->. You can get more info from the properties in the response text</td>
+                    </tr>
+                    <tr>
+                        <td class="dr">523</td>
+                        <td>Origin Unreachable</td>
+                        <td>My server is temporarily offline due to maintenance or an outage. Please be patient and check the <a href="https://status.sv443.net/" target="_blank">status page.</a></td>
+                    </tr>
+                </table>
+                
+                <br>
+
+                <mark>
+                    <span class="col g">Success</span> &bull; <span class="col r">Client Error</span> &bull; <span class="col dr">Server Error</span>
+                </mark>
+
+
+                <!-- #MARKER Troubleshooting -->
+                <div class="docHeader targetBlink" id="troubleshooting"><a class="headerAnchorContainer" href="#troubleshooting" title="Click to link to this header">🔗</a><span class="docHeaderText">Troubleshooting:</span></div>
+                <span class="subHeader">HTTP Error 403:</span>
+                First, please check if you can access all endpoints with your web browser. If not, your IP address has most likely been acting malicious and was permanently blacklisted.<br>
+                If you keep getting HTTP 403 (Forbidden) response codes in your program but not in your browser and have never shown malicious behavior before, the issue is most likely that Cloudflare is blocking your request.<br>
+                This might be due to inconsistent request data or headers that are often used in malicious requests or cyber attacks (most of the time it's gonna be the fault of the library you are using).<br>
+                A fix to this issue would be to explicitly set a <mark>User-Agent</mark> header in each of your requests.<br>
+                This is the user agent of your browser which you should be able to use in the fix described above:<br>
+                <mark id="insUserAgent">(Something didn't work, please enter "navigator.userAgent" without quotes in your JavaScript console instead)</mark><br><br>
+
+                <span class="subHeader">HTTP Error 429:</span>
+                You will get 429 errors occasionally if you exceed the maximum of <!--%#INSERT:RATELIMITCOUNT#%--> requests per minute.<br>
+                In this case, you will need to wait up to one minute to be able to talk to the API again.<br>
+                If you want to index the API or fetch all jokes, are better off going to the <a href="<!--%#INSERT:PROJGITHUBURL#%-->" target="_blank">GitHub repository</a>, where you will find everything you need and you don't need to spam <!--%#INSERT:NAME#%-->.
+
+
+                <!-- #MARKER Endpoints -->
+                <div class="docHeader targetBlink" id="endpoints"><a class="headerAnchorContainer" href="#endpoints" title="Click to link to this header">🔗</a><span class="docHeaderText">Endpoints:</span></div>
+                An endpoint is an access point to send the HTTP requests to and get your response.<br>
+                <!--%#INSERT:NAME#%--> offers these following endpoints:<br><br>
+
+
+                <!--#SECTION Get Joke-->
+                <div class="subHeaderContainer"><a href="#joke-endpoint" title="Click to link to this header">§</a> <span id="joke-endpoint" class="subHeader">Get Joke:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/joke/<span class="placeholder">[Category/-ies]</span></span>
+                </span>
+                <br><br>
+                This endpoint is the one you want to call to get a joke.<br>
+                A valid <a href="#categories">joke category</a> or multiple joke categories, delimited with a comma, plus or minus character (, + -) must be provided in the URL.<br>
+                These are all the available categories: <mark class="insCategories">(Loading...)</mark><br>
+                To see how this URL is built with certain parameters set, <a href="#try-it">visit the "Try It" section.</a><br>
+                <details>
+                    <summary>Full example - click to view</summary>
+                    <span class="requestURLwrapper">
+                        <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/joke/Programming,Misc?format=xml&blacklistFlags=nsfw,sexist&type=single&lang=ru&amount=2</span>
+                    </span><br><br>
+                    Will give two Russian joke from either the Programming or the Misc category, with the payload <a href="#response-formats">format</a> XML,<br>
+                    with the <a href="#joke-type">joke type</a> "single", while also preventing all jokes that are potentially not safe for work and sexist from being served.
+                </details>
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#flags-param">?blacklistFlags</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                    <li><a class="requestURL" href="#id-range">?idRange</a></li>
+                    <li><a class="requestURL" href="#search-string">?contains</a></li>
+                    <li><a class="requestURL" href="#joke-type">?type</a></li>
+                    <li><a class="requestURL" href="#amount">?amount</a></li>
+                </ul>
+                <a href="#url-parameters">(More info on filtering parameters)</a>
+                <br><br>
+
+
+                <!--#SECTION Info-->
+                <div class="subHeaderContainer"><a href="#info-endpoint" title="Click to link to this header">§</a> <span id="info-endpoint" class="subHeader">Info:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/info</span>
+                </span>
+                <br><br>
+                This endpoint provides a lot of information about <!--%#INSERT:NAME#%--> and its jokes:<br>
+                - The version number<br>
+                - The amount of jokes<br>
+                - All the available categories, flags, types and formats<br>
+                - A 13-character UNIX timestamp<br>
+                - The URL to a joke submission form<br>
+                - A list of languages (code and name) <!--%#INSERT:NAME#%--> currently supports<br>
+                - The minimum and maximum values of an ID range per each language<br>
+                - The amount of safe jokes there are per language<br>
+                - A string with some information, like a message of the day
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Categories-->
+                <div class="subHeaderContainer"><a href="#categories-endpoint" title="Click to link to this header">§</a> <span id="categories-endpoint" class="subHeader">Categories:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/categories</span>
+                </span>
+                <br><br>
+                This endpoint returns a list / an array of all available <a href="#categories">joke categories</a>, all available <a href="#category-aliases">category aliases</a> and a 13-character UNIX timestamp.
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION LangCode-->
+                <div class="subHeaderContainer"><a href="#langcodes-endpoint" title="Click to link to this header">§</a> <span id="langcode-endpoint" class="subHeader">Language Code:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/langcode/<span class="placeholder">[Language]</span></span>
+                </span>
+                <br><br>
+                This endpoint returns the ISO 639-1 language code of a provided language. It is searched with a fuzzy search, so you just have to provide the approximate language name.<br>
+                The resulting language code is to be used in <a href="#joke-endpoint">fetching</a> and <a href="#submit-endpoint">submitting</a> jokes in different languages.<br>
+                <br>
+                <details>
+                    <summary>Example 1 - click to view</summary>
+                    <br>
+                    <span class="requestURLwrapper">
+                        <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/langcode/german</span>
+                    </span><br><br>
+<code class="prettyprint"><div class="actualCode">
+{
+    "error": false,
+    "code": "de"
+}
+</div></code>
+                </details>
+                <br>
+                <details>
+                    <summary>Example 2 - click to view</summary>
+                    <br>
+                    <span class="requestURLwrapper">
+                        <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/langcode/sw3d1sh</span>
+                    </span><br><br>
+<code class="prettyprint"><div class="actualCode">
+{
+    "error": false,
+    "code": "sv"
+}
+</div></code>
+                </details>
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Languages-->
+                <div class="subHeaderContainer"><a href="#languages-endpoint" title="Click to link to this header">§</a> <span id="languages-endpoint" class="subHeader">Supported Languages:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/languages</span>
+                </span>
+                <br><br>
+                This endpoint returns lists of supported languages in jokes and supported languages in system messages (error messages).<br>
+                Also, it returns a list of possible ISO 639-1 language codes you can use to submit a joke or add a translation.<br>
+                <details>
+                    <summary>Example - click to view</summary>
+                    <br>
+                    <span class="requestURLwrapper">
+                        <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/languages</span>
+                    </span><br><br>
+<code class="prettyprint"><div class="actualCode">
+{
+    "defaultLanguage": "en",
+    "jokeLanguages": [
+        "de",
+        "en"
+    ],
+    "systemLanguages": [
+        "de",
+        "en",
+        "ru"
+    ],
+    "possibleLanguages": [
+        {
+            "code": "aa",
+            "name": "Afar"
+        },
+        {
+            ...
+        },
+        ...
+    ],
+    "timestamp": 1590929517702
+}
+</div></code>
+                </details>
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Flags-->
+                <div class="subHeaderContainer"><a href="#flags-endpoint" title="Click to link to this header">§</a> <span id="flags-endpoint" class="subHeader">Flags:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/flags</span>
+                </span>
+                <br><br>
+                This endpoint returns a list / an array of all available <a href="#blacklist-flags">blacklist flags</a> and a 13-character UNIX timestamp.
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Formats-->
+                <div class="subHeaderContainer"><a href="#formats-endpoint" title="Click to link to this header">§</a> <span id="formats-endpoint" class="subHeader">Formats:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/formats</span>
+                </span>
+                <br><br>
+                This endpoint returns a list / an array of all available <a href="#response-formats">response formats</a> and a 13-character UNIX timestamp.
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Ping-->
+                <div class="subHeaderContainer"><a href="#ping-endpoint" title="Click to link to this header">§</a> <span id="ping-endpoint" class="subHeader">Ping:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/ping</span>
+                </span>
+                <br><br>
+                This endpoint returns a parameter named "ping" that contains the word "Pong!" and a 13-character UNIX timestamp.<br>
+                It is intended for external uptime monitoring but you can also use it if you want to.
+                <br><br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                    <li><a class="requestURL" href="#lang">?lang</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Endpoints-->
+                <div class="subHeaderContainer"><a href="#endpoints-endpoint" title="Click to link to this header">§</a> <span id="endpoints-endpoint" class="subHeader">Endpoints:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodGET">GET</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/endpoints</span>
+                </span>
+                <br>
+                <br>
+                This endpoint returns a list / an array of all available endpoints, their usage (method, url and supported parameters) and a short description each.<br>
+                Note: the <mark>lang</mark> parameter will not work here.<br>
+                <br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#response-formats">?format</a></li>
+                </ul>
+                <br><br>
+
+
+                <!--#SECTION Submit-->
+                <div class="subHeaderContainer"><a href="#submit-endpoint" title="Click to link to this header">§</a> <span id="submit-endpoint" class="subHeader">Submit Joke:</span></div>
+                <span class="requestURLwrapper">
+                    <span class="requestMethodPUT">POST</span>
+                    <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/submit</span>
+                </span>
+                <br><br>
+                This endpoint is used to programatically submit a joke to be reviewed and added to <!--%#INSERT:NAME#%-->'s official jokes.<br>
+                If you instead just want to manually submit a joke, please <a href="#submit">click here.</a><br><br>
+                To ensure backwards compatibility you can use the <mark>PUT</mark> method even though it is discouraged.<br>
+                Using the <mark>PUT</mark> method might be deprecated in the future.<br><br>
+                The payload needs to be sent as JSON. Different <a href="#response-formats">formats</a> are not allowed here.<br>
+                The request payload has to follow the same object structure that the jokes are served in when using the "joke" endpoint.<br>
+                <br>
+                Make sure to add a "formatVersion" property, though, which just contains the current joke format version (<!--%#INSERT:FORMATVERSION#%-->).<br>
+                This is needed since there were a few changes to the object format since the previous versions and this serves as a kind of acknowledgement.<br>
+                <br>
+                The joke category has to be any valid <a href="#categories">category</a>, excluding "Any".<br>
+                If the <a href="#joke-type">joke type</a> is set to "single", the properties "setup" and "delivery" will need to be omitted and the joke is to be put in the "joke" property.<br>
+                If the <a href="#joke-type">joke type</a> is set to "twopart", the "joke" property is to be replaced with the properties "setup" and "delivery".<br>
+                The <a href="#blacklist-flags">"flags"</a> property needs to be an object that contains the boolean properties "nsfw", "religious", "political", "racist", "sexist" and "explicit". These need to be set accordingly.<br>
+                The joke ID doesn't need to be provided as it will be automatically assigned to the joke.<br>
+                <br>
+                This endpoint has a unique URL parameter; <a href="#dryrun-param">?dry-run</a>, which if present will make the API respond just like the usual "submit" endpoint except the joke isn't actually saved to the API.<br>
+                This functionality can be used to check if a joke is valid / what is wrong with it without having the joke actually submitted to the API.<br>
+                <br>
+                If the joke is valid and the server accepted the submission, you will get a response with the <a href="https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/201" target="_blank">status code 201.</a><br>
+
+                <br>
+<code class="prettyprint"><span class="nocode codeheader">Example Payload:</span>
+<div class="actualCode">{
+    "formatVersion": <!--%#INSERT:FORMATVERSION#%-->,
+    "category": "Misc",
+    "type": "single",
+    "joke": "A horse walks into a bar...",
+    "flags": {
+        "nsfw": true,
+        "religious": false,
+        "political": true,
+        "racist": false,
+        "sexist": false,
+        "explicit": false
+    },
+    "lang": "en"
+}
+</div></code>
+
+                <br>
+                Supported URL parameters:<br>
+                <ul>
+                    <li><a class="requestURL" href="#dryrun-param">?dry-run</a></li>
+                </ul>
+                <br><br>
+
+                <details><summary>Click here to see how the server responds to a joke submission</summary>
+<code class="prettyprint"><span class="nocode codeheader">Server Response:</span>
+<div class="actualCode">{
+    "error": false,
+    "message": "Joke submission was successfully saved. It will soon be checked out by the author.",
+    "submission": {
+        "formatVersion": 2,
+        "category": "Misc",
+        "type": "single",
+        "joke": "A horse walks into a bar...",
+        "flags": {
+            "nsfw": true,
+            "religious": false,
+            "political": true,
+            "racist": false,
+            "sexist": false,
+            "explicit": false
+        },
+        "lang": "en"
+    },
+    "timestamp": 1579685839560
+}
+</div></code>
+                </details>
+                <br><br>
+                
+                <details><summary>Click here to see how the server responds to a submission dry run</summary>
+<code class="prettyprint"><span class="nocode codeheader">Server Response:</span>
+<div class="actualCode">{
+    "error": false,
+    "message": "Dry Run complete! No errors were found.",
+    "timestamp": 1604627398681
+}
+</div></code>
+                                    </details>
+                                    <br><br><br>
+
+
+                <!-- #MARKER Filtering -->
+                <div class="docHeader targetBlink" id="filtering"><a class="headerAnchorContainer" href="#filtering" title="Click to link to this header">🔗</a><span class="docHeaderText">Filtering Jokes:</span></div>
+                I created <!--%#INSERT:NAME#%--> with the main purpose of it having many versatile filtering options as other APIs didn't offer this amount of customization.<br>
+                In this section you will learn how the filtering works.<br><br>
+
+                There are six different filtering methods in <!--%#INSERT:NAME#%-->:<br>
+                <ul>
+                    <li><a href="#categories">Joke Category</a></li>
+                    <li><a href="#blacklist-flags">Blacklist Flags</a></li>
+                    <li><a href="#response-formats">Response Format</a></li>
+                    <li><a href="#joke-type">Joke Type</a></li>
+                    <li><a href="#search-string">Search String</a></li>
+                    <li><a href="#id-range">ID Range</a></li>
+                    <li><a href="#lang">Language</a></li>
+                </ul>
+
+
+
+                <!-- #MARKER API Tokens -->
+                <div class="docHeader targetBlink" id="api-tokens"><a class="headerAnchorContainer" href="#api-tokens" title="Click to link to this header">🔗</a><span class="docHeaderText">API Tokens:</span></div>
+                <!--%#INSERT:NAME#%--> has a way of whitelisting certain clients. This is achieved through an API token.<br>
+                At the moment, you will only receive one of these tokens temporarily if something breaks or if you are a business and need more than <!--%#INSERT:RATELIMITCOUNT#%--> requests per minute.<br>
+                If you do end up receiving a token, using it is as easy as adding an <mark>Authorization</mark> header with the actual token as its value.<br>
+                You will receive a response header called <mark>Token-Valid</mark> which will contain the number 1 if the token is valid.<br>
+                If the token is invalid, the Token-Valid header will be set to 0 and <!--%#INSERT:NAME#%--> will treat your request as if you didn't supply a token in the first place.
+
+
+
+                <!-- #MARKER Examples -->
+                <div class="docHeader targetBlink" id="examples"><a class="headerAnchorContainer" href="#examples" title="Click to link to this header">🔗</a><span class="docHeaderText">Examples:</span></div>
+                These are some examples in some commonly used languages to show you how you could implement <!--%#INSERT:NAME#%-->:<br><br>
+
+                <!--#SECTION Node-->
+                <details><summary id="example-node" class="big">Node.js (click to show)</summary>
+<code class="prettyprint"><span class="nocode codeheader">Node.js:</span>
+<div class="actualCode">const https = require("https"); // Native module, no need to explicitly install
+
+const baseURL = "<!--%#INSERT:DOCSURL#%-->";
+const categories = ["Programming", "Misc", "Pun", "Spooky", "Christmas"];
+const params = [
+    "blacklistFlags=nsfw,religious,racist",
+    "idRange=0-100"
+];
+
+https.get(`${baseURL}/joke/${categories.join(",")}?${params.join("&")}`, res => {
+    console.log("\n");
+    res.on("data", chunk => {
+        // On data received, convert it to a JSON object
+        let randomJoke = JSON.parse(chunk.toString());
+
+        if(randomJoke.type == "single")
+        {
+            // If type == "single", the joke only has the "joke" property
+            console.log(randomJoke.joke);
+            console.log("\n");
+        }
+        else
+        {
+            // If type == "twopart", the joke has the "setup" and "delivery" properties
+            console.log(randomJoke.setup);
+            setTimeout(() => {
+                console.log(randomJoke.delivery);
+                console.log("\n");
+            }, 3000);
+        }
+    });
+
+    res.on("error", err => {
+        // On error, log to console
+        console.error(`Error: ${err}`);
+    });
+});
+</div></code>
+                </details><br>
+
+                <!--#SECTION Web-JS-->
+                <details><summary id="example-js" class="big">Web-JavaScript (click to show)</summary>
+<code class="prettyprint"><span class="nocode codeheader">Web-JS:</span>
+<div class="actualCode">var baseURL = "<!--%#INSERT:DOCSURL#%-->";
+var categories = ["Programming", "Misc", "Pun", "Spooky", "Christmas"];
+var params = [
+    "blacklistFlags=nsfw,religious,racist",
+    "idRange=0-100"
+];
+
+var xhr = new XMLHttpRequest();
+xhr.open("GET", baseURL + "/joke/" + categories.join(",") + "?" + params.join("&"));
+
+xhr.onreadystatechange = function() {
+    if(xhr.readyState == 4 && xhr.status < 300) // readyState 4 means request has finished + we only want to parse the joke if the request was successful (status code lower than 300)
+    {
+        var randomJoke = JSON.parse(xhr.responseText);
+
+        if(randomJoke.type == "single")
+        {
+            // If type == "single", the joke only has the "joke" property
+            alert(randomJoke.joke);
+        }
+        else
+        {
+            // If type == "single", the joke only has the "joke" property
+            alert(randomJoke.setup);
+            alert(randomJoke.delivery);
+        }
+    }
+    else if(xhr.readyState == 4)
+    {
+        alert("Error while requesting joke.\n\nStatus code: " + xhr.status + "\nServer response: " + xhr.responseText);
+    }
+};
+
+xhr.send();
+</div></code>
+                </details><br>
+
+                <!--#SECTION C#-->
+
+                <details><summary id="example-cs" class="big">C# (click to show)</summary>
+                <code class="prettyprint"><span class="nocode codeheader">C#:</span>
+                <div class="actualCode">using System;
+using System.Threading.Tasks;
+using System.Net.Http;
+
+namespace JokeAPIExample
+{
+    class Program
+    {
+        static async Task Main(string[] args)
+        {
+            string joke = await GetJoke();
+            Console.WriteLine("Joke = " + joke);
+        }
+
+        public async static Task<string> GetJoke()
+        {
+            var client = new HttpClient();
+            var response = await client.GetAsync("https://jokeapi.dev/joke/Any?format=txt&type=single&blacklistFlags=nsfw,racist,sexist&lang=en");
+            var joke = await response.Content.ReadAsStringAsync();
+            return joke;
+        }
+    }
+}
+</div></code>
+                </details><br>
+
+
+                <details><summary id="example-cs" class="big">C# - Advanced (click to show)</summary>
+<code class="prettyprint"><span class="nocode codeheader">C#:</span>
+<div class="actualCode">using System;
+using System.Threading.Tasks;
+using System.Net.Http;
+using Newtonsoft.Json;   // Install with NuGet Package Manager: https://www.nuget.org/packages/Newtonsoft.Json/12.0.3?_src=template
+
+namespace TestHttpRequest
+{
+    class Program
+    {
+        static async Task Main(string[] args)
+        {
+            Joke randomJoke = await GetJokeAdvanced();
+            if (randomJoke.type == "single")
+            {
+                Console.WriteLine(randomJoke.joke);
+            }
+            else
+            {
+                Console.WriteLine(randomJoke.setup);
+                System.Threading.Thread.Sleep(3000);
+                Console.WriteLine(randomJoke.delivery);
+            }
+        }
+
+        public async static Task<Joke> GetJokeAdvanced()
+        {
+            const string baseUrl = "https://jokeapi.dev";
+            string[] categories = { "Programming", "Miscellaneous", "Pun", "Spooky", "Christmas" };
+            string[] parameters = {
+                "blacklistFlags=nsfw,religious,racist,sexist",
+                "idRange=0-100"
+            };
+            string requestUrl = $"{baseUrl}/joke/{string.Join(",", categories)}?{string.Join("&", parameters)}";
+            Joke randomJoke;
+
+            using (var httpClient = new HttpClient())
+            {
+                var json = await httpClient.GetStringAsync(requestUrl);
+                randomJoke = JsonConvert.DeserializeObject<Joke>(json);
+            }
+
+            return randomJoke;
+        }
+    }
+
+    public class Joke
+    {
+        public string type { get; set; }
+        public string joke { get; set; }
+        public string setup { get; set; }
+        public string delivery { get; set; }
+        public int id { get; set; }
+        public Flags flags { get; set; }
+    }
+
+
+    public class Flags
+    {
+        public bool nsfw { get; set; }
+        public bool religious { get; set; }
+        public bool political { get; set; }
+        public bool racist { get; set; }
+        public bool sexist { get; set; }
+    }
+}</div></code>
+                </details><br>
+
+
+                <!-- #MARKER Terminology -->
+                <div class="docHeader targetBlink" id="terminology"><a class="headerAnchorContainer" href="#terminology" title="Click to link to this header">🔗</a><span class="docHeaderText">Terminology:</span></div>
+                You might come across some terms in this documentation you don't understand. This section teaches you what those terms mean.<br>
+
+                <!--#SECTION Categories-->
+                <div class="subHeaderContainer"><a href="#categories" title="Click to link to this header">§</a> <span class="subHeader" id="categories">Categories:</span></div>
+                <!--%#INSERT:NAME#%--> has a first, coarse filter that just categorizes the jokes depending on what the joke is about or who the joke is directed at.<br>
+                A joke about programming will be in the "Programming" category, dark humor will be in the "Dark" category and so on.<br>
+                If you want jokes from all categories, you can instead use "Any", which will make <!--%#INSERT:NAME#%--> randomly choose a category.<br>
+                Since version 2.3.0, there are now aliases to these categories. <a href="#category-aliases">This section</a> explains how they work.<br><br>
+                These are all the available categories: <mark class="insCategories">(Loading...)</mark><br>
+                To set the category you want, you need to add it to the URL path, just after the "/joke" endpoint, like this:
+                <mark><!--%#INSERT:DOCSURL#%-->/joke/Programming</mark>
+                <br>
+
+                <div class="subHeaderContainer"><a href="#category-aliases" title="Click to link to this header">§</a> <span class="subHeader" id="category-aliases">Category Aliases:</span></div>
+                Since version 2.3.0, there are now category aliases. You can use these as a shorter or just different version of the normal set of <a href="#categories">categories.</a><br>
+                Internally, these aliases will be resolved to one of the normal set of <a href="#categories">categories</a> and your request (including the API's response) will be treated as if you used the resolved value.<br>
+                <br>
+                The following is a list of aliases and what they will be resolved to internally:<br>
+                <table id="catAliasesContainer" class="alternatingBgTable">
+                    <tr>
+                        <th>Alias</th>
+                        <th>Resolved Category</th>
+                    </tr>
+                </table>
+                <br>
+
+                <details>
+                    <summary>Example - click to view</summary>
+                    The following URL will make <!--%#INSERT:NAME#%--> resolve the alias <mark>Coding</mark>, giving you a joke from the category <mark>Programming</mark>:
+                    <br>
+                    <span class="requestURLwrapper">
+                        <span class="requestURL"><!--%#INSERT:DOCSURL#%-->/joke/Coding</span>
+                    </span>
+                </details><br>
+
+                <!--#SECTION Flags-->
+                <div class="subHeaderContainer"><a href="#blacklist-flags" title="Click to link to this header">§</a> <span class="subHeader" id="blacklist-flags">(Blacklist-) Flags:</span></div>
+                Blacklist Flags (or just "Flags") are a more fine layer of filtering.<br>
+                Multiple flags can be set on each joke and they tell you something about the offensiveness of each joke.<br>
+                You can tell <!--%#INSERT:NAME#%--> to not give you jokes that have the specified flags by using the <a href="#flags-param">?blacklistFlags</a> URL parameter.<br><br>
+                These are all the available flags: <mark class="insFlags">(Loading...)</mark><br>
+
+                <!--#SECTION Formats-->
+                <div class="subHeaderContainer"><a href="#response-formats" title="Click to link to this header">§</a> <span class="subHeader" id="response-formats">(Response-) Formats:</span></div>
+                Response Formats (or just "Formats") are a way to get your data in a different file format.<br>
+                Maybe your environment or language doesn't support JSON natively. In that case, <!--%#INSERT:NAME#%--> is able to convert the JSON-formatted joke to a different format for you.<br>
+                You can tell <!--%#INSERT:NAME#%--> which file format to respond with by using the <a href="#format-param">?format</a> URL parameter.<br><br>
+                These are all the available formats: <mark class="insFormats">(Loading...)</mark><br>
+
+                <!--#SECTION Type-->
+                <div class="subHeaderContainer"><a href="#joke-type" title="Click to link to this header">§</a> <span class="subHeader" id="joke-type">Joke Type:</span></div>
+                Each joke comes with one of two types: single or twopart.<br>
+                If a joke is of type "twopart", it has a setup string and a delivery string, which are both part of the joke.<br>
+                They are separated because you might want to present the users the delivery after a timeout or in a different section of the UI.<br>
+                A joke of type "single" only has a single string, which is the entire joke.<br>
+                If you only want one type of joke, you can use the <a href="#type-param">?type</a> URL parameter.<br><br>
+                These are the available types: <mark>single, twopart</mark><br>
+
+                <!--#SECTION Search String-->
+                <div class="subHeaderContainer"><a href="#search-string" title="Click to link to this header">§</a> <span class="subHeader" id="search-string">Search String:</span></div>
+                If the search string filter is used, only jokes that contain the specified string will be returned.<br>
+                You can specify this search string by using the <a href="#contains-param">?contains</a> URL parameter.<br>
+                <span class="colY">IMPORTANT:</span> If the value of this parameter contains special characters, it needs to be <a href="https://tools.ietf.org/html/rfc3986#section-2.1" target="_blank">percent-encoded</a> according to the URI standard (<a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURIComponent" target="_blank">MDN Reference</a>), otherwise you might get an error or a wrong response.<br>
+
+
+                <!--#SECTION ID Range-->
+                <div class="subHeaderContainer"><a href="#id-range" title="Click to link to this header">§</a> <span class="subHeader" id="id-range">ID Range:</span></div>
+                If this filter is used, you will only get jokes that are within the provided range of IDs.<br>
+                You don't necessarily need to provide an ID <i>range</i> though, a single ID will work just fine as well.<br>
+                Example: an ID range of 0-9 will mean you will only get one of the first 10 jokes, while an ID range of 5 will mean you will only get the 6th joke.<br>
+                You can set the ID range by using the <a href="#idrange-param">?idRange</a> URL parameter.<br>
+                As there are currently <!--%#INSERT:TOTALJOKES#%--> jokes, the ID range can be anywhere in the range of <mark>0</mark> to <mark><!--%#INSERT:TOTALJOKESZEROINDEXED#%--></mark><br>
+                This range varies from language to language though, so I recommend taking a look at the <a href="#info-endpoint">/info endpoint</a>, which contains all possible ID ranges.<br>
+
+
+                <!--#SECTION Amount-->
+                <div class="subHeaderContainer"><a href="#amount" title="Click to link to this header">§</a> <span class="subHeader" id="amount">Joke Amount:</span></div>
+                This filter allows you to set a certain amount of jokes to receive in a single call to the <a href="#joke-endpoint">"Get Joke" endpoint.</a><br>
+                You can set it using the <a href="#amount-param">?amount</a> URL parameter.<br>
+                Setting the filter to an invalid number will result in the API defaulting to serving a single joke.<br>
+                Setting it to a number larger than <!--%#INSERT:MAXJOKEAMOUNT#%--> will make <!--%#INSERT:NAME#%--> default to the maximum (<!--%#INSERT:MAXJOKEAMOUNT#%-->).<br>
+                If you request more than <!--%#INSERT:JOKEENCODEAMOUNT#%--> jokes, the API will try to serve them encoded with Brotli, Gzip or Deflate, depending on whether or not you set a <a href="https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Accept-Encoding" target="_blank">Accept-Encoding</a> header.<br>
+
+
+                <!-- #MARKER Contributors -->
+                <div class="docHeader targetBlink" id="contributors"><a class="headerAnchorContainer" href="#contributors" title="Click to link to this header">🔗</a><span class="docHeaderText">Contributors:</span></div>
+                The following is a list of people that have directly contributed to <!--%#INSERT:NAME#%-->'s development.<br>
+                I am very grateful for this voluntary effort to improve <!--%#INSERT:NAME#%--> and would like to thank anyone who contributed :)<br>
+
+                <br>
+                <div id="contributorsContainer">
+                    (Loading...)
+                </div>
+                <br><br>
+                If you want to contribute to <!--%#INSERT:NAME#%-->, please <a href="<!--%#INSERT:CONTRIBUTORGUIDEURL#%-->" target="_blank">read this guide.</a>
+
+
+                <!-- #MARKER Dependents -->
+                <div class="docHeader targetBlink" id="dependents"><a class="headerAnchorContainer" href="#dependents" title="Click to link to this header">🔗</a><span class="docHeaderText">Dependents:</span></div>
+                These are some of the projects that depend on <!--%#INSERT:NAME#%-->:
+                <ul>
+                    <li><a href="https://github.com/ZephyrVentum/dozens-Advices" target="_blank">dozens Advices</a> by <a href="https://github.com/ZephyrVentum" target="_blank">ZephyrVentum</a></li>
+                    <li><a href="https://github.com/QGIsK/Shadow-bot" target="_blank">shadow-bot</a> by <a href="https://github.com/QGIsK" target="_blank">QGIsK</a></li>
+                    <li><a href="https://github.com/aksty/Jokepy" target="_blank">Jokepy</a> by <a href="https://github.com/aksty/Jokepy" target="_blank">aksty</a></li>
+                    <li><a href="https://github.com/whiteadi/Dark1" target="_blank">Dark1</a> by <a href="https://github.com/whiteadi" target="_blank">whiteadi</a></li>
+                    <li><a href="https://github.com/LiamTownsley/Prejudice-Networks" target="_blank">Prejudice-Networks</a> by <a href="https://github.com/LiamTownsley" target="_blank">LiamTownsley</a></li>
+                    <li><a href="https://irshad.ml/humour.html" target="_blank">https://irshad.ml/humour.html</a> by <a href="https://github.com/draco-malfoy" target="_blank">draco-malfoy</a></li>
+                    <li><a href="https://github.com/variousnabil/Random-Joke-Generator-with-Flutter" target="_blank">Random-Joke-Generator-with-Flutter</a> by <a href="https://github.com/variousnabil" target="_blank">variousnabil</a></li>
+                    <li><a href="https://thatcopy.pw/comma" target="_blank">comma</a> by <a href="https://github.com/ThatCopy" target="_blank">ThatCopy</a></li>
+                    <li><a href="https://github.com/AlHood77/Joke_Teller" target="_blank">Joke Teller</a> by <a href="https://github.com/AlHood77" target="_blank">AlHood77</a></li>
+                    <li><a href="https://github.com/remcoov/jokes" target="_blank">Jokes plugin for Craft CMS 3.x</a> by <a href="https://github.com/remcoov" target="_blank">remcoov</a></li>
+                    <li><a href="https://github.com/ToastIT-dev/PoshBot.Joker" target="_blank">PoshBot.Joker</a> by <a href="https://github.com/ToastIT-dev" target="_blank">ToastIT-dev</a></li>
+                    <li><a href="https://github.com/Sv443/JokeAPI_ComputerCraft" target="_blank">JokeAPI_ComputerCraft</a> by <a href="https://github.com/Sv443" target="_blank">Sv443</a></li>
+                    <li><a href="https://github.com/jonathanbossenger/devdadjokes" target="_blank">Dev Dad Jokes</a> by <a href="https://github.com/jonathanbossenger" target="_blank">Jonathan Bossenger</a></li>
+
+                    <!-- <li><a href="xyz" target="_blank">xyz</a> by <a href="https://github.com/xyz" target="_blank">xyz</a></li> -->
+                </ul>
+                <br>
+                (<a href="https://github.com/Sv443/JokeAPI/issues/new/choose" target="_blank">Submit an issue</a> to get your project added here)
+
+
+                <!-- #MARKER Legal Stuff -->
+                <div class="docHeader targetBlink" id="legal"><a class="headerAnchorContainer" href="#legal" title="Click to link to this header">🔗</a><span class="docHeaderText">Legal Stuff:</span></div>
+
+                <!-- #SECTION License -->
+                <div class="subHeaderContainer"><a href="#license" title="Click to link to this header">§</a> <span class="subHeader" id="license">License:</span></div>
+                <div class="indented">
+                    <!--%#INSERT:NAME#%--> is licensed under the MIT License.<br>
+                    Before modifying and / or redistributing the source code, please take two minutes to read the license text.<br>
+                    <a href="https://sv443.net/LICENSE" target="_blank">You can view the full license text by clicking here.</a>
+                </div>
+
+                <!-- #SECTION Privacy -->
+                <br><br>
+                <div class="subHeaderContainer"><a href="#privacy" title="Click to link to this header">§</a> <span class="subHeader" id="privacy">Privacy:</span></div>
+                <div class="indented">
+                    <!--%#INSERT:NAME#%--> has to collect some anonymized data to work.<br>
+                    To view a list with details, please <span class="mimica" onclick="privPolMoreInfo()">click here.</span><br>
+                    To view the privacy policy, please <a href="https://sv443.net/privacypolicy/en" target="_blank">click here.</a><br>
+                    To view the site notice / imprint, <a href="https://sv443.net/imprint/en" target="_blank">click here.</a>
+                </div>
+
+                <!-- #SECTION Disclaimer -->
+                <br><br>
+                <div class="subHeaderContainer"><a href="#disclaimer" title="Click to link to this header">§</a> <span class="subHeader" id="disclaimer">Disclaimer:</span></div>
+                <div class="indented">
+                    I will hereby not claim any legal responsibility or liability for <!--%#INSERT:NAME#%--> and the jokes it serves (especially those from the "Dark" category).<br>
+                    Whether it is used maliciously or breaks something in your project or someone gets offended by a joke, I can't be held accountable.<br>
+                    Additionally, I will only be able to provide security updates for a small selection of versions, a list of which you can find <a href="<!--%#INSERT:PROJGITHUBURL#%-->/security/policy#supported-versions" target="_blank">here.</a><br>
+                    I am doing my best to ensure security and stability but there's only so much a single developer can do.<br>
+                    Please report any issue that may arise to <a href="<!--%#INSERT:PROJGITHUBURL#%-->/issues/new/choose" target="_blank">the GitHub issue tracker</a> and I will try my best to fix it as soon as possible.<br>
+                    If you want to contact me, you can <a href="https://sv443.net/discord" target="_blank">join my Discord server</a> (fastest way to contact me) or send me an E-Mail: <span class="antiBotE" data-enc="Y29udGFjdEBzdjQ0My5uZXQ=">(click to show)</span>
+                </div>
+
+
+                <!-- #MARKER Attributions -->
+                <div class="docHeader targetBlink" id="attributions"><a class="headerAnchorContainer" href="#attributions" title="Click to link to this header">🔗</a><span class="docHeaderText">Attributions:</span></div>
+                These are the attributions to everything made by third parties that <!--%#INSERT:NAME#%--> is based on or needs to function.
+
+                <!-- #SECTION Fonts -->
+                <br><br>
+                <div class="subHeaderContainer"><a href="#fonts" title="Click to link to this header">§</a> <span class="subHeader" id="fonts">Fonts:</span></div>
+                <div class="indented">
+                    <!--%#INSERT:NAME#%--> uses these third party fonts on its documentation page:
+                    <ul>
+                        <li><a href="https://github.com/Microsoft/Cascadia-Code" style="font-family: 'Cascadia Code';" target="_blank">Cascadia Code</a> by <a href="https://github.com/Microsoft" target="_blank">Microsoft</a></li>
+                        <li><a href="https://github.com/Google/Roboto" style="font-family: 'Roboto Mono', monospace;" target="_blank">Roboto</a> by <a href="https://github.com/Google" target="_blank">Google</a></li>
+                    </ul>
+                </div>
+
+                <br><br>
+                <div class="subHeaderContainer"><a href="#other-attributions" title="Click to link to this header">§</a> <span class="subHeader" id="other-attributions">Other Attributions:</span></div>
+                <div class="indented">
+                    Third party icons used on this page:
+                    <ul>
+                        <li>The Rust icon is owned by <a href="https://mozilla.org/" target="_blank">Mozilla.</a></li>
+                        <li>The Python icon is owned by the <a href="https://www.python.org/psf/" target="_blank">Python Software Foundation.</a></li>
+                        <li>The Node.js icon is a trademark of the <a href="https://openjsf.org/" target="_blank">OpenJS Foundation.</a></li>
+                        <li>The Go icon is owned by <a href="https://about.google/" target="_blank">Google.</a></li>
+                        <li>The Dart icon is owned by <a href="https://about.google/" target="_blank">Google.</a></li>
+                        <li>The C# icon is owned by <a href="https://microsoft.com/" target="_blank">Microsoft.</a></li>
+                        <li>The Kotlin icon is owned by <a href="https://jetbrains.com/" target="_blank">JetBrains.</a></li>
+                        <li>The Java icon is owned by <a href="https://oracle.com/" target="_blank">Oracle.</a></li>
+                        <li>The Android icon is owned by <a href="https://about.google/" target="_blank">Google.</a></li>
+                    </ul>
+                    The Sv443 Network is not endorsed by or affiliated with any of the above mentioned entities.<br>
+                    All of these icons are re-hosted for availability and to preserve the same format and are protected from being indexed by the majority of web crawlers.<br>
+                    <br>
+                    <br>
+                    The rectangular badges, like the ones found in the <a href="#uptime">uptime</a> and <a href="#security">security</a> sections, are powered by <a href="https://shields.io/" target="_blank">Shields.io</a>
+                </div>
+
+                <!-- #SECTION Libraries -->
+                <br><br>
+                <div class="subHeaderContainer"><a href="#libraries" title="Click to link to this header">§</a> <span class="subHeader" id="libraries">Libraries:</span></div>
+                <div class="indented">
+                    <!--%#INSERT:NAME#%--> uses these libraries:
+                    <ul>
+                        <li><a href="https://github.com/google/code-prettify" target="_blank">Code-Prettify</a> by <a href="https://github.com/Google" target="_blank">Google</a></li>
+                        <li><a href="https://github.com/Sv443/JSLib" target="_blank">JSLib</a> by <a href="https://github.com/Sv443" target="_blank">Sv443</a></li>
+                    </ul>
+                    These libraries are served using <a href="https://www.jsdelivr.com/" target="_blank">jsDelivr.</a><br>
+                </div>
+
+                <!-- #SECTION Dependencies -->
+                <br><br>
+                <div class="subHeaderContainer"><a href="#dependencies" title="Click to link to this header">§</a> <span class="subHeader" id="dependencies">Dependencies:</span></div>
+                <div class="indented">
+                    These are the packages <!--%#INSERT:NAME#%--> depends on (excluding dependencies of dependencies and devDependencies):
+                    <ul>
+                        <li><a href="https://npmjs.com/package/@pm2/io" target="_blank">@pm2/io</a></li>
+                        <li><a href="https://npmjs.com/package/dotenv" target="_blank">dotenv</a></li>
+                        <li><a href="https://npmjs.com/package/farmhash" target="_blank">farmhash</a></li>
+                        <li><a href="https://npmjs.com/package/fs-extra" target="_blank">fs-extra</a></li>
+                        <li><a href="https://npmjs.com/package/fuse.js" target="_blank">fuse.js</a></li>
+                        <li><a href="https://npmjs.com/package/http-ratelimit" target="_blank">http-ratelimit</a></li>
+                        <li><a href="https://npmjs.com/package/js2xmlparser" target="_blank">js2xmlparser</a></li>
+                        <li><a href="https://npmjs.com/package/json-to-pretty-yaml" target="_blank">json-to-pretty-yaml</a></li>
+                        <li><a href="https://npmjs.com/package/mysql" target="_blank">mysql</a></li>
+                        <li><a href="https://npmjs.com/package/node-wrap" target="_blank">node-wrap</a></li>
+                        <li><a href="https://npmjs.com/package/promise-all-sequential" target="_blank">promise-all-sequential</a></li>
+                        <li><a href="https://npmjs.com/package/rate-limiter-flexible" target="_blank">rate-limiter-flexible</a></li>
+                        <li><a href="https://npmjs.com/package/request-ip" target="_blank">request-ip</a></li>
+                        <li><a href="https://npmjs.com/package/snyk" target="_blank">snyk</a></li>
+                        <li><a href="https://npmjs.com/package/svjsl" target="_blank">svjsl</a></li>
+                        <li><a href="https://npmjs.com/package/url-parse" target="_blank">url-parse</a></li>
+                        <li><a href="https://npmjs.com/package/xss" target="_blank">xss</a></li>
+                        <!--
+                            <li><a href="https://npmjs.com/package/XXXX" target="_blank">XXXX</a></li>
+                        -->
+                    </ul>
+                </div>
+                <br><br><br><br>
+    
+                <footer id="copyrightNotice">
+                    <span>&copy; Copyright 2018-2021 Sv443 - licensed under the <a href="https://sv443.net/LICENSE" target="_blank">MIT license</a></span><br>
+                    <span>This project is a part of the <a href="https://sv443.net/" target="_blank">Sv443 Network</a></span>
+                </footer>
+            </main>
+        </div>
+
+
+        <!-- #MARKER SideNav -->
+        <div id="sidenav" class="sidenav noselect">
+            <div id="sidenavHeader">
+                <div class="sidenavHeaderItem noselect">Menu</div>
+                <div class="sidenavHeaderItem spacerItem"></div>
+                <a class="sidenavHeaderItem closebtn mimica noul noselect nopad rtext" onclick="closeNav()" title="Click or press Esc to close">&times;</a>
+            </div>
+            <div id="sidenavContent">
+                <a class="noul" onclick="closeNav()" href="#info" title="Some information about <!--%#INSERT:NAME#%-->">Information</a>
+                <a class="noul" onclick="closeNav()" href="#news" title="The latest news about <!--%#INSERT:NAME#%-->">News</a>
+                <a class="noul" onclick="closeNav()" href="#try-it" title="An interactive form to try out the API">Try it out</a>
+                <a class="noul" onclick="closeNav()" href="#getting-started" title="This is where you should start if you are new to <!--%#INSERT:NAME#%-->">Getting Started</a>
+                <a class="noul" onclick="closeNav()" href="#wrappers" title="API Wrappers make using <!--%#INSERT:NAME#%--> easier">API Wrappers</a>
+                <a class="noul" onclick="closeNav()" href="#submit" title="A form to submit a joke to <!--%#INSERT:NAME#%-->">Submit a Joke</a>
+                <div class="sidenavSpacer"></div>
+                <a class="noul" onclick="closeNav()" href="#url-parameters" title="All URL Parameters you can use and how to use them">URL Parameters</a>
+                <a class="noul" onclick="closeNav()" href="#safe-mode" title="How to enable <!--%#INSERT:NAME#%-->'s Safe Mode">Safe Mode</a>
+                <details>
+                    <summary title="All endpoints of <!--%#INSERT:NAME#%-->">
+                        <span class="sidenav-mimica">Endpoints</span>
+                    </summary>
+
+                    <ul>
+                        <li><a class="noul" onclick="closeNav()" href="#joke-endpoint">Get Joke</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#info-endpoint">Info</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#categories-endpoint">Categories</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#langcodes-endpoint">Language Codes</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#languages-endpoint">Supported Languages</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#flags-endpoint">Flags</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#formats-endpoint">Formats</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#ping-endpoint">Ping</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#endpoints-endpoint">Endpoints</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#submit-endpoint">Submit</a></li>
+                    </ul>
+                </details>
+                <a class="noul" onclick="closeNav()" href="#filtering" title="How you can filter jokes">Filtering Jokes</a>
+                <a class="noul" onclick="closeNav()" href="#api-tokens" title="All about <!--%#INSERT:NAME#%-->'s API tokens">API Tokens</a>
+                <a class="noul" onclick="closeNav()" href="#examples" title="Some code examples of how to use <!--%#INSERT:NAME#%-->">Examples</a>
+                <div class="sidenavSpacer"></div>
+                <a class="noul" onclick="closeNav()" href="#errors" title="How errors are structured">Errors</a>
+                <a class="noul" onclick="closeNav()" href="#rate-limiting" title="How <!--%#INSERT:NAME#%--> protects itself against DoS attacks">Rate Limiting</a>
+                <a class="noul" onclick="closeNav()" href="#troubleshooting" title="Some fixes for common errors">Troubleshooting</a>
+                <div class="sidenavSpacer"></div>
+                <!-- <details>
+                    <summary>
+                        <span class="sidenav-mimica">Examples</span> 
+                    </summary>
+
+                    <ul>
+                        <li><a class="noul" onclick="closeNav()" href="#example-node">Node.js</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#example-js">Web-JS</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#example-cs">C#</a></li>
+                    </ul>
+                </details> -->
+                <details>
+                    <summary>
+                        <span class="sidenav-mimica" title="What certain terms from this documentation mean">Terminology</span> 
+                    </summary>
+
+                    <ul>
+                        <li><a class="noul" onclick="closeNav()" href="#categories">Categories</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#category-aliases">Category Aliases</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#blacklist-flags">(Blacklist-)&nbsp;Flags</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#response-formats">Formats</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#joke-type">Joke Type</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#search-string">Search String</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#id-range">ID Range</a></li>
+                        <li><a class="noul" onclick="closeNav()" href="#amount">Joke Amount</a></li>
+                    </ul>
+                </details>
+                <a class="noul" onclick="closeNav()" href="#contributors" title="People who contributed to <!--%#INSERT:NAME#%-->">Contributors</a>
+                <a class="noul" onclick="closeNav()" href="#dependents" title="Projects that use <!--%#INSERT:NAME#%-->">Dependents</a>
+                <a class="noul" onclick="closeNav()" href="#legal" title="All the legal stuff">Legal Stuff</a>
+                <a class="noul bottomspacer" onclick="closeNav()" href="#attributions" title="Attributions to everything made by third parties that <!--%#INSERT:NAME#%--> is based on">Attributions</a>
+            </div>
+        </div>
+    </body>
+</html>

+ 1245 - 0
docs/raw/index.js

@@ -0,0 +1,1245 @@
+"use strict";
+
+var settings = {
+    baseURL: "<!--%#INSERT:DOCSURL#%-->",
+    // baseURL: "http://127.0.0.1:8076/", // DEBUG
+    jokeEndpoint: "joke",
+    anyCategoryName: "Any",
+    defaultFormat: "json",
+    submitUrl: "<!--%#INSERT:DOCSURL#%-->/submit",
+    // submitUrl: "http://127.0.0.1:8076/submit", // DEBUG
+    defaultLang: "en",
+    formatVersion: 3,
+    contributorsObject: JSON.parse('<!--%#INSERT:CONTRIBUTORS#%-->'),
+    categoryAliasesObject: JSON.parse('<!--%#INSERT:CATEGORYALIASES#%-->')
+};
+
+var submission = {};
+
+if(settings.baseURL.endsWith("/"))
+{
+    settings.baseURL = settings.baseURL.substr(0, (settings.baseURL.length - 1));
+}
+
+var tryItOk = false;
+var tryItURL = "";
+var dIHTML = `
+<h2>To provide this service to you, JokeAPI needs to collect some anonymous data.</h2>
+
+<br>
+
+<a href="<!--%#INSERT:PRIVACYPOLICYURL#%-->" target="_blank">View the privacy policy by clicking here.</a>
+
+<br><br><br>
+
+<b>This is a list of everything JokeAPI stores temporarily (this data will be deleted after about a month):</b><br>
+<ul>
+    <li>A hash of your IP address, your request headers, the request method and the request date and time (this will be kept inside a secure log file so I can debug JokeAPI and help you solve any issues you might have)</li>
+</ul>
+
+<br><br>
+
+<b>This is a list of everything JokeAPI stores indefinitely:</b><br>
+<ul>
+    <li>A hash of your IP address <u>if it gets added to the <i>blacklist</i>.</u> This happens if you have shown malicious behavior or have exceeded the rate limiting for too long / often</li>
+    <li>A hash of your IP address <u>if it gets added to a <i>whitelist</i>.</u> This only happens if you contacted me to get more requests per minute or are partnered with me and have been informed that this is happening</li>
+    <li>A hash of your IP address <u>if it gets added to a <i>console blacklist</i>.</u> This (if at all) also only happens if you are partnered with me</li>
+    <li>The requested URL, consisting of the URL path, the URL parameters and the URL anchor</li>
+    <li>The payload of joke submissions (using POST requests on the submission endpoint)</li>
+</ul>
+
+<br><br>
+
+<b>Terminology:</b><br>
+&nbsp;&nbsp;&nbsp;&nbsp;Hash:<br>
+&nbsp;&nbsp;&nbsp;&nbsp;A hash uses an algorithm to encrypt the input to something that cannot be reconstructed to the initial input again.<br>
+&nbsp;&nbsp;&nbsp;&nbsp;In the case of JokeAPI, your IP address gets hashed and stored to a database.<br>
+&nbsp;&nbsp;&nbsp;&nbsp;In this hashed state, your original IP address can not be reconstructed and you will stay completely anonymous in case any data ever gets leaked, while JokeAPI can still uniquely identify you.
+
+<br><br><br>
+
+Please note that the collection of the above listed data is necessary to provide you this service.<br>
+Without it, rate limiting wouldn't be possible. This would lead to the API and all of my other services being taken down by DoS-attacks.<br>
+This has already happened before and it has impacted all of my services to the point of them being completely unresponsive.<br><br>
+You can request to get your collected data deleted or to view the data about you that JokeAPI collected (according to <a href="https://www.privacy-regulation.eu/en/article-12-transparent-information-communication-and-modalities-for-the-exercise-of-the-rights-of-the-data-subject-GDPR.htm">article 12 GDPR</a>) by sending me an e-mail: <a href="mailto:[email protected]">[email protected]</a>
+<br><br><br>
+`;
+
+var rsIHTML = `
+<form onsubmit="submitRestartForm()">
+<input type="password" id="restartFormToken" placeholder="Restart Token" style="width:30vw"> <button type="submit">Send &gt;</button>
+</form>
+`;
+
+var sMenu=new function(){this.new=function(id,title,innerhtml,width,height,border_rounded,closable_ESC,closable_btn,on_close,close_img_src){if(typeof id=="string"&&typeof title=="string"&&typeof innerhtml=="string"&&typeof width=="number"&&typeof height=="number"){if(gebid("jsg_menu_"+id)!=null){console.error("a menu with the ID "+id+" already exists - not creating a new one");return}
+/* eslint-disable-next-line */
+if(!border_rounded)border_rounded=!0;if(typeof closable_ESC!="boolean")closable_ESC=!0;if(typeof closable_btn!="boolean")closable_btn=!0;if(!on_close)on_close=function(){};if(!close_img_src)close_img_src="https://sv443.net/resources/images/jsg_menu_close.png";var menuelem=document.createElement("div");menuelem.style.display="none";menuelem.style.opacity="0";menuelem.style.transition="opacity 0.3s ease-in";menuelem.style.overflow="auto";menuelem.className="jsg_menu";menuelem.id="jsg_menu_"+id;menuelem.style.position="fixed";menuelem.style.top=((100-height)/2)+"vh";menuelem.style.left=((100-width)/2)+"vw";menuelem.style.width=width+"vw";menuelem.style.height=height+"vh";menuelem.style.padding="10px";menuelem.style.border="2px solid #454545";if(border_rounded)menuelem.style.borderRadius="1.2em";else menuelem.style.borderRadius="0";if(closable_btn)var closebtnih='<img onclick="sMenu.close(\''+id+'\')" class="jsg_menuclosebtn" title="Close" src="https://sv443.net/cdn/jsl/closebtn.png" style="cursor:pointer;position:absolute;top:0;right:0;width:1.5em;height:1.5em;">';else closebtnih="";menuelem.style.backgroundColor="#ddd";menuelem.innerHTML="<div class='jsg_menutitle' style='font-size:1.5em;text-align:center;'>"+title+"</div>"+closebtnih+"<br>"+innerhtml;document.body.appendChild(menuelem);if(closable_ESC)document.addEventListener("keydown",function(e){if(e.keyCode==27)sMenu.close(id)})}
+else{console.error("the arguments for Menu.new() are wrong");return!1}}
+this.close=function(id){try{setTimeout(function(){gebid("jsg_menu_"+id).style.display="none"},500);gebid("jsg_menu_"+id).style.opacity="0";gebid("jsg_menu_"+id).style.transition="opacity 0.3s ease-in"}
+catch(err){console.error("couldn't find menu with id "+id+". Is the ID correct and was the menu created correctly?");return!1}}
+this.open=function(id){try{gebid("jsg_menu_"+id).style.display="block";setTimeout(function(){gebid("jsg_menu_"+id).style.opacity="1";gebid("jsg_menu_"+id).style.transition="opacity 0.3s ease-out"},20)}
+catch(err){console.error("couldn't find menu with id "+id+". Is the ID correct and was the menu created correctly?");return!1}}
+this.theme=function(id,theme){try{if(theme=="dark"){gebid("jsg_menu_"+id).style.backgroundColor="#454545";gebid("jsg_menu_"+id).style.color="white";gebid("jsg_menu_"+id).style.borderColor="#ddd";gebid("jsg_menu_"+id).style.transition="background-color 0.4s ease-out, color 0.4s ease-out, border-color 0.4s ease-out"}
+else{gebid("jsg_menu_"+id).style.backgroundColor="#ddd";gebid("jsg_menu_"+id).style.color="black";gebid("jsg_menu_"+id).style.borderColor="#454545";gebid("jsg_menu_"+id).style.transition="background-color 0.4s ease-out, color 0.4s ease-out, border-color 0.4s ease-out"}}
+catch(err){console.error("couldn't find menu with id "+id+". Is the ID correct and was the menu created correctly?");return!1}}
+this.setInnerHTML=function(id,inner_html){try{gebid("jsg_menu_"+id).innerHTML=inner_html}
+catch(err){console.error("couldn't find menu or inner_html is not valid");return!1}}
+this.setOuterHTML=function(id,outer_html){try{gebid("jsg_menu_"+id).outerHTML=outer_html}
+catch(err){console.error("couldn't find menu or outer_html is not valid");return!1}}}
+/**
+ * Alias for document.getElementById()
+ * @param {String} id ID of the element
+ */
+function gebid(id){return document.getElementById(id);}
+
+
+var idRanges = {};
+var maxJokeIdRange = parseInt("<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->");
+
+
+//#MARKER onload
+
+function onLoad()
+{
+    window.jokeapi = {};
+
+    console.log("%cJokeAPI%cDocumentation (v<!--%#INSERT:VERSION#%-->)  -  © Copyright Sv443 Network 2018-2020", "color: #b05ffc; background-color: black; padding: 5px; padding-right: 0; font-size: 1.2em;", "color: white; background-color: black; padding: 5px; font-size: 1.2em;");
+
+    gebid("content").onclick = function(e){tryCloseSideNav(e)};
+    document.getElementsByTagName("header")[0].onclick = function(e){tryCloseSideNav(e)};
+    gebid("docTitle").onclick = function(){window.location.reload()};
+
+    addCodeTabs();
+
+    sMenu.new("privacyPolicy", "What data does JokeAPI collect?", dIHTML, 85, 85, true, true, true);
+    sMenu.theme("privacyPolicy", "dark");
+
+    sMenu.new("restartPrompt", "Restart <!--%#INSERT:NAME#%-->", rsIHTML, 40, 30, true, true, true);
+    sMenu.theme("restartPrompt", "dark");
+
+
+    // eslint-disable-next-line no-undef
+    if(Cookies.get("hideUsageTerms") == "true")
+    {
+        gebid("usageTerms").style.display = "none";
+    }
+    else
+    {
+        gebid("usageTerms").style.display = "inline-block";
+    }
+
+
+    document.addEventListener("keydown", function(e) {
+        if(e.key == "Escape" && window.jokeapi.sidenavOpened)
+            closeNav();
+        else if(e.key == "R" && e.altKey && e.shiftKey)
+        {
+            openRestartForm();
+        }
+    });
+
+    resetTryItForm();
+
+    setTimeout(function() {
+        gebid("usageTerms").dataset.animateBorder = "true";
+    }, 800);
+
+    buildURL();
+
+    gebid("content").addEventListener("click", function(e) {
+        if(document.body.dataset["sidenav"] == "opened")
+        {
+            e.preventDefault();
+            closeNav();
+        }
+    });
+
+    var selectOnInitElems = document.getElementsByClassName("selectOnInit");
+
+    for(var i = 0; i < selectOnInitElems.length; i++)
+    {
+        selectOnInitElems[i].selected = true;
+    }
+
+    var uncheckOnInitElems = document.getElementsByClassName("uncheckOnInit");
+    for(var iii = 0; iii < uncheckOnInitElems.length; iii++)
+    {
+        uncheckOnInitElems[iii].checked = false;
+    }
+
+    var clearOnInitElems = document.getElementsByClassName("clearOnInit");
+    for(var j = 0; j < clearOnInitElems.length; j++)
+    {
+        clearOnInitElems[j].value = "";
+    }
+
+    try
+    {
+        var fileFormats = JSON.parse('<!--%#INSERT:FILEFORMATARRAY#%-->');
+        if(fileFormats.includes("JSON"))
+        {
+            fileFormats.splice(fileFormats.indexOf("JSON"), 1);
+        }
+        Array.from(document.getElementsByClassName("insFormatsS")).forEach(function(el) {
+            el.innerText = fileFormats.join(" and ");
+        });
+
+        var flags = JSON.parse('<!--%#INSERT:FLAGSARRAY#%-->');
+        Array.from(document.getElementsByClassName("insFlags")).forEach(function(el) {
+            el.innerText = flags.join(", ");
+        });
+
+        var formats = JSON.parse('<!--%#INSERT:FILEFORMATARRAY#%-->');
+        Array.from(document.getElementsByClassName("insFormats")).forEach(function(el) {
+            el.innerText = formats.join(", ").toLowerCase();
+        });
+
+        var categories = JSON.parse('<!--%#INSERT:CATEGORYARRAY#%-->');
+        Array.from(document.getElementsByClassName("insCategories")).forEach(function(el) {
+            el.innerText = categories.join(", ");
+        });
+    }
+    catch(err)
+    {
+        return alert("Documentation compilation was unsuccessful: Value insertion error:\n" + err);
+    }
+    
+    //#SECTION submission form
+    var inputElems = [
+        "f_category",
+        "f_type",
+        "f_flags_nsfw",
+        "f_flags_religious",
+        "f_flags_political",
+        "f_flags_racist",
+        "f_flags_sexist",
+        "f_flags_explicit",
+        "f_setup",
+        "f_delivery",
+        "f_language"
+    ];
+
+    for(var ii = 0; ii < inputElems.length; ii++)
+    {
+        var elm = gebid(inputElems[ii]);
+
+        if(elm.tagName.toLowerCase() != "textarea")
+        {
+            elm.onchange = function()
+            {
+                return valChanged(this);
+            }
+        }
+        else
+        {
+            elm.oninput = function()
+            {
+                return valChanged(this);
+            }
+        }
+    }
+
+    var langXhr = new XMLHttpRequest();
+    var langUrl = settings.baseURL + "/languages";
+    var langSelectElems = document.getElementsByClassName("appendLangOpts");
+
+    var otherOpt = document.createElement("option");
+    otherOpt.innerText = "Other / Custom";
+    otherOpt.value = "other";
+    gebid("f_language").appendChild(otherOpt);
+
+    var sysLangsText = "";
+    var jokeLangsText = "";
+
+    langXhr.open("GET", langUrl);
+    langXhr.onreadystatechange = function() {
+        var xErrElem;
+        if(langXhr.readyState == 4 && langXhr.status < 300)
+        {
+            var respJSON = JSON.parse(langXhr.responseText.toString());
+            var languagesArray = respJSON.jokeLanguages;
+
+            sysLangsText = respJSON.systemLanguages.join(", ");
+            jokeLangsText = respJSON.jokeLanguages.join(", ");
+
+            for(var i = 0; i < languagesArray.length; i++)
+            {
+                for(var j = 0; j < langSelectElems.length; j++)
+                {
+                    var langName = "";
+
+                    for(var k = 0; k < respJSON.possibleLanguages.length; k++)
+                    {
+                        if(respJSON.possibleLanguages[k].code == languagesArray[i])
+                        {
+                            langName = respJSON.possibleLanguages[k].name;
+                        }
+                    }
+
+                    xErrElem = document.createElement("option");
+                    xErrElem.value = languagesArray[i];
+                    if(languagesArray[i] == settings.defaultLang)
+                    {
+                        xErrElem.selected = true;
+                    }
+                    xErrElem.innerText = languagesArray[i] + " - " + langName;
+
+                    if(languagesArray[i] == settings.defaultLang)
+                        xErrElem.selected = true;
+
+                    langSelectElems[j].appendChild(xErrElem);
+                    langSelectElems[j].value = settings.defaultLang;
+                }
+            }
+
+            var sysLangsElems = document.getElementsByClassName("insSysLangs");
+            var jokeLangsElems = document.getElementsByClassName("insJokeLangs");
+
+            for(var sI = 0; sI < sysLangsElems.length; sI++)
+            {
+                sysLangsElems[sI].innerText = sysLangsText;
+            }
+
+            for(var jI = 0; jI < sysLangsElems.length; jI++)
+            {
+                jokeLangsElems[jI].innerText = jokeLangsText;
+            }
+        }
+        else if(langXhr.readyState == 4 && langXhr.status >= 300)
+        {
+            for(var ii = 0; ii < langSelectElems.length; ii++)
+            {
+                xErrElem = document.createElement("option");
+                xErrElem.value = "en";
+                xErrElem.innerText = "en - English";
+                xErrElem.selected = true;
+
+                langSelectElems[ii].appendChild(xErrElem);
+            }
+        }
+    };
+    langXhr.send();
+
+
+    var infoXhr = new XMLHttpRequest();
+    infoXhr.open("GET", (settings.baseURL + "/info"));
+
+    infoXhr.onreadystatechange = function() {
+        if(infoXhr.readyState == 4 && infoXhr.status < 300)
+        {
+            var respJSON = JSON.parse(infoXhr.responseText.toString());
+
+            var idrKeys = Object.keys(respJSON.jokes.idRange);
+            for(var i = 0; i < idrKeys.length; i++)
+            {
+                var idrKey = idrKeys[i];
+                idRanges[idrKey] = respJSON.jokes.idRange[idrKey];
+
+                try
+                {
+                    console.info("<!--%#INSERT:NAME#%--> is serving " + (respJSON.jokes.idRange[idrKey][1] + 1) + " jokes from language \"" + idrKey + "\"");
+                }
+                catch(err)
+                {
+                    void(err);
+                }
+            }
+
+            reRender();
+        }
+        else if(infoXhr.readyState == 4 && infoXhr.status >= 300)
+        {
+            console.error("Couldn't get ID range of all languages. Defaulting to the max possible value.");
+        }
+    };
+
+    infoXhr.send();
+
+
+    gebid("submitBtn").addEventListener("click", function() {
+        submitJoke();
+    });
+
+    buildSubmission();
+    setTimeout(function() { buildSubmission() }, 2000);
+
+    gebid("insUserAgent").innerText = navigator.userAgent;
+
+    gebid("lcodeSelect").value = settings.defaultLang;
+
+    var abElems = document.getElementsByClassName("antiBotE");
+    for(var l = 0; l < abElems.length; l++)
+    {
+        abElems[l].onclick = function()
+        {
+            if(!this.classList.contains("shown"))
+            {
+                this.innerText = atob(this.dataset.enc);
+                this.classList.add("shown");
+            }
+        }
+    }
+
+    gebid("lcodeSelect").value = settings.defaultLang;
+    gebid("lcodeSelect").onchange = function() { reRender(true) };
+    gebid("sideNavOpen").onclick = function() { return openNav(); };
+
+
+    loadCategoryAliases();
+    loadContributors();
+}
+
+/**
+ * @param {MouseEvent} e
+ */
+function tryCloseSideNav(e)
+{
+    if(document.body.dataset["sidenav"] == "opened")
+    {
+        e.preventDefault();
+        closeNav();
+    }
+}
+
+function addCodeTabs()
+{
+    var codeElements = document.getElementsByTagName("code");
+
+    for(var i = 0; i < codeElements; i++)
+    {
+        if(codeElements[i].classList.contains("prettyprint"))
+            codeElements[i].innerHTML = codeElements[i].innerHTML.replace(/&tab;/gm, "&nbsp;&nbsp;&nbsp;&nbsp;");
+    }
+}
+
+//#MARKER SideNav
+function openNav()
+{
+    console.info("opening nav");
+
+    setTimeout(function() {
+        document.body.dataset["sidenav"] = "opened";
+    }, 50);
+
+    window.jokeapi.sidenavOpened = true;
+
+    gebid("sidenav").style.width = "280px";
+    gebid("content").style.marginLeft = "280px";
+    document.getElementsByTagName("header")[0].dataset["grayscaled"] = "true";
+    gebid("sideNavOpen").style.visibility = "hidden";
+}
+  
+function closeNav()
+{
+    console.info("closing nav");
+
+    if(document.body.dataset["sidenav"] != "opened")
+        return;
+
+    window.jokeapi.sidenavOpened = false;
+
+    document.body.dataset["sidenav"] = "closed";
+
+    gebid("sidenav").style.width = "0";
+    gebid("content").style.marginLeft = "10px";
+    document.getElementsByTagName("header")[0].dataset["grayscaled"] = "false";
+    gebid("sideNavOpen").style.visibility = "visible";
+}
+
+// function getQueryStringObject()
+// {
+//     var qstrObj = {};
+
+//     if(!window.location.href.includes("?"))
+//         return null;
+
+//     var rawQstr = window.location.href.split("?")[1];
+//     var qstrArr = [];
+
+//     if(rawQstr.includes("#"))
+//         rawQstr = rawQstr.split("#")[0];
+
+//     if(rawQstr != null && rawQstr.includes("&"))
+//         qstrArr = rawQstr.split("&");
+//     else if(rawQstr != null)
+//         qstrArr = [rawQstr];
+//     else return null;
+
+
+//     if(qstrArr.length > 0)
+//         qstrArr.forEach(function(qstrEntry) {
+//             if(qstrEntry.includes("="))
+//                 qstrObj[qstrEntry.split("=")[0]] = qstrEntry.split("=")[1];
+//         });
+//     else return null;
+
+//     return qstrObj;
+// }
+
+/**
+ * @param {Boolean} [langChanged]
+ */
+function reRender(langChanged)
+{
+    console.info("Re-rendering try-it form");
+
+    var allOk = true;
+
+    //#SECTION category
+    var isValid = false;
+    document.getElementsByName("catSelect").forEach(function(el) {
+        if(!el.checked)
+            return;
+
+        if(el.value == "any")
+        {
+            isValid = true;
+            ["cat-cb1", "cat-cb2", "cat-cb3", "cat-cb4", "cat-cb5", "cat-cb6"].forEach(function(cat) {
+                gebid(cat).disabled = true;
+            });
+        }
+        else
+        {
+            var isChecked = false;
+            ["cat-cb1", "cat-cb2", "cat-cb3", "cat-cb4", "cat-cb5", "cat-cb6"].forEach(function(cat) {
+                var cel = gebid(cat);
+                cel.disabled = false;
+
+                if(cel.checked)
+                    isChecked = true;
+            });
+
+            if(isChecked)
+                isValid = true;
+        }
+    });
+
+    if(!isValid)
+    {
+        allOk = false;
+        gebid("categoryWrapper").style.borderColor = "red";
+    }
+    else
+    {
+        gebid("categoryWrapper").style.borderColor = "initial";
+    }
+
+
+    //#SECTION format
+    if(!gebid("typ-cb1").checked && !gebid("typ-cb2").checked)
+    {
+        allOk = false;
+        gebid("typeSelectWrapper").style.borderColor = "red";
+    }  
+    else
+    {
+        gebid("typeSelectWrapper").style.borderColor = "initial";
+    }
+
+
+    //#SECTION id range
+    if(langChanged === true)
+    {
+        console.warn("langchanged")
+
+        var langCode = gebid("lcodeSelect").value;
+
+        if(idRanges[langCode])
+        {
+            var maxRange = parseInt(idRanges[langCode][1]);
+
+            gebid("idRangeInputTo").max = maxRange;
+            gebid("idRangeInputTo").value = maxRange;
+
+            maxJokeIdRange = maxRange;
+        }
+        else
+        {
+            gebid("idRangeInputTo").max = parseInt("<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->");
+
+            maxJokeIdRange = parseInt("<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->");
+        }
+    }
+
+    var numRegex = /^[0-9]+$/gm;
+    var fromVal = gebid("idRangeInputFrom").value;
+    var toVal = gebid("idRangeInputTo").value;
+    var fromValInt = parseInt(fromVal);
+    var toValInt = parseInt(toVal);
+    var outOfRange = (fromValInt < 0 || toValInt > maxJokeIdRange);
+    var notNumber = ((fromVal.match(numRegex) == null) || (toVal.match(numRegex) == null));
+
+    if(outOfRange || notNumber || fromValInt > toValInt)
+    {
+        allOk = false;
+        gebid("idRangeWrapper").style.borderColor = "red";
+    }
+    else
+    {
+        gebid("idRangeWrapper").style.borderColor = "initial";
+    }
+
+    var jokesAmount = parseInt(gebid("jokesAmountInput").value);
+
+    if(jokesAmount > parseInt("<!--%#INSERT:MAXJOKEAMOUNT#%-->") || jokesAmount < 1 || isNaN(jokesAmount))
+    {
+        allOk = false;
+        gebid("jokeAmountWrapper").style.borderColor = "red";
+    }
+    else
+    {
+        gebid("jokeAmountWrapper").style.borderColor = "initial";
+    }
+
+    if(allOk)
+    {
+        tryItOk = true;
+    }
+    else
+    {
+        tryItOk = false;
+    }
+
+    buildURL();
+}
+
+//#MARKER build URL
+function buildURL()
+{
+    var queryParams = [];
+
+    //#SECTION categories
+    var selectedCategories = [settings.anyCategoryName];
+    if(gebid("cat-radio2").checked)
+    {
+        selectedCategories = [];
+        if(gebid("cat-cb1").checked)
+        {
+            selectedCategories.push("Programming");
+        }
+        if(gebid("cat-cb2").checked)
+        {
+            selectedCategories.push("Miscellaneous");
+        }
+        if(gebid("cat-cb3").checked)
+        {
+            selectedCategories.push("Dark");
+        }
+        if(gebid("cat-cb4").checked)
+        {
+            selectedCategories.push("Pun");
+        }
+        if(gebid("cat-cb5").checked)
+        {
+            selectedCategories.push("Spooky");
+        }
+        if(gebid("cat-cb6").checked)
+        {
+            selectedCategories.push("Christmas");
+        }
+
+        if(selectedCategories.length == 0)
+        {
+            selectedCategories.push(settings.anyCategoryName);
+        }
+    }
+
+
+    //#SECTION language
+    var langCode = gebid("lcodeSelect").value || settings.defaultLang;
+    if(langCode != settings.defaultLang)
+        queryParams.push("lang=" + langCode);
+
+
+    //#SECTION flags
+    var flagElems = [gebid("blf-cb1"), gebid("blf-cb2"), gebid("blf-cb3"), gebid("blf-cb4"), gebid("blf-cb5"), gebid("blf-cb6")];
+    var flagNames = JSON.parse('<!--%#INSERT:FLAGSARRAY#%-->');
+    var selectedFlags = [];
+    flagElems.forEach(function(el, i) {
+        if(el.checked)
+        {
+            selectedFlags.push(flagNames[i]);
+        }
+    });
+
+    if(selectedFlags.length > 0)
+    {
+        queryParams.push("blacklistFlags=" + selectedFlags.join(","));
+    }
+
+
+    //#SECTION format
+    var formatElems = [gebid("fmt-cb1"), gebid("fmt-cb2"), gebid("fmt-cb3"), gebid("fmt-cb4")];
+    formatElems.forEach(function(el) {
+        if(el.checked && el.value != settings.defaultFormat)
+        {
+            queryParams.push("format=" + el.value);
+        }
+    });
+
+
+    //#SECTION type
+    var singleJoke = gebid("typ-cb1").checked;
+    var twopartJoke = gebid("typ-cb2").checked;
+    if(singleJoke ^ twopartJoke == 1)
+    {
+        if(singleJoke)
+        {
+            queryParams.push("type=single");
+        }
+        else if(twopartJoke)
+        {
+            queryParams.push("type=twopart");
+        }
+    }
+
+
+    //#SECTION search string
+    var sstr = gebid("searchStringInput").value;
+    if(sstr)
+    {
+        queryParams.push("contains=" + encodeURIComponent(sstr));
+    }
+
+
+    //#SECTION id range
+    var range = [parseInt(gebid("idRangeInputFrom").value), parseInt(gebid("idRangeInputTo").value)];
+    if(!isNaN(range[0]) && !isNaN(range[1]) && range[0] >= 0 && range[1] <= maxJokeIdRange && range[1] >= range[0])
+    {
+        if(range[0] == range[1] && range[0] >= 0 && range[0] <= maxJokeIdRange)
+        {
+            // Use "x" format
+            queryParams.push("idRange=" + range[0]);
+        }
+        else if(range[0] != 0 || range[1] != maxJokeIdRange)
+        {
+            // Use "x-y" format
+            queryParams.push("idRange=" + range[0] + "-" + range[1]);
+        }
+    }
+
+
+    //#SECTION amount
+    var jokeAmount = parseInt(gebid("jokesAmountInput").value);
+    if(jokeAmount != 1 && !isNaN(jokeAmount) && jokeAmount > 0 && jokeAmount <= parseInt("<!--%#INSERT:MAXJOKEAMOUNT#%-->"))
+    {
+        queryParams.push("amount=" + jokeAmount);
+    }
+
+
+    tryItURL = settings.baseURL + "/" + settings.jokeEndpoint + "/" + selectedCategories.join(",");
+
+    if(queryParams.length > 0)
+    {
+        tryItURL += "?" + queryParams.join("&");
+    }
+
+    gebid("urlBuilderUrl").innerText = tryItURL;
+}
+
+//#MARKER send request
+function sendTryItRequest()
+{
+    var sendStartTimestamp = new Date().getTime();
+    var prpr = gebid("urlBuilderPrettyprint");
+    var tryItRequestError = function(err) {
+        if(prpr.classList.contains("prettyprint"))
+        {
+            prpr.classList.remove("prettyprint");
+        }
+
+        if(prpr.classList.contains("prettyprinted"))
+        {
+            prpr.classList.remove("prettyprinted");
+        }
+
+        gebid("tryItResult").innerHTML = "Error:<br><br>" + err;
+    }
+
+    if(!tryItOk)
+    {
+        return tryItRequestError("One or more of the parameters you specified are invalid.\nThey are outlined with a red border.\n\nPlease correct the parameters and try again.");
+    }
+
+    if(!prpr.classList.contains("prettyprint"))
+    {
+        prpr.classList.add("prettyprint");
+    }
+
+    if(prpr.classList.contains("prettyprinted"))
+    {
+        prpr.classList.remove("prettyprinted");
+    }
+
+    if(prpr.classList.contains("lang-json"))
+    {
+        prpr.classList.remove("lang-json");
+    }
+
+    if(prpr.classList.contains("lang-yaml"))
+    {
+        prpr.classList.remove("lang-yaml");
+    }
+
+    if(prpr.classList.contains("lang-xml"))
+    {
+        prpr.classList.remove("lang-xml");
+    }
+
+    prpr.classList.add("lang-json");
+
+    try
+    {
+        var xhr = new XMLHttpRequest();
+        xhr.open("GET", tryItURL);
+        xhr.onreadystatechange = function() {
+            if(xhr.readyState == 4 && (xhr.status < 300 || xhr.status == 429))
+            {
+                var result = "";
+                if(xhr.getResponseHeader("content-type").includes("json"))
+                {
+                    result = JSON.stringify(JSON.parse(xhr.responseText.toString()), null, 4);
+                }
+                else
+                {
+                    if(xhr.getResponseHeader("content-type").includes("xml"))
+                    {
+                        gebid("urlBuilderPrettyprint").classList.remove("lang-json");
+                        gebid("urlBuilderPrettyprint").classList.add("lang-xml");
+                    }
+                    else
+                    {
+                        gebid("urlBuilderPrettyprint").classList.remove("lang-json");
+                        gebid("urlBuilderPrettyprint").classList.add("lang-yaml");
+                    }
+
+                    result = xhr.responseText.toString();
+                    result = result.replace(/[<]/gm, "&lt;");
+                    result = result.replace(/[>]/gm, "&gt;");
+                }
+
+                gebid("tryItResult").innerText = result;
+                gebid("tryItFormLatency").innerText = "Latency: " + (new Date().getTime() - sendStartTimestamp) + " ms";
+
+                PR.prettyPrint(); // eslint-disable-line no-undef
+            }
+            else
+            {
+                tryItRequestError(xhr.responseText);
+            }
+        }
+        xhr.send();
+    }
+    catch(err)
+    {
+        tryItRequestError(err);
+    }
+}
+
+//#MARKER interactive elements
+function resetTryItForm(confirmation)
+{
+    if(confirmation === true && !confirm("Do you really want to reset the form?"))
+        return;
+
+    ["cat-cb1", "cat-cb2", "cat-cb3", "cat-cb4", "cat-cb5", "cat-cb6"].forEach(function(cat) {
+        gebid(cat).checked = false;
+    });
+
+    gebid("cat-radio1").checked = true;
+
+    ["blf-cb1", "blf-cb2", "blf-cb3", "blf-cb4", "blf-cb5", "blf-cb6"].forEach(function(flg) {
+        gebid(flg).checked = false;
+    });
+
+    gebid("fmt-cb1").checked = true;
+
+    ["typ-cb1", "typ-cb2"].forEach(function(type) {
+        gebid(type).checked = true;
+    });
+
+    gebid("searchStringInput").value = "";
+
+    gebid("idRangeInputFrom").value = 0;
+    gebid("idRangeInputTo").value = parseInt("<!--%#INSERT:TOTALJOKESZEROINDEXED#%-->");
+
+    gebid("jokesAmountInput").value = 1;
+
+    reRender();
+}
+
+//#MARKER submit joke
+function submitJoke()
+{
+    var submitBtn = gebid("submitBtn");
+    if(submitBtn.disabled == true)
+    {
+        return;
+    }
+
+    var xhr = new XMLHttpRequest();
+    xhr.open("POST", settings.submitUrl);
+
+    xhr.onreadystatechange = function() {
+        if(xhr.readyState == 4)
+        {
+            var res;
+            try
+            {
+                res = JSON.parse(xhr.responseText);
+            }
+            catch(err)
+            {
+                alert("Error " + res.status + " while sending your submission:\n" + res.message + (res.additionalInfo ? "\n\nAdditional info:\n" + res.additionalInfo : ""));
+                return;
+            }
+
+            if(xhr.status < 300)
+            {
+                if(res.error == false)
+                {
+                    submitBtn.disabled = true;
+
+                    alert(res.message);
+                    
+                    setTimeout(function() {
+                        gebid("submitBtn").disabled = false;
+                    }, 2000);
+                }
+                else if(res.error == true)
+                {
+                    alert("Error " + res.status + " while sending your submission:\n" + res.message + (res.additionalInfo ? "\n\nAdditional info:\n" + res.additionalInfo : ""));
+                }
+            }
+            else if(xhr.status >= 300)
+            {
+                var addInfo = res.message;
+                if(res.additionalInfo)
+                {
+                    addInfo = res.additionalInfo;
+                }
+                alert("Error while sending your submission:\n" + addInfo);
+            }
+        }
+    };
+
+    xhr.send(JSON.stringify(submission, null, 4));
+}
+
+function valChanged(element)
+{
+    if(element.id == "f_type")
+    {
+        if(element.value == "single")
+        {
+            gebid("f_setup").placeholder = "Joke";
+
+            gebid("f_delivery").style.display = "none";
+        }
+        else if(element.value == "twopart")
+        {
+            gebid("f_setup").placeholder = "Setup";
+
+            gebid("f_delivery").style.display = "initial";
+        }
+    }
+
+    if(element.id == "f_language")
+    {
+        if(element.value == "other")
+        {
+            gebid("f_langHideContainer").classList.remove("hidden");
+        }
+        else
+        {
+            gebid("f_langHideContainer").classList.add("hidden");
+        }
+    }
+
+    buildSubmission();
+}
+
+function buildSubmission()
+{
+    var category = gebid("f_category").value;
+    var type = gebid("f_type").value;
+
+    submission = {
+        formatVersion: settings.formatVersion,
+        category: category,
+        type: type
+    }
+
+    if(type == "single")
+    {
+        submission.joke = gebid("f_setup").value;
+    }
+    else if(type == "twopart")
+    {
+        submission.setup = gebid("f_setup").value;
+        submission.delivery = gebid("f_delivery").value;
+    }
+
+    var sLang = gebid("f_language").value || settings.defaultLang;
+
+    if(sLang == "other")
+    {
+        var elVal = gebid("f_customLang").value;
+        if(elVal && elVal.length == 2)
+        {
+            sLang = elVal;
+        }
+        else
+        {
+            sLang = "Please enter 2 char language code";
+        }
+    }
+
+    submission = {
+        ...submission,
+        flags: {
+            nsfw: gebid("f_flags_nsfw").checked,
+            religious: gebid("f_flags_religious").checked,
+            political: gebid("f_flags_political").checked,
+            racist: gebid("f_flags_racist").checked,
+            sexist: gebid("f_flags_sexist").checked,
+            explicit: gebid("f_flags_explicit").checked
+        },
+        lang: sLang
+    };
+
+    var subDisp = gebid("submissionDisplay");
+
+    var escapedSubmission = JSON.parse(JSON.stringify(submission)); // copy value without reference
+    if(type == "single")
+    {
+        escapedSubmission.joke = htmlEscape(submission.joke);
+    }
+    else if(type == "twopart")
+    {
+        escapedSubmission.setup = htmlEscape(submission.setup);
+        escapedSubmission.delivery = htmlEscape(submission.delivery);
+    }
+    subDisp.innerText = JSON.stringify(escapedSubmission, null, 4);
+
+    var subCodeElem = gebid("submissionCodeElement");
+
+    if(!subCodeElem.classList.contains("prettyprint"))
+    {
+        subCodeElem.classList.add("prettyprint");
+    }
+
+    if(subCodeElem.classList.contains("prettyprinted"))
+    {
+        subCodeElem.classList.remove("prettyprinted");
+    }
+
+    if(subCodeElem.classList.contains("lang-json"))
+    {
+        subCodeElem.classList.remove("lang-json");
+    }
+
+    subCodeElem.classList.add("lang-json");
+
+    setTimeout(function() {
+        PR.prettyPrint(); // eslint-disable-line no-undef
+    }, 5);
+}
+
+/**
+ * Escapes unsafe HTML
+ * @param {String} unsafeHTML
+ * @returns {String}
+ */
+function htmlEscape(unsafeHTML)
+{
+    unsafeHTML = unsafeHTML.replace(/</g, "&lt;");
+    unsafeHTML = unsafeHTML.replace(/>/g, "&gt;");
+
+    return unsafeHTML;
+}
+
+//#MARKER privacy policy
+function privPolMoreInfo()
+{
+    sMenu.open("privacyPolicy");
+}
+
+function hideUsageTerms()
+{
+    gebid("usageTerms").style.display = "none";
+    Cookies.set("hideUsageTerms", "true", {"expires": 365}); // eslint-disable-line no-undef
+}
+
+
+//#MARKER misc
+function openRestartForm()
+{
+    sMenu.open("restartPrompt");
+}
+
+function submitRestartForm()
+{
+    restart(gebid("restartFormToken").value || null);
+    sMenu.close("restartPrompt");
+}
+
+function restart(token)
+{
+    if(!token)
+    {
+        token = prompt("Enter restart token:");
+    }
+
+    if(!token)
+    {
+        return;
+    }
+
+    var restartXhr = new XMLHttpRequest();
+    restartXhr.open("POST", settings.baseURL + "/restart");
+    restartXhr.onreadystatechange = function() {
+        if(restartXhr.readyState == 4)
+        {
+            if(restartXhr.status == 400)
+            {
+                console.warn("Error 400 - The entered token is invalid");
+                alert("Error 400 - The entered token is invalid");
+            }
+            else if(restartXhr.status >= 300)
+            {
+                console.warn("Error " + restartXhr.status + " - " + restartXhr.responseText);
+                alert("Error " + restartXhr.status + " - " + restartXhr.responseText);
+            }
+            else if(restartXhr.status < 300)
+            {
+                var xhrData = JSON.parse(restartXhr.responseText);
+                console.info(xhrData.message + "\nInternal Time of Restart: " + toFormattedDate(xhrData.timestamp));
+                alert(xhrData.message + "\nInternal Time of Restart: " + toFormattedDate(xhrData.timestamp));
+            }
+        }
+    };
+    restartXhr.send(token.toString());
+}
+
+function toFormattedDate(unixTimestamp)
+{
+    var d = new Date(unixTimestamp);
+    return d.toLocaleString("de-DE");
+}
+
+/**
+ * Parses the contributor object and puts the contents into the element #contributorsContainer
+ */
+function loadContributors()
+{
+    var container = gebid("contributorsContainer");
+
+    container.innerHTML = "";
+
+    settings.contributorsObject.forEach(function(contributor) {
+        var contrElem = document.createElement("div");
+        contrElem.classList.add("contributor");
+
+        var name = document.createElement("div");
+        name.classList.add("contributorName");
+        name.innerText = contributor.name;
+        contrElem.appendChild(name);
+
+        if(typeof contributor.email == "string")
+        {
+            var email = document.createElement("a");
+            email.href = "mailto:" + contributor.email + "?subject=JokeAPI%20Contribution";
+            email.innerText = contributor.email;
+            email.classList.add("contributorEmail");
+            email.classList.add("contributorContact");
+            contrElem.appendChild(email);
+        }
+
+        if(typeof contributor.url == "string")
+        {
+            var url = document.createElement("a");
+            url.href = contributor.url;
+            url.innerText = contributor.url;
+            url.classList.add("contributorURL");
+            url.classList.add("contributorContact");
+            contrElem.appendChild(url);
+        }
+
+        if(Array.isArray(contributor.contributions))
+        {
+            var contributionList = document.createElement("ul");
+            contributionList.classList.add("contributorContributionsList");
+            contributor.contributions.forEach(function(contribution) {
+                var contributionEl = document.createElement("li");
+                contributionEl.innerText = contribution;
+                contributionList.appendChild(contributionEl);
+            });
+            contrElem.appendChild(contributionList);
+        }
+
+        container.appendChild(contrElem);
+    });
+
+    /*
+    [
+        {
+            "name": "XY",
+            "email": "[email protected]", (OPTIONAL)
+            "url": "https://example.org", (OPTIONAL)
+            "contributions": [
+                "Implemented a whole lot of stuff",
+                "Stuff",
+                "Even more stuff",
+                ...
+            ]
+        },
+        ...
+    ]
+    */
+}
+
+/**
+ * Loads all category aliases, adding them to the element #catAliasesContainer
+ */
+function loadCategoryAliases()
+{
+    var container = gebid("catAliasesContainer");
+    var amt = 0;
+
+    Object.keys(settings.categoryAliasesObject).forEach(function(key) {
+        var value = settings.categoryAliasesObject[key];
+
+        var trElem = document.createElement("tr");
+
+
+        var keyElem = document.createElement("td");
+        keyElem.innerText = key;
+
+        var valElem = document.createElement("td");
+        valElem.innerText = value;
+
+        trElem.appendChild(keyElem);
+        trElem.appendChild(valElem);
+
+
+        container.appendChild(trElem);
+        amt++;
+    });
+
+    console.info("Found " + amt + " category aliases");
+}
+
+
+
+//#MARKER cleanup
+function unused(...args)
+{
+    args.forEach(function(arg) {
+        try{arg.toString();}
+        catch(err) {return;}
+        return;
+    });
+}
+
+unused(openNav, closeNav, onLoad, reRender, privPolMoreInfo, hideUsageTerms, sendTryItRequest, submitRestartForm);