import type { Logger, SurveyQuestionTranslation, SurveyTranslation } from '../types'
import { isArray, isUndefined } from '../utils'

export type DetectSurveyLanguageOptions = {
  overrideLanguage?: unknown
  storedPersonProperties?: unknown
  locale?: unknown
}

function getTrimmedLanguage(value: unknown): string | null {
  return typeof value === 'string' && value.trim() ? value.trim() : null
}

export function getLanguageFromStoredPersonProperties(storedPersonProperties: unknown): string | null {
  if (
    !storedPersonProperties ||
    typeof storedPersonProperties !== 'object' ||
    !('language' in storedPersonProperties)
  ) {
    return null
  }

  return getTrimmedLanguage(storedPersonProperties.language)
}

export function detectSurveyLanguage(
  { overrideLanguage, storedPersonProperties, locale }: DetectSurveyLanguageOptions,
  logger?: Logger
): string | null {
  const explicitLanguage = getTrimmedLanguage(overrideLanguage)
  if (explicitLanguage) {
    logger?.info(`Using override display language: ${explicitLanguage}`)
    return explicitLanguage
  }

  const personLanguage = getLanguageFromStoredPersonProperties(storedPersonProperties)
  if (personLanguage) {
    logger?.info(`Using person property language: ${personLanguage}`)
    return personLanguage
  }

  const detectedLocale = getTrimmedLanguage(locale)
  if (detectedLocale) {
    logger?.info(`Using detected locale: ${detectedLocale}`)
    return detectedLocale
  }

  logger?.info('No user language detected')
  return null
}

export function normalizeLanguageCode(languageCode: string): string {
  return languageCode.toLowerCase()
}

export function getBaseLanguage(languageCode: string): string {
  return languageCode.split('-')[0]
}

export function findBestTranslationMatch(
  translations: Record<string, unknown> | undefined,
  targetLanguage: string,
  logger?: Logger
): string | null {
  if (!translations || !targetLanguage) {
    return null
  }

  const normalizedTarget = normalizeLanguageCode(targetLanguage)

  const exactMatch = Object.keys(translations).find((key) => normalizeLanguageCode(key) === normalizedTarget)
  if (exactMatch) {
    logger?.debug(`Found exact translation match: ${exactMatch}`)
    return exactMatch
  }

  if (normalizedTarget.includes('-')) {
    const baseLanguage = getBaseLanguage(normalizedTarget)
    const baseMatch = Object.keys(translations).find((key) => normalizeLanguageCode(key) === baseLanguage)
    if (baseMatch) {
      logger?.debug(`Found base language translation match: ${baseMatch} (from ${targetLanguage})`)
      return baseMatch
    }
  }

  return null
}

function isTranslatedChoices(
  questionTranslation: SurveyQuestionTranslation
): questionTranslation is SurveyQuestionTranslation & { choices: string[] } {
  return isArray(questionTranslation.choices)
}

function hasThankYouTranslation(translation: SurveyTranslation): boolean {
  return (
    !isUndefined(translation.thankYouMessageHeader) ||
    !isUndefined(translation.thankYouMessageDescription) ||
    !isUndefined(translation.thankYouMessageCloseButtonText)
  )
}

type TranslatableSurveyAppearance = {
  thankYouMessageHeader?: string
  thankYouMessageDescription?: string | null
  thankYouMessageCloseButtonText?: string
}

type TranslatableSurveyQuestion = {
  question: string
  description?: string | null
  buttonText?: string
  link?: string | null
  lowerBoundLabel?: string
  upperBoundLabel?: string
  choices?: string[]
  translations?: Record<string, SurveyQuestionTranslation>
}

type TranslatableSurvey<TQuestion extends TranslatableSurveyQuestion = TranslatableSurveyQuestion> = {
  name: string
  translations?: Record<string, SurveyTranslation>
  appearance?: TranslatableSurveyAppearance | null
  questions: TQuestion[]
}

function mergeQuestionTranslation<TQuestion extends TranslatableSurveyQuestion>(
  question: TQuestion,
  targetLanguage: string,
  logger?: Logger
): { question: TQuestion; matchedKey: string | null; hasChanges: boolean } {
  const translationKey = findBestTranslationMatch(question.translations, targetLanguage, logger)
  if (!translationKey) {
    return { question, matchedKey: null, hasChanges: false }
  }

  const questionTranslation = question.translations?.[translationKey]
  if (!questionTranslation) {
    return { question, matchedKey: null, hasChanges: false }
  }

  const translated: TQuestion = { ...question }
  let hasChanges = false

  if (!isUndefined(questionTranslation.question)) {
    translated.question = questionTranslation.question
    hasChanges = true
  }
  if (!isUndefined(questionTranslation.description)) {
    translated.description = questionTranslation.description
    hasChanges = true
  }
  if (!isUndefined(questionTranslation.buttonText)) {
    translated.buttonText = questionTranslation.buttonText
    hasChanges = true
  }

  if ('link' in translated && !isUndefined(questionTranslation.link)) {
    translated.link = questionTranslation.link
    hasChanges = true
  }

  if ('lowerBoundLabel' in translated && !isUndefined(questionTranslation.lowerBoundLabel)) {
    translated.lowerBoundLabel = questionTranslation.lowerBoundLabel
    hasChanges = true
  }
  if ('upperBoundLabel' in translated && !isUndefined(questionTranslation.upperBoundLabel)) {
    translated.upperBoundLabel = questionTranslation.upperBoundLabel
    hasChanges = true
  }

  if ('choices' in translated && isTranslatedChoices(questionTranslation)) {
    translated.choices = questionTranslation.choices
    hasChanges = true
  }

  return {
    question: hasChanges ? translated : question,
    matchedKey: hasChanges ? translationKey : null,
    hasChanges,
  }
}

export function applySurveyTranslation<
  TQuestion extends TranslatableSurveyQuestion,
  TSurvey extends TranslatableSurvey<TQuestion>,
>(survey: TSurvey, targetLanguage: string, logger?: Logger): { survey: TSurvey; matchedKey: string | null } {
  const translationKey = findBestTranslationMatch(survey.translations, targetLanguage, logger)

  const translated: TSurvey = { ...survey }
  let hasTranslation = false

  if (translationKey) {
    const translation = survey.translations?.[translationKey]
    if (translation) {
      logger?.info(`Applying survey-level translation for language: ${translationKey}`)

      if (!isUndefined(translation.name)) {
        translated.name = translation.name
        hasTranslation = true
      }

      if (translated.appearance) {
        translated.appearance = { ...translated.appearance }

        if (!isUndefined(translation.thankYouMessageHeader)) {
          translated.appearance.thankYouMessageHeader = translation.thankYouMessageHeader
          hasTranslation = true
        }
        if (!isUndefined(translation.thankYouMessageDescription)) {
          translated.appearance.thankYouMessageDescription = translation.thankYouMessageDescription
          hasTranslation = true
        }
        if (!isUndefined(translation.thankYouMessageCloseButtonText)) {
          translated.appearance.thankYouMessageCloseButtonText = translation.thankYouMessageCloseButtonText
          hasTranslation = true
        }
      } else if (hasThankYouTranslation(translation)) {
        hasTranslation = true
      }
    }
  }

  const translatedResults = survey.questions.map((question) =>
    mergeQuestionTranslation(question, targetLanguage, logger)
  )
  const translatedQuestions = translatedResults.map((result) => result.question)
  const anyQuestionTranslated = translatedResults.some((result) => result.hasChanges)

  let questionMatchedKey: string | null = null
  if (!translationKey) {
    questionMatchedKey = translatedResults.find((result) => result.matchedKey)?.matchedKey || null
  }

  if (anyQuestionTranslated) {
    translated.questions = translatedQuestions
    hasTranslation = true
    logger?.info(`Applied question-level translations for language: ${targetLanguage}`)
  }

  return {
    survey: translated,
    matchedKey: hasTranslation ? translationKey || questionMatchedKey : null,
  }
}
