template.ts 3.1 KB
Newer Older
fxy060608's avatar
fxy060608 已提交
1
import path from 'path'
fxy060608's avatar
fxy060608 已提交
2
import { EmittedAsset } from 'rollup'
fxy060608's avatar
fxy060608 已提交
3
import { LINEFEED } from '@dcloudio/uni-shared'
fxy060608's avatar
fxy060608 已提交
4
import { normalizeMiniProgramFilename } from '../utils'
fxy060608's avatar
fxy060608 已提交
5

fxy060608's avatar
fxy060608 已提交
6
export interface MiniProgramCompilerOptions {
fxy060608's avatar
fxy060608 已提交
7 8 9 10 11 12
  /**
   * 需要延迟渲染的组件,通常是某个组件的某个事件会立刻触发,需要延迟到首次 render 之后,比如微信 editor 的 ready 事件,快手 switch 的 change
   */
  lazyElement?: {
    [name: string]: string[]
  }
fxy060608's avatar
fxy060608 已提交
13 14 15 16 17 18
  event?: {
    format(
      name: string,
      opts: { isCatch?: boolean; isCapture?: boolean; isComponent?: boolean }
    ): string
  }
fxy060608's avatar
fxy060608 已提交
19 20 21 22 23 24
  class: {
    /**
     * 是否支持绑定 array 类型
     */
    array: boolean
  }
fxy060608's avatar
fxy060608 已提交
25
  slot: {
fxy060608's avatar
fxy060608 已提交
26 27 28 29
    /**
     * 是否支持 $slots.default 访问
     */
    $slots?: boolean
fxy060608's avatar
fxy060608 已提交
30 31 32
    /**
     * 是否支持后备内容
     */
33 34 35 36 37
    fallbackContent?: boolean
    /**
     * 是否支持动态插槽名
     */
    dynamicSlotNames?: boolean
fxy060608's avatar
fxy060608 已提交
38 39 40 41 42 43 44
  }
  filter?: {
    lang: string
  }
  directive: string
  emitFile?: (emittedFile: EmittedAsset) => string
}
fxy060608's avatar
fxy060608 已提交
45 46
export interface MiniProgramFilterOptions {
  id: string
fxy060608's avatar
fxy060608 已提交
47
  type: string
fxy060608's avatar
fxy060608 已提交
48 49 50 51 52
  name: string
  src?: string
  code: string
}

fxy060608's avatar
fxy060608 已提交
53 54 55 56
type GenFilterFn = (
  filter: MiniProgramFilterOptions,
  filename: string
) => string | void
fxy060608's avatar
fxy060608 已提交
57 58

const templateFilesCache = new Map<string, string>()
fxy060608's avatar
fxy060608 已提交
59
const templateFiltersCache = new Map<string, MiniProgramFilterOptions[]>()
fxy060608's avatar
fxy060608 已提交
60

fxy060608's avatar
fxy060608 已提交
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
function relativeFilterFilename(
  filename: string,
  filter: MiniProgramFilterOptions
) {
  if (!filter.src) {
    return ''
  }
  return (
    './' +
    normalizeMiniProgramFilename(
      path.relative(path.dirname(filename), filter.src)
    )
  )
}

fxy060608's avatar
fxy060608 已提交
76 77 78 79 80 81 82
export function findMiniProgramTemplateFiles(genFilter?: GenFilterFn) {
  const files: Record<string, string> = Object.create(null)
  templateFilesCache.forEach((code, filename) => {
    if (!genFilter) {
      files[filename] = code
    } else {
      const filters = getMiniProgramTemplateFilters(filename)
fxy060608's avatar
fxy060608 已提交
83
      if (filters && filters.length) {
fxy060608's avatar
fxy060608 已提交
84
        files[filename] =
fxy060608's avatar
fxy060608 已提交
85 86 87 88 89
          filters
            .map((filter) =>
              genFilter(filter, relativeFilterFilename(filename, filter))
            )
            .join(LINEFEED) +
fxy060608's avatar
fxy060608 已提交
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
          LINEFEED +
          code
      } else {
        files[filename] = code
      }
    }
  })
  return files
}

export function clearMiniProgramTemplateFiles() {
  templateFilesCache.clear()
}

export function addMiniProgramTemplateFile(filename: string, code: string) {
  templateFilesCache.set(filename, code)
}

function getMiniProgramTemplateFilters(filename: string) {
fxy060608's avatar
fxy060608 已提交
109
  return templateFiltersCache.get(filename)
fxy060608's avatar
fxy060608 已提交
110 111 112 113 114 115 116 117 118 119 120 121
}

export function clearMiniProgramTemplateFilter(filename: string) {
  templateFiltersCache.delete(filename)
}

export function addMiniProgramTemplateFilter(
  filename: string,
  filter: MiniProgramFilterOptions
) {
  const filters = templateFiltersCache.get(filename)
  if (filters) {
fxy060608's avatar
fxy060608 已提交
122 123 124 125 126 127
    const filterIndex = filters.findIndex((f) => f.id === filter.id)
    if (filterIndex > -1) {
      filters.splice(filterIndex, 1, filter)
    } else {
      filters.push(filter)
    }
fxy060608's avatar
fxy060608 已提交
128
  } else {
fxy060608's avatar
fxy060608 已提交
129
    templateFiltersCache.set(filename, [filter])
fxy060608's avatar
fxy060608 已提交
130 131
  }
}