Skip to content

Latest commit

 

History

History
225 lines (182 loc) · 6.82 KB

File metadata and controls

225 lines (182 loc) · 6.82 KB

Customizing configurations

We can use the mechanism of customize to adapt the configuration. In the following example, we replace the footer.hbs-partial by a different version. We do this by specifying a new partial directory. Partials with the same name as in the previous directory will overwrite the old one.

const customize = require('customize')
customize()
  .registerEngine('handlebars', require('customize-engine-handlebars'))
  .load(require('./config-module.js'))
  .merge({
    handlebars: {
      partials: 'partials2'
    }
  })
  .run()
  .then(console.log)

The new footer.hbs writes the blog url instead of the username.

------
Blog: {{{github.blog}}}

The output of this example is

{
  "handlebars": {
    "subdir/text3.txt": "------\nBlog: https://blog.knappi.org\n",
    "text1.txt": "I'm nknapp\n\nI'm living in Darmstadt.\n\n------\nBlog: https://blog.knappi.org\n",
    "text2.txt": "I'm nknapp\n\nI'm living in DARMSTADT.\n\n------\nBlog: https://blog.knappi.org\n"
  }
}

In a similar fashion, we could replace other parts of the configuration, like templates, helpers and the pre-processor. If we would provide a new preprocessor, it could call the old one, by calling this.parent(args)

Name of the current target-file

In some cases, we need to know which file we are actually rendering at the moment. If we are rendering the template some/template.txt.hbs, the file some/template.txt will be written (at least if customize-write-files is used. If we want to create relative links from this file, we need this information within helpers. The parameter options.customize.targetFile that is passed to each helper, contains this information. The following configuration registers a helper that return the targetFile:

const customize = require('customize')
customize()
  .registerEngine('handlebars', require('customize-engine-handlebars'))
  .load(require('./config-module.js'))
  .merge({
    handlebars: {
      templates: 'templates',
      partials: 'partials-targetFile',
      helpers: {
        // Helper that returns the targetFile
        targetFile: function(options) {
          return options.customize.targetFile
        }
      }
    }
  })
  .run()
  .then(console.log)

Each template includes the {{>footer}}-partial, which calls the {{targetFile}}-helper to include the name of the current file.

------
File: {{targetFile}}

The output of this configuration is

{
  "handlebars": {
    "subdir/text3.txt": "------\nBlog: https://blog.knappi.org\n",
    "text1.txt": "I'm nknapp\n\nI'm living in Darmstadt.\n\n------\nBlog: https://blog.knappi.org\n",
    "text2.txt": "I'm nknapp\n\nI'm living in DARMSTADT.\n\n------\nBlog: https://blog.knappi.org\n"
  }
}

Accessing engine and configuration helpers

The configuration and the engine itself is passed as additional option into each helper call:

module.exports = {
    function(value, options) {
        console.log("handlebars", options.customize.engine)
        console.log("customizeConfig", options.customize.config)
    }
}

Which partial generates what? (Method 1)

When we want to overriding parts of the output, we are looking for the correct partial to do so. For this purpose, the engine allows to specify a "wrapper function" for partials. This function is called with the contents and the name of a partial and returns the new content. Programs like Thought can optionally include the partial names into the output to show the user which partial to override in order to modify a given part of the output.

const customize = require('customize')
customize()
  .registerEngine('handlebars', require('customize-engine-handlebars'))
  .load(require('./config-module.js'))
  .merge({
    handlebars: {
      partials: 'partials2',
      partialWrapper: function(contents, name) {
        return '[BEGIN ' + name + ']\n' + contents + '[END ' + name + ']'
      }
    }
  })
  .run()
  .then(console.log)
{
  "handlebars": {
    "subdir/text3.txt": "------\nBlog: https://blog.knappi.org\n",
    "text1.txt": "I'm nknapp\n\nI'm living in Darmstadt.\n\n------\nBlog: https://blog.knappi.org\n",
    "text2.txt": "I'm nknapp\n\nI'm living in DARMSTADT.\n\n------\nBlog: https://blog.knappi.org\n"
  }
}

Which partial generates what? (Method 2)

Another method for gathering information about the source of parts of the output are source-locators. The engine incoorporates the library handlebars-source-locators to integrate a kind of "source-map for the poor" into the output. Source-locators are activated by setting the option addSourceLocators to true:

const customize = require('customize')
customize()
  .registerEngine('handlebars', require('customize-engine-handlebars'))
  .load(require('./config-module.js'))
  .merge({
    handlebars: {
      partials: 'partials2',
      addSourceLocators: true
    }
  })
  .run()
  .then(console.log)

The output contain tags that contain location-information of the succeeding text:

  • <sl line="1" col="12" file="templates/text1.txt.hbs"></sl>text... for text the originate from a template file
  • <sl line="1" col="0" partial="footer" file="partials2/footer.hbs"></sl>text... for text the originate from a partial

Example output:

{
  "handlebars": {
    "subdir/text3.txt": "------\nBlog: https://blog.knappi.org\n",
    "text1.txt": "I'm nknapp\n\nI'm living in Darmstadt.\n\n------\nBlog: https://blog.knappi.org\n",
    "text2.txt": "I'm nknapp\n\nI'm living in DARMSTADT.\n\n------\nBlog: https://blog.knappi.org\n"
  }
}

Asynchronous helpers

The customize-engine-handlebars uses the promised-handlebars package as wrapper around Handlebars. It allows helpers to return promises instead of real values.

Documenting configurations

It is not possible to generate docs directly with ``. But using the docEngine, you can generate a JSON that you can render to HTML or Markdown (for example using Handlebars).

const customize = require('customize')
customize()
  .registerEngine('handlebars', require('customize-engine-handlebars').docEngine)
  .load(require('./config-module.js'))
  .merge({
    handlebars: {
      partials: 'partials2'
    }
  })
  .run()
  .then(console.log)

The docEngine performs different transformations on the input configuration. For example, helpers are not loaded, but the path to the file is collected into an array.

{
  "handlebars": {
    "subdir/text3.txt": "------\nBlog: https://blog.knappi.org\n",
    "text1.txt": "I'm nknapp\n\nI'm living in Darmstadt.\n\n------\nBlog: https://blog.knappi.org\n",
    "text2.txt": "I'm nknapp\n\nI'm living in DARMSTADT.\n\n------\nBlog: https://blog.knappi.org\n"
  }
}