builder/lib/processors/themeBuilder.js

  1. import {getLogger} from "@ui5/logger";
  2. const log = getLogger("builder:processors:themeBuilder");
  3. import posixPath from "node:path/posix";
  4. import less from "less-openui5";
  5. import Resource from "@ui5/fs/Resource";
  6. const libraryMatchPattern = /^\/resources\/(.*)\/themes\/[^/]*\/library\.source\.less$/i;
  7. /**
  8. * @public
  9. * @module @ui5/builder/processors/ThemeBuilder
  10. */
  11. /**
  12. * Builds a library theme
  13. *
  14. * @public
  15. * @class
  16. */
  17. export class ThemeBuilder {
  18. /**
  19. * Constructor
  20. *
  21. * @public
  22. * @param {fs|module:@ui5/fs/fsInterface} fs Node fs or custom
  23. * [fs interface]{@link module:@ui5/fs/fsInterface}
  24. */
  25. constructor({fs}) {
  26. this.builder = new less.Builder({fs});
  27. }
  28. /**
  29. * Starts the theme build
  30. *
  31. * @public
  32. * @param {@ui5/fs/Resource[]} resources Library files
  33. * @param {object} [options] Build options
  34. * @param {boolean} [options.compress=false] Compress build output (CSS / JSON)
  35. * @param {boolean} [options.cssVariables=false] Generates the CSS variables
  36. * (css-variables.css, css-variables.source.less) and the skeleton for a theme
  37. * (library-skeleton.css, [library-skeleton-RTL.css])
  38. * @returns {Promise<@ui5/fs/Resource[]>} Resolving with array of created files
  39. */
  40. build(resources, {compress = false, cssVariables = false} = {}) {
  41. const files = [];
  42. const compile = (resource) => {
  43. log.verbose(`Compiling ${resource.getPath()}`);
  44. let libraryName;
  45. const libraryMatch = libraryMatchPattern.exec(resource.getPath());
  46. if (libraryMatch) {
  47. libraryName = libraryMatch[1].replace(/\//g, ".");
  48. }
  49. return this.builder.build({
  50. lessInputPath: resource.getPath(),
  51. library: {
  52. name: libraryName
  53. },
  54. compiler: {
  55. compress
  56. },
  57. cssVariables
  58. }).then((result) => {
  59. const themeDir = posixPath.dirname(resource.getPath());
  60. const libCss = new Resource({
  61. path: themeDir + "/library.css",
  62. string: result.css
  63. });
  64. const libCssRtl = new Resource({
  65. path: themeDir + "/library-RTL.css",
  66. string: result.cssRtl
  67. });
  68. const libParams = new Resource({
  69. path: themeDir + "/library-parameters.json",
  70. string: JSON.stringify(result.variables, null, compress ? null : "\t")
  71. });
  72. files.push(libCss, libCssRtl, libParams);
  73. if (cssVariables) {
  74. const libCssVarsSource = new Resource({
  75. path: themeDir + "/css_variables.source.less",
  76. string: result.cssVariablesSource
  77. });
  78. const libCssVars = new Resource({
  79. path: themeDir + "/css_variables.css",
  80. string: result.cssVariables
  81. });
  82. const libCssSkel = new Resource({
  83. path: themeDir + "/library_skeleton.css",
  84. string: result.cssSkeleton
  85. });
  86. const libCssSkelRtl = new Resource({
  87. path: themeDir + "/library_skeleton-RTL.css",
  88. string: result.cssSkeletonRtl
  89. });
  90. files.push(libCssVarsSource, libCssVars, libCssSkel, libCssSkelRtl);
  91. }
  92. }, (err) => {
  93. log.error(`Error while compiling ${resource.getPath()}: ${err.message}`);
  94. throw err;
  95. });
  96. };
  97. return Promise.all(resources.map(compile)).then(() => {
  98. return files;
  99. });
  100. }
  101. /**
  102. * Clears all cached build results.
  103. *
  104. * Use this method to prevent high memory consumption when building many themes within the same process.
  105. *
  106. * @public
  107. */
  108. clearCache() {
  109. this.builder.clearCache();
  110. }
  111. }
  112. /**
  113. *
  114. * @public
  115. * @typedef {object} ThemeBuilderOptions
  116. * @property {boolean} [compress=false] Compress build output (CSS / JSON)
  117. * @property {boolean} [cssVariables=false] Generates the CSS variables
  118. * (css-variables.css, css-variables.source.less) and the skeleton for a theme
  119. * (library-skeleton.css, [library-skeleton-RTL.css])
  120. */
  121. /**
  122. * Builds a library theme.
  123. *
  124. * @public
  125. * @function default
  126. * @static
  127. *
  128. * @alias @ui5/builder/processors/themeBuilder
  129. * @param {object} parameters Parameters
  130. * @param {@ui5/fs/Resource[]} parameters.resources List of <code>library.source.less</code>
  131. * resources to be processed
  132. * @param {fs|module:@ui5/fs/fsInterface} parameters.fs Node fs or custom
  133. * [fs interface]{@link module:@ui5/fs/fsInterface}
  134. * @param {module:@ui5/builder/processors/ThemeBuilder~ThemeBuilderOptions} [parameters.options] Options
  135. * @returns {Promise<@ui5/fs/Resource[]>} Promise resolving with theme resources
  136. */
  137. export default async function({
  138. resources,
  139. fs,
  140. options = {}
  141. }) {
  142. const {compress, cssVariables} =
  143. /** @type {module:@ui5/builder/processors/ThemeBuilder~ThemeBuilderOptions} */ (options);
  144. const themeBuilder = new ThemeBuilder({fs});
  145. return themeBuilder.build(resources, {
  146. compress,
  147. cssVariables
  148. }).then((files) => {
  149. themeBuilder.clearCache();
  150. return files;
  151. });
  152. }