// ==UserScript==
// @name         Runcity
// @namespace    http://tampermonkey.net/
// @version      2024-11-18
// @description  Prettify pages & store data on server
// @author       You
// @match        https://rst.runcity.org/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=runcity.org
// @grant        none
// @require https://code.jquery.com/jquery-3.7.1.min.js
// @require https://cdn.jsdelivr.net/npm/select2@4.1.0-rc.0/dist/js/select2.min.js
// @require https://cdn.jsdelivr.net/npm/swiper@11/swiper-bundle.min.js
// ==/UserScript==

(function () {
  'use strict'

  const adminSite = "runcity.geo.rictum.ru"

  const links = {
    "Волонтеры": "suv_comp",
    "Контрольные пункты": "cp_mgmt",
    "+ КП": "cp_mgmt/?action=edit",
    "Маршруты": "route_mgmt",
    "Тесты": "route_mgmt?test=1",
    "+ Тест": "suv_comp?action=test_online",
    "Склеить": "suv_comp?action=test_merge",
    "Прохождение": "callcenter?action=online_log",
    "КП в трассах": "route_mgmt?action=cp_list",
    "Связанные КП": relatedPointsHref(),
  }

  const catLinks = {
    "Редактор маршрута": (catId) => `route_mgmt?action=edit&cat_id=${catId}`,
    "Конструктор маршрута": (catId) => `route_mgmt/?action=build2&cat_id=${catId}`,
    "Карта": (catId) => `route_mgmt?action=map&cat_id=${catId}`,
    "Этапы": (catId) => `route_mgmt/?action=stages&cat_id=${catId}`,
    "Легенда": (catId) => `route_mgmt?action=preview&cat_id=${catId}&locale_id=ru`,
  }

  const fileInputs = {
    "attachment1": {
      "l": "legend_photo",
      "h": "history_photo"
    },
    "attachment4": {
      "l": "legend_files",
      "h": "history_files"
    },
    "attachment2": "admin_photo",
    "attachment3": "admin_files"
  }

  const fileInpitVariants = {
    "l": "для легенды",
    "h": "для ист. справки"
  }

  const mapsCenterByCompetition = {
    "msk2025": {
      lat: 55.839808,
      ifSouthern: {
        lat: 55.798531,
        lon: 37.690380,
      },
      ifEverywhere: {
        lat: 55.839759,
        lon: 37.706577
      }
    }
  }

  const localStorageItems = {
    NEED_UPDATE_ID: "needUpdateId",
    JUST_CREATED: "justCreated",
    PREV_CREATED: "prevCreated",
    LATTITUDE: "lattitude",
    LONGITUDE: "longitude",
    NOT_PRETTIFY_EDIT_CP: "notPrettifyEditCp",
    REDIRECT_EXIT: "redirectExit",
    DISABLED_PAGES: "disabledPages"
  }

  const ZOOM = 17
  const METERS = 510

  const stageColors = ["#ff8080", "#ffc680", "#ffff80", "#80ff80", "#80ffff", "#8080ff", "#ff80ff"]

  class Tag {
    static make(name, params = {}) {
      let element = document.createElement(name)
      for (const [name, value] of Object.entries(params)) {
        if (name == "on") {
          for (const [event, listener] of Object.entries(value)) {
            element.addEventListener(event, listener)
          }
        }
        else if (name == "classes") {
          element.classList.add(...value.split(" "))
        }
        else if (name == "children") {
          element.append(...value)
        }
        else if (name.startsWith("_")) {
          element.setAttribute(name.substring(1), value)
        }
        else
          element[name] = value
      }

      return element
    }

    static div(params) {
      return this.make("div", params)
    }

    static span(params) {
      return this.make("span", params)
    }

    static button(params) {
      return this.make("button", params)
    }

    static input(params) {
      return this.make("input", params)
    }

    static a(params) {
      return this.make("a", params)
    }

    static img(params) {
      return this.make("img", params)
    }
  }

  class Property {
    name
    content
    desc

    constructor(row = null) {
      if (row == null) return

      let columns = [...row.querySelectorAll('td, th')].map(el => el.cloneNode(true))
      if (columns.length === 1) {
        this.content = columns[0].innerHTML.trim()
      }
      else if (columns.length >= 2) {
        this.name = columns[0].innerHTML.trim()
        let elementsToRemove = columns[1].querySelector("input[name=\"read_gps2\"]")
        if (elementsToRemove) {
          columns[1].removeChild(elementsToRemove)
        }

        this.content = columns[1].innerHTML
          .replaceAll("<br>", "")
          .replaceAll(/\s*Перейти к легенде.*$/gs, "")
          .replaceAll(`общие ("c")`, "")
          .replaceAll(`("l")`, "")
          .replaceAll(`("h")`, "")
          .replaceAll("Получить координаты КП из картинки", "")
          .trim()
      }

      if (columns.length === 3) {
        this.desc = columns[2].innerHTML.replaceAll("<br>", "").replaceAll(/(\n)\s*/g, "$1").trim()
      }
    }

    toDiv(altName = null, altDesc = null) {
      let desc = altDesc ?? this.desc
      let descTooltip = desc ? Tag.div({
        classes: 'desc-icon-container',
        title: desc,
        children: [
          Tag.img({
            classes: 'desc-icon',
            src: ''
          })
        ]
      }) : ""

      return Tag.div({
        children: [
          Tag.make("label", {
            _for: '',
            innerHTML: altName ?? this.name ?? ""
          }),
          Tag.div({
            innerHTML: this.content ?? "",
          }),
          descTooltip
        ]
      })
    }
  }

  function createFrom(rows, classList, data) {
    let container = Tag.div({
      classes: classList
    })
    for (const options of data) {
      let div
      if (options.index != null) {
        let prop = new Property(rows[options.index])
        div = prop.toDiv(options.name, options.desc)
      }
      else {
        let prop = new Property()
        div = prop.toDiv(options.name)
      }
      container.append(div)
    }

    return container
  }

  function createFromMulti(rows, classList, data) {
    let rowData = []
    let toIndex = data.to ?? rows.length - 1
    for (let i = data.from; i <= toIndex; i++) {
      rowData.push({ index: i })
    }

    return createFrom(rows, classList, rowData)
  }

  function addCss(link) {
    document.head.append(
      Tag.make("link", {
        href: link,
        rel: "stylesheet"
      })
    )
  }

  function addJs(link) {
    document.head.append(
      Tag.make("script", {
        src: link
      })
    )
  }

  function addStylesToHead(styles) {
    document.head.append(
      Tag.make("style", {
        textContent: styles
      })
    )
  }

  const sleep = ms => new Promise(res => setTimeout(res, ms))

  function pathNameSplit() {
    return window.location.pathname.split('/')
  }

  function getCompetition() {
    return pathNameSplit()[1]
  }

  function getPageType() {
    return pathNameSplit()[2]
  }

  function urlParams() {
    return new URLSearchParams(location.search)
  }

  function getAction() {
    let params = urlParams()
    return params.get("action")
  }

  function isCpManagement() {
    return getPageType() === "cp_mgmt"
  }

  function isRouteManagement() {
    return getPageType() === "route_mgmt"
  }

  function isCpListPage() {
    return isCpManagement() && getAction() === null
  }

  function isCpEditPage() {
    return isCpManagement() && getAction() === "edit"
  }

  function isCpDeletePage() {
    return isCpManagement() && getAction() === "delete"
  }

  function isRouteListPage() {
    return isRouteManagement() && getAction() === null
  }

  function isRouteBuildPage() {
    return isRouteManagement() && getAction() === "build2"
  }

  function isRouteEditPage() {
    return isRouteManagement() && getAction() === "edit"
  }

  function isRouteStagesPage() {
    return isRouteManagement() && getAction() === "stages"
  }

  function isRouteMapPage() {
    return isRouteManagement() && getAction() === "map"
  }

  function cpLink(cpId) {
    return `/${getCompetition()}/cp_mgmt/?action=edit&cp_id=${cpId}`
  }

  async function updatePoint(formData) {
    return await fetch(`https://${adminSite}/api/competitions/${getCompetition()}/update`, {
      method: 'POST',
      body: formData
    })
  }

  async function deletePoint() {
    return await fetch(`https://${adminSite}/api/points/${urlParams().get("cp_id")}`, {
      method: 'DELETE'
    })
  }

  async function getPointsByField(field, values) {
    let params = new URLSearchParams()
    for (const value of values) {
      params.append(`${field}[]`, value)
    }
    let response = await fetch(`https://${adminSite}/api/competitions/${getCompetition()}/points?${params.toString()}`, {
      method: 'GET'
    })
    return await response.json()
  }

  async function updateRoute(formData) {
    return await fetch(`https://${adminSite}/api/competitions/${getCompetition()}/route`, {
      method: 'POST',
      body: formData
    })
  }

  async function getRoutes() {
    let response = await fetch(`https://${adminSite}/api/points/${urlParams().get("cp_id")}/routes`, {
      method: 'GET'
    })
    return await response.json()
  }

  async function getUnknownPoints() {
    let response = await fetch(`https://${adminSite}/api/competitions/${getCompetition()}/routes/${urlParams().get("cat_id")}/unknown`, {
      method: 'GET'
    })
    return await response.json()
  }

  function relatedPointsHref() {
    return `https://${adminSite}/competitions/${getCompetition()}/relations`
  }

  function createPseudoButton(button, callback, clickAfter = true) {
    let pseudoButton = Tag.button({
      type: "button",
      textContent: button.value,
      classes: [...button.classList].join(" ") + " pseudo-save",
      on: {
        click: async () => {
          await callback()
          if (clickAfter) {
            button.click()
          }
        }
      }
    })

    button.style.display = "none"
    button.parentElement.insertBefore(pseudoButton, button)
  }

  function copyCoordinates() {
    return Tag.button({
      classes: "copy-button",
      on: {
        click: async event => {
          event.preventDefault()
          let lat = document.querySelector(`input[name="cp[lattitude]"]`).value
          let lon = document.querySelector(`input[name="cp[longitude]"]`).value

          const text = new Blob([`${lat}, ${lon}`], { type: "text/plain" })
          const data = new ClipboardItem({ "text/plain": text })
          await navigator.clipboard.write([data])
        }
      },
      children: [
        Tag.img({
          src: "https://upload.wikimedia.org/wikipedia/commons/a/aa/Bw_copy_icon_320x320.svg"
        })
      ]
    })
  }

  function yandexMaps(lat, lon, zoom) {
    return `https://yandex.ru/maps/?ll=${lon}%2C${lat}&z=${zoom}`
  }

  function googleMaps(lat, lon, meters) {
    return `https://www.google.ru/maps/@${lat},${lon},${meters}m`
  }

  function pastvu(lat, lon, zoom) {
    return `https://pastvu.com/?g=${lat},${lon}&z=${zoom}&s=yandex&t=scheme&type=1`
  }

  function twoGis(lat, lon, zoom) {
    return `https://2gis.ru?m=${lon}%2C${lat}%2F${zoom}`
  }

  function makeRef(linkCallback, iconSrc, zoom) {
    return Tag.a({
      href: linkCallback(lat, lon, zoom),
      target: "_blank",
      on: {
        click: function (e) {
          let lat = document.querySelector(`input[name="cp[lattitude]"]`).value
          let lon = document.querySelector(`input[name="cp[longitude]"]`).value
          this.href = linkCallback(lat, lon, zoom)
        }
      },
      children: [
        Tag.img({
          src: iconSrc,
          classes: "map-icon"
        })
      ]
    })
  }

  let makeCoordinatesLinks = (function () {
    let executed = false
    return function () {
      if (executed) return
      executed = true

      let linksContainer = Tag.div({
        children: [
          copyCoordinates(),
          makeRef(yandexMaps, "https://upload.wikimedia.org/wikipedia/commons/7/72/Yandex_Maps_icon.svg", ZOOM),
          makeRef(googleMaps, "https://upload.wikimedia.org/wikipedia/commons/a/aa/Google_Maps_icon_%282020%29.svg", METERS),
          makeRef(pastvu, "https://pastvu.com/coast-icon.png", ZOOM),
          makeRef(twoGis, "https://d-assets.2gis.ru/favicon.png", ZOOM)
        ]
      })

      document.querySelector(`div:has(> div > input[name="cp[longitude]"])`).after(linksContainer)
    }
  })()

  function makeDownloadLink(name, href = null) {
    return Tag.a({
      download: name,
      textContent: "Скачать",
      target: "_blank",
      href: href
    })
  }

  async function sendFileDeleted(inputName, fileInputVariant = null) {
    let formData = new FormData()

    let normalInputName = fileInputVariant !== null ? fileInputs[inputName][fileInputVariant] : fileInputs[inputName]
    formData.set(`cp[id]`, document.querySelector("input[name='cp[id]']").value)
    formData.set(`cp[${normalInputName}]`, "-1")
    let result = await updatePoint(formData)
    try {
      console.log(result.ok)
    }
    catch (e) {
      console.log(e)
    }
  }

  function parsePurpose(cell) {
    return cell?.querySelector("td:nth-child(2)")?.innerHTML.match(/для[^\|]+/g)?.[0].trim()
  }

  function parsefileInputVariant(cell) {
    return Object.keys(fileInpitVariants).find(key => fileInpitVariants[key] === parsePurpose(cell))
  }

  function addDisplayedFile(inputName, type, file, preview, removeLink, removedFilesLinks, variant = null) {
    if (!inputName || !file || !removeLink) return

    let fileListContainer
    if (variant !== null) {
      if (inputName == "attachment1" || inputName == "attachment4") {
        if (variant == "l") {
          fileListContainer = document.querySelector(`[data-container="${fileInputs[inputName][variant]}"] .file-list-container`)
        }
        else if (variant == "h") {
          let filesContainer = document.querySelector(`[data-container="${fileInputs[inputName].h}"]`)
          fileListContainer = getFileListContainer(filesContainer)
        }
      }
    }

    let input = document.querySelector(`input[name="${inputName}\[\]"]`)
    fileListContainer ??= input.parentElement.parentElement.querySelector(".file-list-container")
    let fileContainer = Tag.div({
      classes: "file-container"
    })

    const displayedFile = Tag.make(type ?? "a", {
      classes: "preview-small"
    })

    if (type == "img") {
      displayedFile.src = preview
      displayedFile.dataset.origin = file
      displayedFile.addEventListener("click", () => {
        let dialog = document.querySelector("dialog")
        makeSwiper(dialog, fileListContainer, file)
        dialog.showModal()
      })
    }
    else if (type == "audio") {
      displayedFile.src = file
      displayedFile.setAttribute("controls", "")
    }
    else {
      displayedFile.href = file
      displayedFile.textContent = "Файл"
    }

    let fileButtonsContainer = Tag.div({
      classes: "file-buttons-container"
    })

    fileButtonsContainer.append(makeDownloadLink(/[^/]*$/.exec(new URL(file).pathname)[0], file))

    let deleteButton = Tag.button({
      type: "button",
      classes: "button-delete",
      textContent: "x",
      on: {
        click: async e => {
          if (!confirm("Точно?")) return

          await fetch(removeLink)
          await sendFileDeleted(inputName, variant)
          removedFilesLinks.push(removeLink)

          fileListContainer.removeChild(fileContainer)
        }
      }
    })

    fileButtonsContainer.append(deleteButton)

    fileContainer.append(displayedFile)
    fileContainer.append(fileButtonsContainer)

    fileListContainer.append(fileContainer)
  }

  function dataURLtoFile(dataurl, filename) {
    let arr = dataurl.split(',')
    let mime = arr[0].match(/:(.*?);/)[1]
    let bstr = atob(arr[arr.length - 1])
    let n = bstr.length
    let u8arr = new Uint8Array(n)
    while (n--) {
      u8arr[n] = bstr.charCodeAt(n)
    }
    return new File([u8arr], filename, { type: mime })
  }

  function getHtmlElementByFileType(file) {
    const htmlElementsForTypes = {
      "image/": "img",
      "audio/": "audio"
    }
    let res = "div"
    for (const [type, el] of Object.entries(htmlElementsForTypes)) {
      if (file.type.startsWith(type)) {
        res = el
        break
      }
    }
    let el = Tag.make(res)
    if (res == "audio") {
      el.setAttribute("controls", "")
    }

    return el
  }

  function getAttachmentIndex(input) {
    return input.name.replace(/\D/g, '')
  }

  function getContainersByVariant(variant, attachmentIndex) {
    let invertedVariant = variant == "l" ? "h" : "l"
    let variants = fileInputs[`attachment${attachmentIndex}`]

    return [document.querySelector(`[data-container="${variants[invertedVariant]}"]`), document.querySelector(`[data-container="${variants[variant]}"]`)]
  }

  function makeFileListContainer() {
    return Tag.div({
      classes: "file-list-container"
    })
  }
  
  function getFileListContainer(filesContainer) {
    let fileListContainer = filesContainer.querySelector(".file-list-container")
    if (fileListContainer == null) {
      fileListContainer = makeFileListContainer()
      filesContainer.append(fileListContainer)
    }

    return fileListContainer
  }

  function makeHandleFilesFunc() {
    let storedFiles = []

    return function () {
      let dt = new DataTransfer()

      if (storedFiles.length) {
        for (const oldFile of storedFiles) {
          let oldFileObj = dataURLtoFile(oldFile.data, oldFile.name)
          dt.items.add(oldFileObj)
        }
      }

      for (let i = 0; i < this.files.length; i++) {
        const file = this.files[i]
        dt.items.add(file)

        let variant = this.parentElement.querySelector('input[type="radio"]:checked')?.value
        let fileListContainer

        if (variant) {
          let [_, filesContainer] = getContainersByVariant(variant, getAttachmentIndex(this))
          fileListContainer = getFileListContainer(filesContainer)
        }
        else {
          fileListContainer = this.parentElement.parentElement.querySelector(".file-list-container")
        }

        let fileContainer = Tag.div({
          classes: "file-container file-container-new"
        })

        const displayedFile = getHtmlElementByFileType(file)
        displayedFile.classList.add("preview")
        displayedFile.file = file
        displayedFile.addEventListener("click", () => {
          let dialog = document.querySelector("dialog")
          makeSwiper(dialog, fileListContainer, displayedFile.src)
          dialog.showModal()
        })

        let fileButtonsContainer = Tag.div({
          classes: "file-buttons-container"
        })

        let downloadLink = makeDownloadLink(file.name)
        fileButtonsContainer.append(downloadLink)

        let deleteButton = Tag.button({
          classes: "button-delete",
          textContent: "x",
          type: "button",
          on: {
            click: e => {
              let index = [...fileContainer.parentElement.children].indexOf(fileContainer)

              if (!confirm("Точно?")) return

              storedFiles.splice(storedFiles.length - this.files.length + index, 1)

              let rdt = new DataTransfer()
              for (const file of this.files) {
                rdt.items.add(file)
              }
              rdt.items.remove(index)
              this.files = rdt.files

              fileContainer.parentElement.removeChild(fileContainer)
            }
          }
        })

        fileButtonsContainer.append(deleteButton)

        fileContainer.append(displayedFile)
        fileContainer.append(fileButtonsContainer)

        fileListContainer.append(fileContainer)

        const reader = new FileReader()
        reader.onload = (e) => {
          displayedFile.src = e.target.result
          downloadLink.setAttribute("href", displayedFile.src)

          storedFiles.push({ data: displayedFile.src, name: file.name })
        }
        reader.readAsDataURL(file)
      }

      this.files = dt.files
    }
  }

  let prettifyFiles = (function () {
    let executed = false
    return function (insertedFileRows, removedFilesLinks) {
      if (executed) return
      executed = true

      document.querySelector("#new").querySelectorAll("input[type=file]").forEach((element, index) => {
        element.id = `input-file-${index}`
        element.setAttribute("multiple", "multiple")
        element.dataset.index = index
        let pseudoInput = Tag.make("label", {
          classes: "custom-file-upload",
          _for: element.id,
          textContent: "+"
        })
        element.parentElement.insertBefore(pseudoInput, element)

        element.parentElement.parentElement.append(makeFileListContainer())

        element.addEventListener("change", makeHandleFilesFunc(), false)
      })

      moveFileInputValues(insertedFileRows, removedFilesLinks)
    }
  })()

  function saveInputValues(from) {
    let fromInputs = [...from.querySelectorAll("input, textarea")]
    let result = new Map()

    for (const fromEl of fromInputs) {
      result.set(fromEl, fromEl.type == "radio" ? fromEl.checked : fromEl.value)
    }

    return result
  }

  function setInputValues(to, values) {
    let toInputs = [...to.querySelectorAll("input, textarea")]

    for (const [fromEl, value] of values) {
      let currentToInput = toInputs.find(toEl =>
        toEl.type !== 'file' &&
        toEl.type === fromEl.type &&
        toEl.name === fromEl.name &&
        toEl.id === fromEl.id &&
        (toEl.type !== "radio" || toEl.value === fromEl.value)
      )

      if (currentToInput) {
        currentToInput.checked = fromEl.checked
        currentToInput.value = fromEl.value
      }
    }
  }

  function moveFileInputValues(insertedFileRows, removedFilesLinks) {
    for (const [inputRowIndex, rows] of insertedFileRows) {
      for (const row of rows) {
        let fileInput = null, type = null, fileUrl = null, deleteUrl = null, preview = null
        if (row.querySelector("img")) {
          type = "img"
          fileUrl = row.querySelector("td:first-child a").href
          preview = row.querySelector("img").src
        }
        else if (row.querySelector("audio")) {
          type = "audio"
          fileUrl = row.querySelector("audio").src
        }
        else {
          fileUrl = row.querySelector("td:first-child a").href
        }

        deleteUrl = row.querySelector("td:nth-child(2) a").href

        if (inputRowIndex == 23) {
          fileInput = "attachment1"
        }
        else if (inputRowIndex == 27) {
          fileInput = "attachment4"
        }
        else if (inputRowIndex == 31) {
          fileInput = "attachment2"
        }
        else if (inputRowIndex == 35) {
          fileInput = "attachment3"
        }
        else {
          console.log(inputRowIndex)
          return
        }

        let variant = parsefileInputVariant(row)

        addDisplayedFile(fileInput, type, fileUrl, preview, deleteUrl, removedFilesLinks, variant)
      }
    }
  }

  function moveNewFilesOnVariantChange() {
    document.querySelectorAll(`input[name^="new_file_type"]`).forEach(el => el.addEventListener("change", function () {
      let attachmentIndex = getAttachmentIndex(this)

      let [from, to] = getContainersByVariant(this.value, attachmentIndex)
      let newFiles = [...from.querySelectorAll(`.file-container-new`)]
      for (const newFile of newFiles) {
        getFileListContainer(to).append(newFile)
      }
    }))
  }

  function removeDeletedFilesFromUglyVersion(removeLink) {
    let shortHref = removeLink.split("/").at(-1)
    document.querySelector(`tr:has(a[href="${shortHref}"])`).remove()
  }

  function moveInputValues(form, was, became) {
    let inputValues = saveInputValues(was)
    form.removeChild(was)
    form.append(became)
    setInputValues(became, inputValues)
    createSendButtons()
  }

  function prettifyEditCpForm(form, was, became, insertedFileRows, removedFilesLinks) {
    moveInputValues(form, was, became)
    prettifyFiles(insertedFileRows, removedFilesLinks)

    moveNewFilesOnVariantChange()
    $("#cps_main").select2()
    makeCoordinatesLinks()
  }

  function uglifyEditCpForm(form, was, became, removedFilesLinks) {
    moveInputValues(form, was, became)
    for (const removedFileLink of removedFilesLinks) {
      removeDeletedFilesFromUglyVersion(removedFileLink)
    }
    removedFilesLinks.length = 0
  }

  function createAlwaysPrettifyInput(index) {
    let alwaysPrettify = Tag.div({
      classes: "always-prettify-container"
    })

    let alwaysPrettifyLabel = Tag.make("label", {
      _for: "always-prettify-" + index,
      textContent: "Всегда"
    })

    alwaysPrettify.append(alwaysPrettifyLabel)

    let alwaysPrettifyCheckbox = Tag.input({
      type: "checkbox",
      id: "always-prettify-" + index,
      name: "always-prettify-" + index,
      on: {
        change: function () {
          let otherCheckboxes = document.querySelectorAll(`input[name^="always-prettify-"]`)
          for (let checkbox of otherCheckboxes) {
            if (checkbox.id !== this.id) {
              checkbox.checked = this.checked
            }
          }
          if (this.checked)
            localStorage.removeItem(localStorageItems.NOT_PRETTIFY_EDIT_CP)
          else
            localStorage.setItem(localStorageItems.NOT_PRETTIFY_EDIT_CP, "+")
        }
      }
    })

    alwaysPrettify.append(alwaysPrettifyCheckbox)

    return alwaysPrettify
  }

  function hide(elList) {
    elList.forEach(el => el.classList.add("hidden"))
  }

  function show(elList) {
    elList.forEach(el => el.classList.remove("hidden"))
  }

  function toggleRows(rows, from, to, cycle, showIndex) {
    for (let i = from; i < to; i += cycle) {
      for (let j = 0; j < cycle; j++) {
        if (!showIndex.includes(j)) {
          rows[i + j]?.classList.toggle("collapsed")
        }
      }
    }
  }

  function toggleText(current, ...variants) {
    let index = variants.indexOf(current)
    if (index == -1) return null

    return variants[index + 1 < variants.length ? index + 1 : 0]
  }

  function makeSwiper(dialog, fileListContainer, src) {
    let files = [...fileListContainer.querySelectorAll(`:is(.preview, .preview-small)`)]

    let swiperWrapper = Tag.div({
      classes: "swiper-wrapper"
    })

    for (const file of files) {
      let swiperSlide = Tag.div({
        classes: "swiper-slide",
        children: [
          Tag.a({
            classes: "swiper-download-link",
            href: file.dataset.origin,
            text: "Скачать",
            target: "_blank"
          }),
          Tag.img({
            src: file.dataset.origin
          })
        ]
      })

      swiperWrapper.append(swiperSlide)
    }

    let swiperDiv = Tag.div({
      classes: "swiper",
      children: [
        swiperWrapper,
        Tag.div({
          classes: "swiper-button-prev"
        }),
        Tag.div({
          classes: "swiper-button-next"
        })
      ]
    })

    dialog.append(swiperDiv)

    new Swiper('.swiper', {
      initialSlide: files.findIndex(el => el.dataset.origin && el.dataset.origin === src || el.src === src),

      navigation: {
        nextEl: '.swiper-button-next',
        prevEl: '.swiper-button-prev',
      },
    })

    bindArrowsForGallery('.swiper-button-prev', '.swiper-button-next')
  }

  const bindArrowsForGallery = (function () {
    let executed = false
    return function(leftButtonQuery, rightButtonQuery) {
      if (executed) return
      executed = true

      document.addEventListener("keydown", function (e) {
        if (!document.querySelector("dialog").open) return
        switch (e.key) {
          case "ArrowLeft":
            document.querySelector(leftButtonQuery).click()
            break

          case "ArrowRight":
            document.querySelector(rightButtonQuery).click()
            break

          default: return
        }
        e.preventDefault()
      })
    }
  })()

  async function sendForm() {
    let formData = new FormData(document.querySelector("form"))

    let fileInputNames = Object.values(fileInputs).reduce((res, el) => typeof el == "string" ? [...res, el] : [...res, ...Object.values(el)], [])
    for (const fileInput of fileInputNames) {
      let fileListContainer = document.querySelector(`[data-container=${fileInput}] .file-list-container`)
      if (fileListContainer) {
        let count = [...fileListContainer.children].filter(el => !el.classList.contains("file-container-new")).length
        formData.set(`cp[${fileInput}]`, count)
      }
    }

    if (formData.get("cp[id]") == '') {
      localStorage.setItem(localStorageItems.NEED_UPDATE_ID, true)
    }

    let props = document.querySelectorAll("input[name^=\"prop_\"]")
    for (let prop of props) {
      let parent = prop.parentElement
      let propName

      if (parent.tagName === "DIV") {
        propName = parent.parentElement.querySelector("label").textContent
      }
      else if (parent.tagName === "TD") {
        propName = parent.parentElement.querySelector("td:first-child").textContent
      }

      formData.set(`propname[${prop.name}]`, propName)
    }

    let result = await updatePoint(formData)

    try {
      console.log(result.ok)
    }
    catch (e) {
      console.log(e)
    }
  }

  function createSaveAndNewButton() {
    let saveAndNewButton = Tag.button({
      textContent: "+",
      type: "button",
      classes: "safe-action",
      on: {
        click: async () => {
          await sendForm()
          
          let cpNumber = document.querySelector(`input[name="cp[number]"]`).value
          let lattitude = document.querySelector(`input[name="cp[lattitude]"]`).value
          let longitude = document.querySelector(`input[name="cp[longitude]"]`).value

          localStorage.setItem(localStorageItems.JUST_CREATED, cpNumber)
          localStorage.setItem(localStorageItems.LATTITUDE, lattitude)
          localStorage.setItem(localStorageItems.LONGITUDE, longitude)

          document.querySelector(`input[name="save_go"]`).click()
        }
      }
    })

    return saveAndNewButton
  }

  function createSendButtons() {
    if (document.querySelector(".pseudo-save")) return

    let saveAndStayButtons = document.querySelectorAll(`input[name="save_go"]`)
    let saveAndExitButtons = document.querySelectorAll(`input[name="save_exit"]`)
    let saveButtons = [...saveAndStayButtons, ...saveAndExitButtons]
    let cpIsKnown = document.querySelector(`input[name="cp[id]"]`).value != ''

    for (let saveButton of saveButtons) {
      createPseudoButton(saveButton, async () => {
        await sendForm()

        if (cpIsKnown) return

        if (saveButton.name == "save_exit") {
          localStorage.setItem(localStorageItems.REDIRECT_EXIT, true)
        }

        saveAndStayButtons[0].click()
      }, cpIsKnown)
    }

    saveAndStayButtons.forEach(el => el.after(createSaveAndNewButton()))
  }

  function redirectAfterNewCpIfNeeded() {
    let justCreated = localStorage.getItem(localStorageItems.JUST_CREATED)
    let prevCreated = localStorage.getItem(localStorageItems.PREV_CREATED)
    let needUpdateId = localStorage.getItem(localStorageItems.NEED_UPDATE_ID)
    let exit = localStorage.getItem(localStorageItems.REDIRECT_EXIT)

    if (needUpdateId) {
      let formData = new FormData()
      formData.set("cp[id]", urlParams().get("cp_id"))
      formData.set("cp[number]", document.querySelector(`input[name="cp[number]"]`).value)
      formData.set("update_id", true)
      updatePoint(formData).then(() => localStorage.removeItem(localStorageItems.NEED_UPDATE_ID))
    }

    if (exit) {
      localStorage.removeItem(localStorageItems.REDIRECT_EXIT)

      location.href = location.href.replace(location.search, '')
      return
    }

    if (justCreated !== null) {
      localStorage.setItem(localStorageItems.PREV_CREATED, justCreated)
      localStorage.removeItem(localStorageItems.JUST_CREATED)

      location.href = location.href.replace(location.search, "?action=edit")
      return
    }
    if (prevCreated !== null) {
      document.querySelector(`input[name="cp[number]"]`).value = parseInt(localStorage.getItem(localStorageItems.PREV_CREATED)) + 1
      document.querySelector(`input[name="cp[lattitude]"]`).value = localStorage.getItem(localStorageItems.LATTITUDE)
      document.querySelector(`input[name="cp[longitude]"]`).value = localStorage.getItem(localStorageItems.LONGITUDE)

      localStorage.removeItem(localStorageItems.PREV_CREATED)
      localStorage.removeItem(localStorageItems.LATTITUDE)
      localStorage.removeItem(localStorageItems.LONGITUDE)
    }
  }

  function addUglyDeleteListener() {
    let links = [...document.querySelectorAll("a[href*='del_stored']")]

    for (const link of links) {
      let shortHref = link.href.split("/").at(-1)
      let fileInput = document.querySelector(`tr:has(a[href$="${shortHref}"]) ~ tr:has(input[type="file"]) input[type="file"]`)
      let fileInputVariant = parsefileInputVariant(document.querySelector(`tr:has(a[href$="${shortHref}"])`))

      link.addEventListener("click", async e => {
        e.preventDefault()
        if (!confirm("Точно?")) return

        await sendFileDeleted(fileInput.name.split("[")[0], fileInputVariant)
        location.href = link.href
      })
    }
  }

  function createMenuFromLinks(links, blanks = []) {
    let menu = Tag.make("menu")

    for (const [label, href] of Object.entries(links)) {
      let menuItem = Tag.make("li")
      let link = Tag.a({
        href: href,
        innerText: label
      })

      if (blanks.includes(label)) {
        link.target = "_blank"
      }

      menuItem.append(link)
      menu.append(menuItem)
    }

    return menu
  }

  function addStickyMenu() {
    let competition = getCompetition()
    if (!competition) return

    let menuContainer = Tag.make("nav", {
      classes: "sticky-menu"
    })

    let mainLinksFormatted = {}
    for (const [label, href] of Object.entries(links)) {
      mainLinksFormatted[label] = href.startsWith("http") ? href : `/${competition}/${href}`
    }
    menuContainer.append(createMenuFromLinks(mainLinksFormatted, ["Легенда"]))

    let goToContainer = Tag.make("li", {
      classes: "go-to-container",
      children: [
        Tag.input({
          type: "text",
          id: "go-to-cp"
        }),
        Tag.span({
          textContent: "✏️ КП",
          on: {
            click: async function () {
              let number = document.querySelector("#go-to-cp").value
              let json = await getPointsByField("number", [number])
              let id = json[number].cp_id

              if (id == null) {
                alert("Нет КП с таким номером")
                return
              }

              location.href = cpLink(id)
            }
          }
        })
      ]
    })

    ;[...menuContainer.querySelectorAll(`li`)].find(el => el.querySelector(`a`).textContent.trim() == "Контрольные пункты")?.after(goToContainer)

    let catId = urlParams().get("cat_id")
    if (catId !== null) {
      let catLinksFormatted = {}
      for (const [label, href] of Object.entries(catLinks)) {
        catLinksFormatted[label] = href(catId)
      }

      menuContainer.append(createMenuFromLinks(catLinksFormatted))
    }

    document.querySelector("#header").after(menuContainer)

    addStylesToHead(`
      .sticky-menu {
        position: sticky;
        top: 0;
        z-index: 9999;
      
        menu {
          display: flex;
          gap: .3rem;
          background: white;
          font-size: 1.2rem;
          padding-left: 10px;

          li {
            list-style-type: none;

            &:not(:last-child)::after {
              content: " | "
            }
          }
        }

        .go-to-container {
          input {
            width: 5em;
          }

          span {
            margin-left: .5em;
            cursor: pointer;
          }
        }
      }
    `)
  }

  function addClearBoth() {
    let clearEl = Tag.div()
    clearEl.style.clear = "both"
    document.body.append(clearEl)
  }

  function hasMap() {
    return document.querySelector(".leaflet-container") !== null
  }

  function initMapbox() {
    addJs('https://api.mapbox.com/mapbox.js/plugins/leaflet-fullscreen/v1.0.1/Leaflet.fullscreen.min.js')
    addCss('https://api.mapbox.com/mapbox.js/plugins/leaflet-fullscreen/v1.0.1/leaflet.fullscreen.css')
  }

  function addFullscreenButton() {
    $(async () => {
      if (document.querySelector("#map") !== null) {
        while (L.Control.Fullscreen === undefined) {
          await sleep(500)
        }
        map.addControl(new L.Control.Fullscreen())
      }
    })
  }

  function addStageLink() {
    let linkCells = [...document.querySelectorAll("#content table td:nth-child(2)")]
    for (const linkCell of linkCells) {
      let href = linkCell.querySelector("a").href
      let catId = new URLSearchParams(href).get("cat_id")
      let constructorLink = linkCell.querySelector(`a[href="${catLinks["Конструктор маршрута"](catId).replace("route_mgmt/", "")}"]`)

      if (constructorLink) {
        let innerText = "Этапы"
        let stagesLink = Tag.a({
          innerText: innerText,
          href: catLinks[innerText](catId)
        })

        constructorLink.after(stagesLink)
        constructorLink.after(". ")
      }
    }
  }

  function hideNonTesters() {
    if (urlParams().get("test") == null) return

    document.querySelectorAll(`td:nth-child(2)`).forEach(el => {
      let innerHTML = el.innerHTML
      let from = innerHTML.indexOf("Отчёты тестировщиков")
      if (from == -1)
        from = innerHTML.length

      el.innerHTML = innerHTML.substring(from)
    })
  }

  function getRows() {
    return [...document.querySelectorAll('#props > tbody > tr')]
  }

  function getInsertedFileRows(rows) {
    let insertedFileRows = new Map()
    let i = 0
    while (i < rows.length) {
      if ([23, 27, 31, 35].includes(i) && rows[i].querySelector(`input[type="file"]`) == null) {
        if (!insertedFileRows.has(i)) {
          insertedFileRows.set(i, [rows[i]])
        }
        else {
          let savedRows = insertedFileRows.get(i)
          savedRows.push(rows[i])
          insertedFileRows.set(i, savedRows)
        }
        rows.splice(i, 1)
      }
      else {
        i++
      }
    }

    return insertedFileRows
  }

  function makeContainer() {
    return Tag.div({
      id: "new"
    })
  }

  function makeHeader(rows) {
    let headerContainer = createFrom(rows, "header", [
      { index: 2, name: "№", desc: "" },
      { index: 8, name: "Название" }
    ])

    let copyLink = Tag.div({
      innerHTML: new Property(rows[2]).desc
    })
    headerContainer.append(copyLink)

    return headerContainer
  }

  function bindDeleteButton() {
    createPseudoButton(document.querySelector(`input[name="delete_go"]`), deletePoint)
  }

  function makeTopAndBottomButtons(rows, form, oldTable, container, insertedFileRows, removedFilesLinks) {
    let topButtonsContainer = createFrom(rows, "buttons", [
      { index: 0 }
    ])

    let bottomButtonsContainer = topButtonsContainer.cloneNode(true)

    document.querySelectorAll("#props tr:is(:first-child, :last-child) th").forEach((el, index) => {
      let rowContentWrapper = Tag.div({
        classes: "buttons-row__content-wrapper"
      })
      while ([...el.children].length > 0) {
        let child = el.firstChild
        rowContentWrapper.append(child)
      }

      let prettifyButton = Tag.button({
        type: "button",
        textContent: "Сделать красиво",
        on: {
          click: () => {
            prettifyEditCpForm(form, oldTable, container, insertedFileRows, removedFilesLinks)
          }
        }
      })

      rowContentWrapper.append(prettifyButton)
      rowContentWrapper.append(createAlwaysPrettifyInput(index))

      el.append(rowContentWrapper)
    })

    ;[topButtonsContainer, bottomButtonsContainer].forEach((el, index) => {
      let unglifyButton = Tag.button({
        type: "button",
        textContent: "Сделать как было",
        on: {
          "click": () => {
            uglifyEditCpForm(form, container, oldTable, removedFilesLinks)
          }
        }
      })

      let topRowContentWrapper = el.querySelector("div > div > div")
      topRowContentWrapper.append(unglifyButton)
      topRowContentWrapper.append(createAlwaysPrettifyInput(index))
    })

    return [topButtonsContainer, bottomButtonsContainer]
  }

  function makeTopOptions(rows) {
    return createFrom(rows, "options", [
      { index: 3 },
      { index: 4, name: "КП-загадка" },
      { index: 16, name: "Этапник" },
      { index: 7, name: "Пиктограмма" },
      { index: 12, name: "Нужна ИС" },
      { index: 17, name: "Спрятать ИС" },
      { index: 14, name: "Основной КП" },
      { index: 10, name: "Широта" },
      { index: 11, name: "Долгота" },
      { index: 5, name: "Старт" },
      { index: 6, name: "Финиш" },
      { index: 15, name: "Знак" }
    ])
  }

  function makeComment(rows) {
    return createFrom(rows, "comment", [
      { index: 9, desc: "" }
    ])
  }

  function makeLegend(container, rows) {
    let legendContainer = Tag.div({
      classes: "legend-container"
    })

    /* LEGEND DESC */

    const LEGEND_RU_LABEL = "Русский"
    const LEGEND_EN_LABEL = "Английский"

    let legendDescHeader = Tag.div({
      classes: "legend-container__desc-header"
    })

    let legendLang = Tag.div({
      classes: "legend-desc__lang",
      textContent: LEGEND_RU_LABEL
    })
    legendDescHeader.append(legendLang)

    let legendEnSwitchContainer = createFrom(rows, "legend-switch-container", [
      { index: 48 }
    ])
    legendEnSwitchContainer.addEventListener("click", event => {
      hide([legendRuDescContainer, legendRuHiddenDescContainer, legendEnSwitchContainer])
      show([legendEnDescContainer, legendEnHiddenDescContainer, legendRuSwitchContainer])
      legendLang.textContent = LEGEND_EN_LABEL
    })
    legendDescHeader.append(legendEnSwitchContainer)

    let copyDescButton = Tag.button({
      type: "button",
      textContent: "Копировать",
      on: {
        click: () => {
          let ruInputs = [...container.querySelectorAll(":is(input, textarea)[name^=\"cp_strings\[ru\]\"]")]
          let enInputs = [...container.querySelectorAll(":is(input, textarea)[name^=\"cp_strings\[en\]\"]")]
          for (const [i, enInput] of enInputs.entries()) {
            enInput.value = ruInputs[i].value
          }
        }
      }
    })

    let legendRuSwitchContainer = createFrom(rows, "legend-switch-container hidden", [
      { index: 39 }
    ])
    legendRuSwitchContainer.addEventListener("click", event => {
      hide([legendEnDescContainer, legendEnHiddenDescContainer, legendRuSwitchContainer])
      show([legendRuDescContainer, legendRuHiddenDescContainer, legendEnSwitchContainer])
      legendLang.textContent = LEGEND_RU_LABEL
    })
    legendDescHeader.append(legendRuSwitchContainer)

    legendDescHeader.append(copyDescButton)

    let legendRuDescContainer = createFrom(rows, "legend-desc", [
      { index: 40, desc: "" },
      { index: 41, desc: "" },
      { index: 42, desc: "" },
      { index: 43, desc: "" }
    ])

    let legendRuHiddenDescContainer = createFrom(rows, "legend-desc collapsible collapsed", [
      { index: 44, desc: "" },
      { index: 45, desc: "" },
      { index: 46, desc: "" },
      { index: 47, desc: "" },
    ])

    let legendEnDescContainer = createFrom(rows, "legend-desc hidden", [
      { index: 49, desc: "" },
      { index: 50, desc: "" },
      { index: 51, desc: "" },
      { index: 52, desc: "" }
    ])

    let legendEnHiddenDescContainer = createFrom(rows, "legend-desc collapsible collapsed hidden", [
      { index: 53, desc: "" },
      { index: 54, desc: "" },
      { index: 55, desc: "" },
      { index: 56, desc: "" }
    ])

    let hider = Tag.div({
      children: [
        Tag.button({
          classes: "collapse-button",
          type: "button",
          on: {
            click: event => {
              let container = document.querySelector('.legend-container')
              container.querySelectorAll(".legend-desc.collapsible").forEach(element => {
                element.classList.toggle("collapsed")
              })
            }
          }
        })
      ]
    })

    let legendDescContainer = Tag.div({
      classes: "legend-container__desc",
      children: [
        legendDescHeader,
        legendRuDescContainer,
        legendRuHiddenDescContainer,
        legendEnDescContainer,
        legendEnHiddenDescContainer,
        hider
      ]
    })

    legendContainer.append(legendDescContainer)

    return legendContainer
  }

  function chooseLegendAsDefaultVariant() {
    document.querySelector(`#props input[name="new_file_type1"][value="l"]`).click()
    document.querySelector(`#props input[name="new_file_type4"][value="l"]`).click()
  }

  function makeLegendFiles(rows) {
    let legendFilesContainer = Tag.div({
      classes: "legend-container__files"
    })

    let imagesForLegendContainer = createFrom(rows, "files-container", [
      { index: 23, name: "Фото в легенде" }
    ])
    imagesForLegendContainer.dataset.container = fileInputs.attachment1.l
    legendFilesContainer.append(imagesForLegendContainer)

    let imagesForHistoryContainer = createFrom(rows, "files-container", [
      { name: "Фото для ИС" }
    ])
    imagesForHistoryContainer.dataset.container = fileInputs.attachment1.h
    legendFilesContainer.append(imagesForHistoryContainer)

    let audioForLegendContainer = createFrom(rows, "files-container", [
      { index: 27, name: "Файлы в легенде" }
    ])
    audioForLegendContainer.dataset.container = fileInputs.attachment4.l
    legendFilesContainer.append(audioForLegendContainer)

    let audioForHistoryContainer = createFrom(rows, "files-container", [
      { name: "Файлы для ИС" }
    ])
    audioForHistoryContainer.dataset.container = fileInputs.attachment4.h
    legendFilesContainer.append(audioForHistoryContainer)

    return legendFilesContainer
  }

  function makeAdminFiles(rows) {
    let adminFilesContainer = Tag.div({
      classes: 'admin-files-container'
    })

    let imagesForAdminContainer = createFrom(rows, "files-container admin-photo-container", [
      { index: 31, name: "Фото в админке" }
    ])
    imagesForAdminContainer.dataset.container = fileInputs.attachment2
    adminFilesContainer.append(imagesForAdminContainer)

    let audioForAdminContainer = createFrom(rows, "files-container admin-files-container", [
      { index: 35, name: "Файлы в админке" }
    ])
    audioForAdminContainer.dataset.container = fileInputs.attachment3
    adminFilesContainer.append(audioForAdminContainer)

    return adminFilesContainer
  }

  function makeBottomOptions(rows) {
    return createFromMulti(rows, "options bottom-options", {
      from: 59,
      to: rows.length - 3
    })
  }

  function makeDialog() {
    let dialog = Tag.make("dialog", {
      id: "dialog",
      on: {
        click: e => {
          if (e.target == dialog) {
            e.target.close()
          }
        },
        close: e => {
          e.target.innerHTML = ""
        }
      }
    })
    document.body.append(dialog)
  }

  function formatMap() {
    if (map !== undefined && L !== undefined) {
      let content = document.querySelector("#content")
      let contentWrapper = Tag.div({
        id: "content-wrapper",
        children: [
          document.querySelector("form"),
          document.querySelector("#map-wrapper")
        ]
      })
      content.append(contentWrapper)

      let panToCenter = Tag.button({
        type: "button",
        textContent: "В центр",
        on: {
          click: () => {
            let lat = document.querySelector("input[name=\"cp\[lattitude\]\"").value
            let lon = document.querySelector("input[name=\"cp\[longitude\]\"").value
            map.setView(new L.LatLng(parseFloat(lat), parseFloat(lon)), 16)
          }
        }
      })

      document.querySelector("#map_controls").append(panToCenter)
    }
  }

  function makeCurrentPointHigherOnMap() {
    map.eachLayer(function (layer) {
      if (layer instanceof L.Marker) {
        if (layer.getElement().classList.contains("leaflet-marker-draggable")) {
          layer.setZIndexOffset(9999)
        }
      }
    })
  }

  function addRoutesList() {
    (async () => {
      let routes = await getRoutes()

      if (routes.error || routes.length == 0) return

      let result = Tag.div()
      result.append("Маршруты: ")
      for (const [i, route] of routes.entries()) {
        if (i !== 0) {
          result.append(", ")
        }

        result.append(Tag.a({
          href: `/${getCompetition()}/` + catLinks["Редактор маршрута"](route.cat_id),
          textContent: route.name
        }))
      }

      document.querySelector(`#map_controls`).after(result)
    })()
  }

  function checkIfAlwaysPrettify(form, oldTable, container, insertedFileRows, removedFilesLinks) {
    if (localStorage.getItem(localStorageItems.NOT_PRETTIFY_EDIT_CP) === null) {
      document.querySelector(`input[name^="always-prettify-0"]`).click()
      prettifyEditCpForm(form, oldTable, container, insertedFileRows, removedFilesLinks)
    }
  }

  function addPrettifyEditPageCss() {
    addStylesToHead(`
      #content-wrapper {
        display: flex;
        gap: 20px;
      
        #props {
          width: unset !important;
        }

        form {
          width: 50%;
        }

        #map-wrapper {
          width: 50%;
        }

        #map {
          width: unset !important;
        }

        #new {
          display: flex;
          flex-direction: column;

          & > :not(:first-child) {
            padding-top: 5px;
            border-top: 1px solid #ddd;
          }

          & > :not(:last-child) {
            padding-bottom: 5px;
          }

          .map-icon {
            width: 15px;
            height: 15px;
          }

          .copy-button {
            all: unset;
            cursor: pointer;
          }

          .copy-button img {
            width: 15px;
            heigth: 15px;
          }

          input[type=radio], input[type=checkbox] {
            margin: 0;
          }

          input[type=radio] {
            margin-right: 5px;
          }

          input[type="file"] {
            display: none;
          }

          div > input[type="checkbox"] {
            display: flex;
            align-items: center;
          }

          input#cp_number {
            width: 3em;
          }

          select#cps_main {
            width: 6em;
          }

          input:is(#lat, #lng) {
            width: 5.5em;
          }

          input[name="cp[name_int]"] {
            width: 34em;
          }
        }

        #cps_main {
          width: 150px;
        }

        input[type=submit] {
          width: fit-content;
        }

        .desc-icon-container {
          display: flex;
          align-items: center;
        }

        .desc-icon {
          width: 15px;
          height: 15px;
          cursor: help;
        }

        button, input[type=submit] {
          cursor: pointer;
        }
        
        .buttons-row__content-wrapper {
          display: flex;
          justify-content: center;
          gap: 10px;
          font-weight: normal;
        }

        .custom-file-upload {
          display: block;
          width: fit-content;
          border: 1px solid black;
          padding: 5px 14px;
          border-radius: 5px;
          box-shadow: 0 1px 3px #ddd;
          background: linear-gradient(white, #ddd);
        }

        .buttons > * > * {
          display: flex;
          gap: 10px;
        }

        .always-prettify-container {
          display: flex;
          align-items: center;
          gap: 5px;
        }

        .header > * {
          display: flex;
          gap: 5px;
        }

        :is(.header, .options) input {
          width: unset;
        }
          
        .header {
          display: flex;
          gap: 30px;
          align-items: start;
        }

        .options {
          display: flex;
          flex-wrap: wrap;
          gap: 10px;
        }

        .options > * {
          display: flex;
          gap: 5px;
          align-items: center;
        }

        .legend-container {
          display: flex;
        }

        .legend-container > :not(:last-child) {
          padding-right: 15px;
          border-right: 1px solid #ddd;
        }

        .legend-container > :not(:first-child) {
          padding-left: 15px;
        }

        .legend-container > * {
          display: flex;
          flex-direction: column;
          flex: 0 1 48%;
          gap: 5px;
        }

        .legend-container__desc-header {
          display: flex;
          gap: 10px;
          align-items: center;
          justify-content: end;
        }

        .legend-desc {
          display: flex;
          flex-direction: column;
          gap: 5px;
        }

        .collapsed, .hidden {
          display: none;
        }

        .collapse-button::after {
          content: "Свернуть"
        }

        .collapsed + * > .collapse-button::after {
          content: "Развернуть"
        }
        
        :is(.comment, .legend-desc) > * {
          display: flex;
          gap: 5px;
        }

        :is(.comment, .legend-desc) label {
          display: block;
          width: 10em;
        }

        :is(.comment, .legend-desc) > * > :nth-child(2) {
          flex: 1 1 auto;
        }

        textarea {
          width: 100%;
        }
      }

      .files-container input[type=radio][value="c"] {
        display: none;
      }

      .file-list-container {
        display: flex;
        flex-wrap: wrap;
        gap: 10px;
        margin-top: 5px;
      }

      .file-container {
        display: flex;
        flex-direction: column;
        gap: 5px;
      }

      .file-container img.preview {
        width: 120px;
      }

      .file-container img.preview-small {
        width: 60px;
      }

      .file-container img:is(.preview, .preview-small) {
        cursor: zoom-in;
      }

      dialog .preview,
      .swiper {
        max-width: 1000px;
        width: 100%;
        max-height: 500px;
        height: 100%;
      }

      .swiper-button-next,
      .swiper-button-prev {
        user-select: none;
        -webkit-user-select: none;
      }

      .swiper-wrapper {
        align-items: center;
      }

      .swiper-wrapper > .swiper-slide {
        width: 100%;
        max-height: 500px;
        height: 100%;
      }

      .swiper-slide img {
        width: 100%;
        max-height: 500px;
        height: 100%;
        object-fit: contain;
        object-position: center;
      }

      .swiper-download-link {
        display: block;
        position: absolute;
        top: 0;
        right: 0;
        background: white;
        padding: 5px 10px;
        margin: 5px;
        border: 1px solid #ccc;
        border-radius: 7.5px;
      }

      button.button-delete {
        padding: 0;
        border: 0;
        margin: 0;
        background: none;
        font-size: 18px;
      }

      .admin-files-container {
        display: flex;
      }

      .admin-files-container > :not(:last-child) {
        padding-right: 10px;
        border-right: 1px solid #ddd;
      }

      .admin-files-container > :not(:first-child) {
        padding-left: 10px;
      }

      dialog img.preview {
        width: auto;
      }

      .file-buttons-container {
        display: flex;
        align-items: center;
        gap: 5px;
      }
    `)
  }

  function prettifyEditCpPage() {
    addCss("https://cdn.jsdelivr.net/npm/select2@4.1.0-rc.0/dist/css/select2.min.css")
    addCss("https://cdn.jsdelivr.net/npm/swiper@11/swiper-bundle.min.css")
    addPrettifyEditPageCss()

    /* NEW DEFALUT VALUES */

    chooseLegendAsDefaultVariant()

    /* CONTAINER */

    let form = document.querySelector('form')
    let rows = getRows()
    let insertedFileRows = getInsertedFileRows(rows)
    let removedFilesLinks = []

    let container = makeContainer()
    let oldTable = document.querySelector("#props")

    /* HEADER */

    let headerContainer = makeHeader(rows)

    /* TOP BUTTONS */

    let [topButtonsContainer, bottomButtonsContainer] = makeTopAndBottomButtons(rows, form, oldTable, container, insertedFileRows, removedFilesLinks)

    addUglyDeleteListener()
    createSendButtons()

    /* LEGEND */

    let legendContainer = makeLegend(container, rows)
    legendContainer.append(makeLegendFiles(rows))

    /* APPEND ALL */

    container.append(topButtonsContainer)
    container.append(headerContainer)
    container.append(makeTopOptions(rows))
    container.append(makeComment(rows))
    container.append(legendContainer)
    container.append(makeAdminFiles(rows))
    container.append(makeBottomOptions(rows))
    container.append(bottomButtonsContainer)

    /* MAP */

    formatMap()
    makeCurrentPointHigherOnMap()
    addRoutesList()

    /* DIALOG */

    makeDialog()

    /* PRETTIFY CHECKBOX */

    checkIfAlwaysPrettify(form, oldTable, container, insertedFileRows, removedFilesLinks)
  }

  function isAllPointsSouthOfLat(lat) {
    map.eachLayer(function (layer) {
      if (layer instanceof L.Marker) {
        let latLng = layer.getLatLng()
        if (latLng.lat > lat) {
          return false
        }
      }
    })
  }

  function centerMap() {
    if (map !== undefined && L !== undefined) {
      let coords = mapsCenterByCompetition[getCompetition()]
      if (coords == null) return

      if (isAllPointsSouthOfLat(lat)) {
        map.setView(new L.LatLng(coords.ifSouthern.lat, coords.ifSouthern.lon), 13)
      }
      else {
        map.setView(new L.LatLng(coords.ifEverywhere.lat, coords.ifEverywhere.lon), 12)
      }
    }
  }

  function prettifyRouteBuildPage() {
    function changeColumnWidth() {
      let styles = `
        #content table table {
          td:nth-child(2) {
            width: 10%;

            select {
              width: 5em !important;
            }
          }

          td:nth-child(3) {
            width: 12%;
          }

          td:nth-child(4) {
            width: 12%;

            input {
              width: 4em;
            }
          }
        }
      `
      addStylesToHead(styles)
    }

    function hideDescription() {
      document.querySelectorAll(`#content > form > table > tbody > tr:is(:nth-child(3), :nth-child(6)) `).forEach(el => el.remove())
    }

    function hideStartRow() {
      document.querySelector(`#content table table tr:nth-child(2) `).remove()
    }

    function getRows() {
      return [...document.querySelectorAll(`#content > form > table > tbody > tr:nth-child(3) tr`)]
    }

    function useColspanForFinishWarning() {
      let rows = getRows()
      let finishRow = rows.find(el => el.querySelector(`td`)?.textContent.trim() == "Финиш маршрута")

      finishRow.querySelectorAll(`td:not(:nth-child(2))`).forEach(el => el.remove())
      finishRow.querySelector(`td`).colSpan = 4
    }

    function makeStatsSpan(text) {
      let statsSpan = Tag.span({
        textContent: text,
        classes: "stats"
      })

      return statsSpan
    }

    function getPointCells() {
      return [...document.querySelectorAll(`#props table td:first-child`)]
    }

    function getPointNumberFromCell(cell) {
      return cell.textContent.trim().match(/^\d+/)?.[0]
    }

    function getPointNumbers() {
      let pointNumbers = {}
      for (const [rowIndex, pointCell] of getPointCells().entries()) {
        let pointNumber = getPointNumberFromCell(pointCell)
        if (pointNumber != null)
          pointNumbers[rowIndex] = pointNumber
      }

      return pointNumbers
    }

    async function getPointsFromRows() {
      return await getPointsByField("number", Object.values(getPointNumbers()))
    }

    function countCp(points) {
      let rows = getRows()

      let fullCount = 0
      let stageCount = 0
      let fullPuzzleCount = 0
      let stagePuzzleCount = 0
      let fullNeedHistoryCount = 0
      let stageNeedHistoryCount = 0
      let fullHistoryCount = 0
      let stageHistoryCount = 0

      let stageRow = null
      for (const [i, row] of rows.entries()) {
        if (i < 2) continue
        let isFinish = row.querySelector(".attention") !== null

        if (isFinish || row.querySelector(`th`)) {
          if (stageRow) {
            let stageHeader = stageRow.querySelector("th")
            stageHeader.append(makeStatsSpan(`КП в этапе: ${stageCount}`))
            stageHeader.append(makeStatsSpan(`Загадок: ${stagePuzzleCount}`))
            stageHeader.append(makeStatsSpan(`Нужно ИС: ${stageNeedHistoryCount}`))
            stageHeader.append(makeStatsSpan(`ИС: ${stageHistoryCount}`))
          }

          if (isFinish) break

          stageRow = row
          stageCount = 0
          stagePuzzleCount = 0
          stageNeedHistoryCount = 0
          stageHistoryCount = 0
          continue
        }

        let pointNumber = getPointNumberFromCell(row.querySelector("td"))
        if (!pointNumber) continue

        fullCount++
        stageCount++

        let point = points[pointNumber]
        if (!point) continue

        if (point.is_puzzle) {
          fullPuzzleCount++
          stagePuzzleCount++
        }

        if (point.history_needed) {
          fullNeedHistoryCount++
          stageNeedHistoryCount++
        }

        if (point.history_ru) {
          fullHistoryCount++
          stageHistoryCount++
        }
      }

      let tableHeader = document.querySelector("table tr:nth-child(2) th")
      tableHeader.append(makeStatsSpan(`Всего КП: ${fullCount}`))
      tableHeader.append(makeStatsSpan(`Загадок: ${fullPuzzleCount}`))
      tableHeader.append(makeStatsSpan(`Нужно ИС: ${fullNeedHistoryCount}`))
      tableHeader.append(makeStatsSpan(`ИС: ${fullHistoryCount}`))

      addStylesToHead(`
        .stats {
          color: red;
          margin-left: .5em;
        }  
      `)
    }

    function addLinksToCp(points) {
      let pointCells = getPointCells()

      for (const [i, pointCell] of pointCells.entries()) {
        let pointNumber = getPointNumberFromCell(pointCell)
        if (pointNumber == null) continue

        let cellContent = pointCell.textContent
        pointCell.innerHTML = ''
        pointCell.append(Tag.a({
          href: cpLink(points[pointNumber].cp_id),
          target: "_blank",
          textContent: cellContent
        }))
      }
    }

    function showHistoryData(points) {
      let pointCells = getPointCells()

      for (const [i, pointCell] of pointCells.entries()) {
        let pointNumber = getPointNumberFromCell(pointCell)
        if (pointNumber == null) continue

        let point = points[pointNumber]

        if (point.history_needed) {
          pointCell.append(Tag.span({ classes: "point-info", textContent: "Нужна ИС" }))
        }

        if (point.history_ru) {
          pointCell.append(Tag.span({ classes: "point-info", textContent: "Есть ИС" }))
        }

        if (point.photo_answer) {
          pointCell.append(Tag.span({ classes: "point-info", textContent: "Селфи" }))
        }
      }

      addStylesToHead(`
        .point-info {
          &:first-of-type {
            margin-left: 5px;
          }

          & + &::before {
            content: " | "
          }
        }
      `)
    }

    function getRouteName() {
      return document.querySelector(`#content h1`).textContent.match(/(?<=\")[^\"]+/)
    }

    function sendRouteData() {
      createPseudoButton(document.querySelector(`input[name="save_sort"]`), async () => {
        let formData = new FormData(document.querySelector(`form[name="main_form"]`))

        let catId = urlParams().get("cat_id")
        formData.set("cat_id", catId)
        formData.set("cat_name", getRouteName())
        await updateRoute(formData)
      })
    }

    function showUnknownPoints() {
      (async () => {
        let unknownPoints = await getUnknownPoints()
        if (unknownPoints.error || !unknownPoints.length) return

        let unknownPointsContainer = Tag.div()
        unknownPointsContainer.append("Несохраненные точки: ")
        for (const [i, cpId] of unknownPoints.entries()) {
          if (i !== 0) {
            unknownPointsContainer.append(", ")
          }
  
          unknownPointsContainer.append(Tag.a({
            href: cpLink(cpId),
            textContent: cpId
          }))
        }
        document.querySelector(`#content h1`).after(unknownPointsContainer)
      })()
    }

    addJs("https://cdnjs.cloudflare.com/ajax/libs/jquery-cookie/1.4.1/jquery.cookie.min.js")
    changeColumnWidth()
    hideDescription()
    hideStartRow()
    useColspanForFinishWarning()
    ;(async () => {
      let points = await getPointsFromRows()
      countCp(points)
      addLinksToCp(points)
      showHistoryData(points)
    })()
    sendRouteData()
    showUnknownPoints()
  }

  function getCpNumberFromOption(option) {
    return option.textContent.match(/(?<=^\s*#)\d+/g)?.[0]
  }

  function matchNumberFromSelect(select) {
    return [...document.querySelector(select).options].reduce(
      (res, el) => ({ ...res, [getCpNumberFromOption(el)]: el.value }),
      {}
    )
  }

  function unselectAll(select) {
    [...document.querySelector(select).selectedOptions].forEach(el => el.selected = false)
  }

  function scrollSelectToBottom(select) {
    document.querySelector(`${select} option:last-child`).scrollIntoView()
  }

  function prettifyRouteEditPage() {
    function createPointsInpit() {
      let pointInput = Tag.input({
        id: "add-point",
        type: "text",
        on: {
          focus: () => {
            pointInput.classList.remove("success", "error")
          }
        }
      })

      let addButton = Tag.button({
        type: "button",
        textContent: "Добавить",
        id: "add-button",
        on: {
          click: () => {
            let point = pointInput.value
            if (point == "") return

            let availablePoints = matchNumberFromSelect("#cps_avail")
            let addedPoints = matchNumberFromSelect("#cps_in")

            if (point in addedPoints) {
              alert("Точка уже добавлена")
              return
            }

            if (!(point in availablePoints)) {
              alert("Точка не найдена")
              return
            }

            unselectAll("#cps_avail")
            let option = document.querySelector(`#cps_avail option[value="${availablePoints[point]}"]`)
            option.selected = true
            document.querySelector(`#cps_add`).click()
            pointInput.classList.add("success")
            pointInput.value = ""
            option.selected = false
            scrollSelectToBottom("#cps_in")
          }
        }
      })

      let removeButton = Tag.button({
        type: "button",
        textContent: "Убрать",
        on: {
          click: () => {
            let point = pointInput.value
            if (point == "") return

            let addedPoints = matchNumberFromSelect("#cps_in")

            if (!(point in addedPoints)) return

            unselectAll("#cps_in")
            let option = document.querySelector(`#cps_in option[value="${addedPoints[point]}"]`)
            option.selected = true
            document.querySelector(`#cps_delete`).click()
            pointInput.classList.add("error")
            pointInput.value = ""
            option.selected = false
            scrollSelectToBottom("#cps_in")
          }
        }
      })

      let addPointsContainer = Tag.div({
        children: [
          addButton,
          pointInput,
          removeButton
        ]
      })
      document.querySelector("table tr:nth-child(16) td").prepend(addPointsContainer)

      addStylesToHead(`
        #cps_in, #cps_avail {
          height: 15em;
        }

        .success {
          border-color: green;
        }

        .error {
          border-color: red;
        }

        #add-point {
          width: 5em;
        }

        #add-button {
          margin-left: 420px;
        }
      `)
    }

    function makeTextareasOneRow() {
      document.querySelectorAll(`textarea:is([name="track[comment_int]"], [name="track[comment_ext]"])`).forEach(el => {
        el.rows = 1
      })
    }

    function hideRedundantRows() {
      document.querySelectorAll(`table tr:is(:nth-child(3), :nth-child(4), :nth-child(5))`).forEach(el => el.classList.add("hidden"))
    }

    async function getPointsFromSelect() {
      let options = [...document.querySelectorAll(`#cps_in option`)]
      return await getPointsByField("number", options.map(option => getCpNumberFromOption(option)))
    }

    function paintCpList(points) {
      let options = [...document.querySelectorAll(`#cps_in option`)]
      let catId = urlParams().get("cat_id")

      for (const option of options) {
        let category = points[getCpNumberFromOption(option)].categories.find(category => category.cat_id === catId)
        if (category && category.pivot.bonus_time == "0") {
          let stage = parseInt(category.pivot.stage)
          option.style['background-color'] = stageColors[(stage - 1) % stageColors.length]
        } 
      }
    }

    function hideAlreadyAddedPoints(points) {
      let options = [...document.querySelectorAll(`#cps_avail option`)]

      for (const option of options) {
        if (getCpNumberFromOption(option) in points) {
          option.style.display = "none"
        }
      }
    }

    makeTextareasOneRow()
    hideRedundantRows()
    createPointsInpit()
    ;(async () => {
      let points = await getPointsFromSelect()
      paintCpList(points)
      hideAlreadyAddedPoints(points)
    })()
  }

  function toggleStagePageRows(rows) {
    let showIndex = [0, 3]
    let to = rows.findIndex(el => el.querySelector("th")?.textContent.trim() == "Бонусы")
    if (to == -1)
      to = rows.length

    toggleRows(rows, 3, to, 8, showIndex)
    toggleRows(rows, to, rows.length, 1, [])
  }

  function hideUselessRowsFromRouteStagesPage() {
    let styles = `
      tr.collapsed {
        display: block;
        width: 0;
        height: 0;
        overflow: hidden;
      }
    `

    let rows = [...document.querySelectorAll(`#content tbody tr`)]
    let collapseButton = Tag.button({
      type: "button",
      textContent: "Показать",
      on: {
        click: () => {
          toggleStagePageRows(rows)
          collapseButton.textContent = toggleText(collapseButton.textContent, "Показать", "Скрыть")
        }
      }
    })

    toggleStagePageRows(rows)

    document.querySelector(`table tr:first-child th`).append(collapseButton)

    addStylesToHead(styles)
  }

  function addCpNameToOptions() {
    let options = [...document.querySelectorAll(`option`)]
    let cpIds = [...new Set(
      options
        .map($option => $option.value)
        .filter($value => $value != null)
      )
    ]

    ;(async () => {
      let points = await getPointsByField("cp_id", cpIds)

      for (const option of options) {
        if (option.value != null && points[option.value] != null) {
          option.innerText = option.innerText + " - " + points[option.value].name_int
        }
      }
    })()
  }

  function addCommonStyles() {
    addStylesToHead(`
      #content h1 {
        margin-bottom: 0;
      }
    `)
  }

  function getDisabledPages() {
    return JSON.parse(localStorage.getItem(localStorageItems.DISABLED_PAGES) ?? "{}")
  }

  function saveDisabledPages(pages) {
    localStorage.setItem(localStorageItems.DISABLED_PAGES, JSON.stringify(pages))
  }

  function isPageDisabled() {
    let disabledPages = getDisabledPages()
    let pageType = getPageType()

    return disabledPages[pageType]?.[getAction()]
  }

  function addDisabledPage() {
    let disabledPages = getDisabledPages()
    let pageType = getPageType()
    if (!(pageType in disabledPages)) {
      disabledPages[pageType] = {}
    }
    disabledPages[pageType][getAction()] = true

    saveDisabledPages(disabledPages)
  }

  function removeDisabledPage() {
    let disabledPages = getDisabledPages()
    let pageType = getPageType()
    if (!(pageType in disabledPages)) {
      return
    }
    delete disabledPages[pageType][getAction()]

    saveDisabledPages(disabledPages)
  }

  function addEnableButtons() {
    let enableButton = Tag.button({
      type: "button",
      id: "enable-button",
      textContent: "Сделать красиво",
      on: {
        click: () => {
          if (pretty) return

          hide([enableButton])
          prettify()
        }
      }
    })

    if (!isPageDisabled()) {
      hide([enableButton])
    }

    let header = document.querySelector("#header")
    header.insertBefore(enableButton, header.querySelector("#globalmenu"))
    header.insertBefore(createAlwaysEnable(), header.querySelector("#globalmenu"))

    addStylesToHead(`
      #always-enable-container {
        display: inline-block;
      }

      .hidden {
        display: none;
      }
    `)
  }

  function addDisableButton() {
    let disableButton = Tag.button({
      id: "disable-button",
      type: "button",
      textContent: "Сделать как было",
      on: {
        click: () => {
          if (!pretty) return

          if (!confirm("Это действие отключит скрипт на этой странице и удалит все несохраненные данные")) return

          addDisabledPage()
          pretty = false
          location.reload()
        }
      }
    })

    header.querySelector("#enable-button").after(disableButton)
  }

  function createAlwaysEnable() {
    let alwaysEnableContainer = Tag.div({
      id: "always-enable-container",
      children: [
        Tag.make("label", {
          _for: "always-enable",
          textContent: "Всегда красиво"
        })
      ]
    })

    let alwaysEnableCheckbox = Tag.input({
      type: "checkbox",
      id: "always-enable",
      name: "always-enable",
      on: {
        change: function () {
          if (this.checked)
            removeDisabledPage()
          else
            addDisabledPage()
        }
      }
    })


    if (!isPageDisabled()) {
      alwaysEnableCheckbox.checked = true
    }

    alwaysEnableContainer.append(alwaysEnableCheckbox)

    return alwaysEnableContainer
  }

  function prettify() {
    if (pretty) return

    pretty = true

    addDisableButton()

    addCommonStyles()

    /* SWITCH FOR DIFFERENT PAGES */

    addStickyMenu()

    if (isCpEditPage()) {
      prettifyEditCpPage()
    }

    if (isRouteListPage()) {
      addStageLink()
      hideNonTesters()
    }

    if (isRouteBuildPage()) {
      prettifyRouteBuildPage()
    }

    if (isRouteEditPage()) {
      prettifyRouteEditPage()
    }

    if (isRouteStagesPage()) {
      hideUselessRowsFromRouteStagesPage()
      addCpNameToOptions()
    }
  }

  function prettifyIfEnabled() {
    if (isPageDisabled()) return

    prettify()
  }

  /* REDIRECTS */

  redirectAfterNewCpIfNeeded()

  /* PRETTIFY */

  let pretty = false

  addEnableButtons()

  if (isCpDeletePage()) {
    bindDeleteButton()
    return
  }

  prettifyIfEnabled()

  /* MAPS */

  addClearBoth()

  if (hasMap()) {
    initMapbox()
    addFullscreenButton()

    if (isRouteBuildPage() || isRouteMapPage()) {
      centerMap()
    }
  }
})();