<template>
  <div class="h-full overflow-auto">
    <div class="relative flex flex-col">
      <div class="flex flex-wrap gap-2 p-2.5 rounded-md default-border bg-gray">
        <div class="w-full flex flex-col">
          <label class="mb-1.5 font-titles" for="name">Name</label>
          <input v-model="spriteName" class="input-field" type="text" name="name" placeholder="New sprite" />
        </div>

        <div class="form-field-half">
          <label class="mb-1.5 font-titles" for="name">Width override</label>
          <input v-model="spriteWidth" class="input-field" type="number" name="width" />
        </div>
        <div class="form-field-half">
          <label class="mb-1.5 font-titles" for="name">Height override</label>
          <input v-model="spriteHeight" class="input-field" type="number" name="height" />
        </div>

        <div class="w-full flex gap-2 mt-2 pb-4 relative">
          <button class="btn-cyan px-4 py-2 flex-1 sm:flex-none sm:min-w-24" type="button" @click.prevent="saveSprite">Save</button>
          <button class="btn-red px-4 py-2 flex-1 sm:flex-none sm:min-w-24" type="button" @click.prevent="deleteSprite">Delete</button>
          <button class="btn-indigo px-4 py-2 flex-1 sm:flex-none" type="button" @click.prevent="copySprite">
            <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" fill="none" viewBox="0 0 24 24" stroke="currentColor">
              <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z" />
            </svg>
          </button>
        </div>
      </div>

      <button class="btn-cyan py-2 my-4" type="button" @click.prevent="addNewImage">New action</button>
      <Accordion v-for="action in spriteActions" :key="action.id">
        <template #header>
          <div class="flex items-center">
            {{ action.action }}
            <div class="ml-auto space-x-2">
              <button class="btn-cyan px-4 py-1.5 min-w-24" type="button" @click.stop.prevent="openPreviewModal(action)">View</button>
              <button class="btn-red px-4 py-1.5 min-w-24" type="button" @click.stop.prevent="() => spriteActions.splice(spriteActions.indexOf(action), 1)">Delete</button>
            </div>
          </div>
        </template>
        <template #content>
          <form class="flex gap-2.5 flex-wrap" @submit.prevent="saveSprite">
            <div class="form-field-full">
              <label for="action">Action</label>
              <input v-model="action.action" class="input-field" type="text" name="action" placeholder="Action" />
            </div>
            <div class="form-field-half">
              <label for="origin-x">Origin X</label>
              <input v-model.number="action.originX" class="input-field" type="number" step="any" name="origin-x" placeholder="Origin X" />
            </div>
            <div class="form-field-half">
              <label for="origin-y">Origin Y</label>
              <input v-model.number="action.originY" class="input-field" type="number" step="any" name="origin-y" placeholder="Origin Y" />
            </div>
            <div class="form-field-full">
              <label for="frame-speed">Frame rate</label>
              <input v-model.number="action.frameRate" class="input-field" type="number" step="any" name="frame-speed" placeholder="Frame rate" />
            </div>
            <div class="form-field-full">
              <SpriteActionsInput v-model="action.sprites" @tempOffsetChange="(index, offset) => handleTempOffsetChange(action, index, offset)" />
            </div>
          </form>
        </template>
      </Accordion>
      <SpritePreview
        v-if="selectedAction"
        :sprites="selectedAction.sprites"
        :frame-rate="selectedAction.frameRate"
        :is-modal-open="isModalOpen"
        :temp-offset-index="tempOffsetData.index"
        :temp-offset="tempOffsetData.offset"
        @update:frame-rate="updateFrameRate"
        @update:is-modal-open="isModalOpen = $event"
      />
    </div>
  </div>
</template>

<script setup lang="ts">
import { SocketEvent } from '@/application/enums'
import type { Sprite, SpriteAction } from '@/application/types'
import { downloadCache, uuidv4 } from '@/application/utilities'
import SpriteActionsInput from '@/components/gameMaster/assetManager/partials/sprite/partials/SpriteImagesInput.vue'
import SpritePreview from '@/components/gameMaster/assetManager/partials/sprite/partials/SpritePreview.vue'
import Accordion from '@/components/utilities/Accordion.vue'
import { socketManager } from '@/managers/SocketManager'
import { SpriteStorage } from '@/storage/storages'
import { useAssetManagerStore } from '@/stores/assetManagerStore'
import { computed, onBeforeUnmount, onMounted, ref, watch } from 'vue'

const assetManagerStore = useAssetManagerStore()

const selectedSprite = computed(() => assetManagerStore.selectedSprite)

const spriteName = ref('')
const spriteWidth = ref(0)
const spriteHeight = ref(0)
const spriteActions = ref<SpriteAction[]>([])
const isModalOpen = ref(false)
const selectedAction = ref<SpriteAction | null>(null)

if (!selectedSprite.value) {
  console.error('No sprite selected')
}

if (selectedSprite.value) {
  spriteName.value = selectedSprite.value.name
  spriteWidth.value = selectedSprite.value.width
  spriteHeight.value = selectedSprite.value.height
  spriteActions.value = sortSpriteActions(selectedSprite.value.spriteActions)
}

async function deleteSprite() {
  socketManager.emit(SocketEvent.GM_SPRITE_DELETE, { id: selectedSprite.value?.id }, async (response: boolean) => {
    if (!response) {
      console.error('Failed to delete sprite')
      return
    }

    await downloadCache('sprites', new SpriteStorage())
    await refreshSpriteList()
  })
}

async function copySprite() {
  socketManager.emit(SocketEvent.GM_SPRITE_COPY, { id: selectedSprite.value?.id }, async (response: boolean) => {
    if (!response) {
      console.error('Failed to copy sprite')
      return
    }

    await downloadCache('sprites', new SpriteStorage())
    await refreshSpriteList(false)
  })
}

async function refreshSpriteList(unsetSelectedSprite = true) {
  socketManager.emit(SocketEvent.GM_SPRITE_LIST, {}, (response: Sprite[]) => {
    assetManagerStore.setSpriteList(response)

    if (unsetSelectedSprite) {
      assetManagerStore.setSelectedSprite(null)
    }
  })
}

async function saveSprite() {
  if (!selectedSprite.value) {
    console.error('No sprite selected')
    return
  }

  const updatedSprite = {
    id: selectedSprite.value.id,
    name: spriteName.value,
    width: spriteWidth.value,
    height: spriteHeight.value,
    spriteActions:
      spriteActions.value?.map((action) => {
        return {
          action: action.action,
          sprites: action.sprites,
          originX: action.originX,
          originY: action.originY,
          frameRate: action.frameRate,
          frameWidth: action.frameWidth,
          frameHeight: action.frameHeight
        }
      }) ?? []
  }

  socketManager.emit(SocketEvent.GM_SPRITE_UPDATE, updatedSprite, async (response: boolean) => {
    if (!response) {
      console.error('Failed to save sprite')
      return
    }

    await downloadCache('sprites', new SpriteStorage())
    await refreshSpriteList(false)
  })
}

function addNewImage() {
  if (!selectedSprite.value) return

  const newImage: SpriteAction = {
    id: uuidv4(),
    sprite: selectedSprite.value.id,
    action: 'new_action',
    sprites: [],
    originX: 0,
    originY: 0,
    frameRate: 0,
    frameWidth: 0,
    frameHeight: 0
  }

  if (!spriteActions.value) {
    spriteActions.value = []
  }

  spriteActions.value = sortSpriteActions([...spriteActions.value, newImage])
}

function sortSpriteActions(actions: SpriteAction[]): SpriteAction[] {
  if (!actions) return []
  return [...actions].sort((a, b) => a.action.localeCompare(b.action))
}

function openPreviewModal(action: SpriteAction) {
  selectedAction.value = action
  isModalOpen.value = true
}

function updateFrameRate(value: number) {
  if (selectedAction.value) {
    selectedAction.value.frameRate = value
  }
}

const tempOffsetData = ref<{ index: number | undefined; offset: { x: number; y: number } | undefined }>({
  index: undefined,
  offset: undefined
})

function handleTempOffsetChange(action: SpriteAction, index: number, offset: { x: number; y: number }) {
  if (selectedAction.value === action) {
    tempOffsetData.value = { index, offset }
  }
}

watch(selectedSprite, (sprite: Sprite | null) => {
  if (!sprite) return
  spriteName.value = sprite.name
  spriteWidth.value = sprite.width
  spriteHeight.value = sprite.height
  spriteActions.value = sortSpriteActions(sprite.spriteActions)
})

watch(isModalOpen, (newValue) => {
  if (!newValue) {
    selectedAction.value = null
  }
})

onMounted(() => {
  if (!selectedSprite.value) return
})

onBeforeUnmount(() => {
  assetManagerStore.setSelectedSprite(null)
})
</script>