Source: decorator.js

  1. /**
  2. * @module decorator
  3. */
  4. import {isString, checkType, getFullDocPath} from "./utils.js";
  5. /**
  6. * @description Layout decorator.
  7. */
  8. class Decorator {
  9. /**
  10. * @param {Logger} logger
  11. * @param {Compiler} compiler
  12. * @return {Decorator}
  13. */
  14. constructor(logger, compiler) {
  15. this.logger = logger;
  16. this.compiler = compiler;
  17. this._ = new Map();
  18. }
  19. /**
  20. * @callback decorateCallback
  21. * @param {Object} ctx
  22. * @return {String}
  23. */
  24. /**
  25. * @description Register a decorate function.
  26. * @param {String} layout
  27. * @param {decorateCallback|String} fn If string, will call Compiler on this
  28. * filename while decorating.
  29. * @param {Object} [ctx=null] Custom context if you want to pass something to
  30. * this decorator. This is deprecated and you should use helper function with
  31. * layout filter.
  32. */
  33. register(layout, fn, ctx = null) {
  34. checkType(layout, "layout", "String");
  35. checkType(fn, "fn", "Function", "String");
  36. if (ctx != null) {
  37. this.logger.warn("Hikaru suggests you to pass context for specific layouts by using helper function with layout filter because passing context while registering decorator function is deprecated!");
  38. }
  39. this._.set(layout, {layout, fn, ctx});
  40. }
  41. /**
  42. * @description Unregister a decorate function.
  43. * @param {String} layout
  44. */
  45. unregister(layout) {
  46. checkType(layout, "layout", "String");
  47. this._.delete(layout);
  48. }
  49. /**
  50. * @description Decorate input file with layout.
  51. * @param {File} file
  52. * @return {String}
  53. */
  54. async decorate(file, ctx) {
  55. const layout = this.getFileLayout(file);
  56. if (layout != null) {
  57. this.logger.debug(`Hikaru is decorating \`${
  58. this.logger.cyan(getFullDocPath(file))
  59. }\` with layout \`${
  60. this.logger.blue(layout)
  61. }\`...`);
  62. const handler = this._.get(layout);
  63. if (handler == null) {
  64. throw new Error(`Decorator for \`${layout}\` is not registered.`);
  65. }
  66. let fn = handler["fn"];
  67. if (isString(fn)) {
  68. fn = await this.compiler.compile(handler["fn"]);
  69. }
  70. // We support two methods to access file properties, first the file
  71. // properties will be merged with context properties, so you could
  72. // directly access them via property names, and then you could also access
  73. // file properties via `file` object, this is recommended because it is
  74. // easier to prevent the single `file` property name from conflicting and
  75. // it is always accessible.
  76. return fn(Object.assign(file, handler["ctx"], ctx, {"file": file}));
  77. }
  78. return file["content"];
  79. }
  80. /**
  81. * @description List registered layout.
  82. * @return {String[]}
  83. */
  84. list() {
  85. return [...this._.keys()];
  86. }
  87. /**
  88. * @private
  89. * @description Get simpified layout for file.
  90. * @param {File} file
  91. * @return {String}
  92. */
  93. getFileLayout(file) {
  94. if (file["layout"] == null) {
  95. return null;
  96. }
  97. if (!this._.has(file["layout"])) {
  98. return "page";
  99. }
  100. return file["layout"];
  101. }
  102. }
  103. export default Decorator;