<script setup>
import { computed, nextTick, ref, watch } from 'vue'
import { find, map, toUpper } from 'lodash'
import useTabStore from 'stores/tab.js'
import ContentServerStatus from '@/components/content_value/ContentServerStatus.vue'
import Status from '@/components/icons/Status.vue'
import { useThemeVars } from 'naive-ui'
import { BrowserTabType } from '@/consts/browser_tab_type.js'
import Terminal from '@/components/icons/Terminal.vue'
import Log from '@/components/icons/Log.vue'
import Detail from '@/components/icons/Detail.vue'
import ContentValueWrapper from '@/components/content_value/ContentValueWrapper.vue'
import ContentCli from '@/components/content_value/ContentCli.vue'
import Monitor from '@/components/icons/Monitor.vue'
import ContentSlog from '@/components/content_value/ContentSlog.vue'
import ContentMonitor from '@/components/content_value/ContentMonitor.vue'
import { decodeRedisKey } from '@/utils/key_convert.js'
import ContentPubsub from '@/components/content_value/ContentPubsub.vue'
import Subscribe from '@/components/icons/Subscribe.vue'

const themeVars = useThemeVars()

/**
 * @typedef {Object} ServerStatusItem
 * @property {string} name
 * @property {Object} info
 * @property {boolean} autoRefresh
 * @property {boolean} loading loading status for refresh
 * @property {boolean} autoLoading loading status for auto refresh
 */

const props = defineProps({
    server: String,
})

const tabStore = useTabStore()
const tab = computed(() =>
    map(tabStore.tabs, (item) => ({
        key: item.name,
        label: item.title,
    })),
)

const tabContent = computed(() => {
    const tab = find(tabStore.tabs, { name: props.server })
    if (tab == null) {
        return {}
    }
    return {
        name: tab.name,
        subTab: tab.subTab,
        type: toUpper(tab.type),
        db: tab.db,
        keyPath: tab.keyCode != null ? decodeRedisKey(tab.keyCode) : tab.key,
        keyCode: tab.keyCode,
        ttl: tab.ttl,
        value: tab.value,
        size: tab.size || 0,
        length: tab.length || 0,
        decode: tab.decode,
        format: tab.format,
        matchPattern: tab.matchPattern || '',
        end: tab.end === true,
        loading: tab.loading === true,
    }
})

const isBlankValue = computed(() => {
    return tabContent.value?.keyPath == null
})

const selectedSubTab = computed(() => {
    const { subTab = 'status' } = tabStore.currentTab || {}
    return subTab
})

// BUG: naive-ui tabs will set the bottom line to '0px' after switch to another page and back again
// watch parent tabs' changing and call 'syncBarPosition' manually
const tabsRef = ref(null)
const cliRef = ref(null)
watch(
    () => tabContent.value?.name,
    (name) => {
        if (name === props.server) {
            nextTick().then(() => {
                tabsRef.value?.syncBarPosition()
                cliRef.value?.resizeTerm()
            })
        }
    },
)
</script>

<template>
    <div class="content-container flex-box-v">
        <n-tabs
            ref="tabsRef"
            :tabs-padding="5"
            :theme-overrides="{
                tabFontWeightActive: 'normal',
                tabGapSmallLine: '10px',
                tabGapMediumLine: '10px',
                tabGapLargeLine: '10px',
            }"
            :value="selectedSubTab"
            class="content-sub-tab"
            default-value="status"
            pane-class="content-sub-tab-pane"
            placement="top"
            tab-style="padding-left: 10px; padding-right: 10px;"
            type="line"
            @update:value="tabStore.switchSubTab">
            <!-- server status pane -->
            <n-tab-pane :name="BrowserTabType.Status.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <status
                                :inverse="selectedSubTab === BrowserTabType.Status.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.status') }}</span>
                    </n-space>
                </template>
                <content-server-status
                    :pause="selectedSubTab !== BrowserTabType.Status.toString()"
                    :server="props.server" />
            </n-tab-pane>

            <!-- key detail pane -->
            <n-tab-pane :name="BrowserTabType.KeyDetail.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <detail
                                :inverse="selectedSubTab === BrowserTabType.KeyDetail.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.key_detail') }}</span>
                    </n-space>
                </template>
                <content-value-wrapper :blank="isBlankValue" :content="tabContent" />
            </n-tab-pane>

            <!-- cli pane -->
            <n-tab-pane :name="BrowserTabType.Cli.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <terminal
                                :inverse="selectedSubTab === BrowserTabType.Cli.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.cli') }}</span>
                    </n-space>
                </template>
                <content-cli ref="cliRef" :name="props.server" />
            </n-tab-pane>

            <!-- slow log pane -->
            <n-tab-pane :name="BrowserTabType.SlowLog.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <log
                                :inverse="selectedSubTab === BrowserTabType.SlowLog.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.slow_log') }}</span>
                    </n-space>
                </template>
                <content-slog :server="props.server" />
            </n-tab-pane>

            <!-- command monitor pane -->
            <n-tab-pane :name="BrowserTabType.CmdMonitor.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <monitor
                                :inverse="selectedSubTab === BrowserTabType.CmdMonitor.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.cmd_monitor') }}</span>
                    </n-space>
                </template>
                <content-monitor :server="props.server" />
            </n-tab-pane>

            <!-- pub/sub message pane -->
            <n-tab-pane :name="BrowserTabType.PubMessage.toString()" display-directive="show:lazy">
                <template #tab>
                    <n-space :size="5" :wrap-item="false" align="center" inline justify="center">
                        <n-icon size="16">
                            <subscribe
                                :inverse="selectedSubTab === BrowserTabType.PubMessage.toString()"
                                :stroke-color="themeVars.tabColor"
                                stroke-width="4" />
                        </n-icon>
                        <span>{{ $t('interface.sub_tab.pub_message') }}</span>
                    </n-space>
                </template>
                <content-pubsub :server="props.server" />
            </n-tab-pane>
        </n-tabs>
    </div>
</template>

<style lang="scss" scoped>
@import '@/styles/content';

.content-container {
    //padding: 5px 5px 0;
    //padding-top: 0;
    box-sizing: border-box;
    background-color: v-bind('themeVars.tabColor');
}
</style>

<style lang="scss">
.content-sub-tab {
    background-color: v-bind('themeVars.tabColor');
    height: 100%;
}

.content-sub-tab-pane {
    padding: 0 !important;
    height: 100%;
    background-color: v-bind('themeVars.bodyColor');
    overflow: hidden;
}

.n-tabs .n-tabs-bar {
    transition: none !important;
}
</style>