npm

kdu-loader
TypeScript icon, indicating that this package has built-in type declarations

17.0.0 • Public • Published

kdu-loader

webpack loader for Kdu Single-File Components

v16+ Only Options

  • reactivityTransform: boolean: enable Kdu Reactivity Transform (SFCs only).

  • refSugar: boolean: removed. use reactivityTransform instead.

  • customElement: boolean | RegExp: enable custom elements mode. An SFC loaded in custom elements mode inlines its <style> tags as strings under the component's styles option. When used with defineCustomElement from Kdu core, the styles will be injected into the custom element's shadow root.

    • Default is /\.ce\.kdu$/
    • Setting to true will process all .kdu files in custom element mode.
  • enableTsInTemplate: boolean (16.8+): allow TS expressions in templates when <script> has lang="ts". Defaults to true.

    • When used with ts-loader, due to ts-loader's cache invalidation behavior, it sometimes prevents the template from being hot-reloaded in isolation, causing the component to reload despite only the template being edited. If this is annoying, you can set this option to false (and avoid using TS expressions in templates).

    • Alternatively, leave this option on (by default) and use esbuild-loader to transpile TS instead, which doesn't suffer from this problem (it's also a lot faster). However, do note you will need to rely on TS type checking from other sources (e.g. IDE or kdu-tsc).

What is Kdu Loader?

kdu-loader is a loader for webpack that allows you to author Kdu components in a format called Single-File Components (SFCs):

<template>
  <div class="example">{{ msg }}</div>
</template>

<script>
export default {
  data() {
    return {
      msg: 'Hello world!',
    }
  },
}
</script>

<style>
.example {
  color: red;
}
</style>

There are many cool features provided by kdu-loader:

  • Allows using other webpack loaders for each part of a Kdu component, for example Sass for <style> and Pug for <template>;
  • Allows custom blocks in a .kdu file that can have custom loader chains applied to them;
  • Treat static assets referenced in <style> and <template> as module dependencies and handle them with webpack loaders;
  • Simulate scoped CSS for each component;
  • State-preserving hot-reloading during development.

In a nutshell, the combination of webpack and kdu-loader gives you a modern, flexible and extremely powerful front-end workflow for authoring Kdu.js applications.

How It Works

The following section is for maintainers and contributors who are interested in the internal implementation details of kdu-loader, and is not required knowledge for end users.

kdu-loader is not a simple source transform loader. It handles each language blocks inside an SFC with its own dedicated loader chain (you can think of each block as a "virtual module"), and finally assembles the blocks together into the final module. Here's a brief overview of how the whole thing works:

  1. kdu-loader parses the SFC source code into an SFC Descriptor using @kdujs/compiler-sfc. It then generates an import for each language block so the actual returned module code looks like this:

    // code returned from the main loader for 'source.kdu'
    
    // import the <template> block
    import render from 'source.kdu?kdu&type=template'
    // import the <script> block
    import script from 'source.kdu?kdu&type=script'
    export * from 'source.kdu?kdu&type=script'
    // import <style> blocks
    import 'source.kdu?kdu&type=style&index=1'
    
    script.render = render
    export default script

    Notice how the code is importing source.kdu itself, but with different request queries for each block.

  2. We want the content in script block to be treated like .js files (and if it's <script lang="ts">, we want to to be treated like .ts files). Same for other language blocks. So we want webpack to apply any configured module rules that matches .js also to requests that look like source.kdu?kdu&type=script. This is what KduLoaderPlugin (src/plugins.ts) does: for each module rule in the webpack config, it creates a modified clone that targets corresponding Kdu language block requests.

    Suppose we have configured babel-loader for all *.js files. That rule will be cloned and applied to Kdu SFC <script> blocks as well. Internally to webpack, a request like

    import script from 'source.kdu?kdu&type=script'

    Will expand to:

    import script from 'babel-loader!kdu-loader!source.kdu?kdu&type=script'

    Notice the kdu-loader is also matched because kdu-loader are applied to .kdu files.

    Similarly, if you have configured style-loader + css-loader + sass-loader for *.scss files:

    <style scoped lang="scss">

    Will be returned by kdu-loader as:

    import 'source.kdu?kdu&type=style&index=1&scoped&lang=scss'

    And webpack will expand it to:

    import 'style-loader!css-loader!sass-loader!kdu-loader!source.kdu?kdu&type=style&index=1&scoped&lang=scss'
  3. When processing the expanded requests, the main kdu-loader will get invoked again. This time though, the loader notices that the request has queries and is targeting a specific block only. So it selects (src/select.ts) the inner content of the target block and passes it on to the loaders matched after it.

  4. For the <script> block, this is pretty much it. For <template> and <style> blocks though, a few extra tasks need to be performed:

    • We need to compile the template using the Kdu template compiler;
    • We need to post-process the CSS in <style scoped> blocks, after css-loader but before style-loader.

    Technically, these are additional loaders (src/templateLoader.ts and src/stylePostLoader.ts) that need to be injected into the expanded loader chain. It would be very complicated if the end users have to configure this themselves, so KduLoaderPlugin also injects a global Pitching Loader (src/pitcher.ts) that intercepts Kdu <template> and <style> requests and injects the necessary loaders. The final requests look like the following:

    // <template lang="pug">
    import 'kdu-loader/template-loader!pug-loader!source.kdu?kdu&type=template'
    
    // <style scoped lang="scss">
    import 'style-loader!kdu-loader/style-post-loader!css-loader!sass-loader!kdu-loader!source.kdu?kdu&type=style&index=1&scoped&lang=scss'

Readme

Keywords

none

Package Sidebar

Install

npm i kdu-loader

Weekly Downloads

1

Version

17.0.0

License

MIT

Unpacked Size

59.8 kB

Total Files

37

Last publish

Collaborators

  • nkduy