import { resolve, relative, dirname } from 'path'
import { existsSync } from 'fs'
import visit from 'unist-util-visit'
import { readFileSync, hash } from './util'
import { render } from './page'
import { parseComponent } from 'vue-template-compiler'
import { escape } from 'lodash'
import { RE_LOCALE } from './i18n'

const DOCS_DIR = resolve(__dirname, '../docs')

export default function attacher () {
  return (tree, file) => {
    let { path, data: fileData } = file
    visit(tree, 'shortcode', node => {
      let {
        identifier,
        data: nodeData,
        attributes: { src, browser } = {}
      } = node
      if (identifier !== 'demo' || !src) {
        return
      }

      if (!nodeData) {
        node.data = nodeData = {}
      }
      if (!fileData) {
        file.data = fileData = {}
      }
      if (!fileData.demos) {
        fileData.demos = {}
      }
      if (!fileData.deps) {
        fileData.deps = {}
      }

      let demoPath =
        src.indexOf('/') === 0
          ? resolve(DOCS_DIR, src.slice(1))
          : resolve(dirname(path), src)

      let demoRelativePath = relative(DOCS_DIR, demoPath)

      if (!existsSync(demoPath)) {
        console.warn(`Demo not found at '${demoPath}'`)
        return
      }

      fileData.deps[demoPath] = true

      let name = getComponentName(demoPath)
      nodeData.hName = name

      let localPath = `/${relative(DOCS_DIR, path)}`
      let [, locale] = localPath.match(RE_LOCALE) || []

      let { content, doc } = extractDoc(demoPath, { locale })
      fileData.demos[name] = {
        filePath: demoPath,
        path: demoRelativePath,
        browser,
        code: render('```vue\n' + content + '\n```', demoPath).contents,
        desc: render(doc, demoPath).contents
      }
    })
  }
}

function getComponentName (path) {
  return `one-demo-${hash(path)}`
}

function extractDoc (file, { locale }) {
  let content = readFileSync(file)
  let vueComponent = parseComponent(content)
  let { customBlocks } = vueComponent
  let i = customBlocks.findIndex(({ type, attrs }) => {
    return ((locale && locale === attrs.locale) || !locale) && type === 'docs'
  })

  let doc = null
  if (i !== -1) {
    doc = customBlocks[i].content
  }

  return {
    content: stringifyVueComponent(vueComponent),
    doc
  }
}

function stringifyVueComponent (component) {
  let content = []
  let { template, script, styles } = component
  if (template) {
    content.push(
      `<template${stringifyAttrs(template.attrs)}>${
        template.content
      }</template>`
    )
  }
  if (script) {
    content.push(
      `<script${stringifyAttrs(script.attrs)}>${script.content}</script>`
    )
  }
  styles.filter(({ attrs }) => !attrs.docs).forEach(style => {
    content.push(
      `<style${stringifyAttrs(style.attrs)}>${style.content}</style>`
    )
  })
  return content.join('\n\n')
}

function stringifyAttrs (attrs) {
  let result = Object.keys(attrs)
    .map(key => {
      let val = attrs[key]
      if (typeof val !== 'boolean') {
        return `${escape(key)}="${escape(val)}"`
      }
      return val ? `${escape(key)}` : ''
    })
    .join(' ')

  return result ? ` ${result}` : ''
}