builder/lib/processors/libraryLessGenerator.js

  1. import {getLogger} from "@ui5/logger";
  2. const log = getLogger("builder:processors:libraryLessGenerator");
  3. import {promisify} from "node:util";
  4. import posixPath from "node:path/posix";
  5. import Resource from "@ui5/fs/Resource";
  6. const IMPORT_PATTERN = /@import .*"(.*)";/g;
  7. const BASE_LESS_PATTERN = /^\/resources\/sap\/ui\/core\/themes\/([^/]+)\/base\.less$/;
  8. const GLOBAL_LESS_PATTERN = /^\/resources\/sap\/ui\/core\/themes\/([^/]+)\/global\.less$/;
  9. class LibraryLessGenerator {
  10. constructor({fs}) {
  11. const readFile = promisify(fs.readFile);
  12. this.readFile = async (filePath) => readFile(filePath, {encoding: "utf8"});
  13. }
  14. async generate({filePath, fileContent}) {
  15. return `/* NOTE: This file was generated as an optimized version of ` +
  16. `"library.source.less" for the Theme Designer. */\n\n` +
  17. (await this.resolveLessImports({
  18. filePath,
  19. fileContent
  20. }));
  21. }
  22. static getPathToRoot(baseDir) {
  23. return posixPath.relative(baseDir, "/") + "/";
  24. }
  25. async resolveLessImports({filePath, fileContent}) {
  26. const imports = this.findLessImports(fileContent);
  27. if (!imports.length) {
  28. // Skip processing when no imports are found
  29. return fileContent;
  30. }
  31. const replacements = await Promise.all(imports.map(async (importMatch) => {
  32. const baseDir = posixPath.dirname(filePath);
  33. const resolvedFilePath = posixPath.resolve(baseDir, importMatch.path);
  34. importMatch.content = await this.resolveLessImport(importMatch.path, resolvedFilePath, baseDir);
  35. return importMatch;
  36. }));
  37. // Apply replacements in reverse order to not modify the relevant indices
  38. const array = Array.from(fileContent);
  39. for (let i = replacements.length - 1; i >= 0; i--) {
  40. const replacement = replacements[i];
  41. if (!replacement.content) {
  42. continue;
  43. }
  44. array.splice(
  45. /* index */ replacement.matchStart,
  46. /* count */ replacement.matchLength,
  47. /* insert */ replacement.content
  48. );
  49. }
  50. return array.join("");
  51. }
  52. async resolveLessImport(originalFilePath, resolvedFilePath, baseDir) {
  53. // Rewrite base.less imports
  54. const baseLessMatch = BASE_LESS_PATTERN.exec(resolvedFilePath);
  55. if (baseLessMatch) {
  56. let baseLessThemeName = baseLessMatch[1];
  57. if (baseLessThemeName === "base") {
  58. baseLessThemeName = "baseTheme";
  59. }
  60. const baseLessPath = LibraryLessGenerator.getPathToRoot(baseDir) +
  61. "Base/baseLib/" + baseLessThemeName + "/base.less";
  62. return "@import \"" + baseLessPath + "\"; /* ORIGINAL IMPORT PATH: \"" + originalFilePath + "\" */\n";
  63. }
  64. // Rewrite library imports to correct file name
  65. if (posixPath.basename(resolvedFilePath) === "library.source.less") {
  66. return `@import "${originalFilePath.replace(/library\.source\.less$/, "library.less")}";`;
  67. }
  68. // Excluding global.less within sap.ui.core
  69. // It must be imported by the Theme Designer (also see declaration in sap/ui/core/.theming)
  70. if (GLOBAL_LESS_PATTERN.test(resolvedFilePath)) {
  71. return null;
  72. }
  73. /*
  74. * Throw error in case of files which are not in the same directory as the current file because
  75. * inlining them would break relative URLs.
  76. * A possible solution would be to rewrite relative URLs when inlining the content.
  77. *
  78. * Keeping the import will cause errors since only "library.less" and "global.less" are
  79. * configured to be available to the Theme Designer (.theming generated in generateThemeDesignerResources).
  80. */
  81. const relativeFilePath = posixPath.relative(baseDir, resolvedFilePath);
  82. if (relativeFilePath.includes(posixPath.sep)) {
  83. throw new Error(
  84. `Could not inline import '${resolvedFilePath}' outside of theme directory '${baseDir}'. ` +
  85. `Stylesheets must be located in the theme directory (no sub-directories).`
  86. );
  87. }
  88. let importedFileContent;
  89. try {
  90. importedFileContent = await this.readFile(resolvedFilePath);
  91. } catch (err) {
  92. if (err.code === "ENOENT") {
  93. throw new Error(
  94. `libraryLessGenerator: Unable to resolve import '${originalFilePath}' from '${baseDir}'\n` +
  95. err.message
  96. );
  97. } else {
  98. throw err;
  99. }
  100. }
  101. return `/* START "${originalFilePath}" */\n` +
  102. (await this.resolveLessImports({
  103. filePath: resolvedFilePath,
  104. fileContent: importedFileContent
  105. })) +
  106. `\n/* END "${originalFilePath}" */\n`;
  107. }
  108. findLessImports(fileContent) {
  109. const imports = [];
  110. let match;
  111. while ((match = IMPORT_PATTERN.exec(fileContent)) !== null) {
  112. imports.push({
  113. path: match[1],
  114. matchStart: match.index,
  115. matchLength: match[0].length
  116. });
  117. }
  118. return imports;
  119. }
  120. }
  121. /**
  122. * @public
  123. * @module @ui5/builder/processors/libraryLessGenerator
  124. */
  125. /**
  126. * Creates a "library.less" file for the SAP Theme Designer based on a "library.source.less" file.
  127. *
  128. * <ul>
  129. * <li>Bundles all *.less file of the theme by replacing the import with the corresponding file content</li>
  130. * <li>Imports to "base.less" are adopted so that they point to the "BaseLib" that is available within
  131. * the Theme Designer infrastructure</li>
  132. * <li>Imports to "global.less" are kept as they should not be bundled</li>
  133. * <li>Imports to "library.source.less" are adopted to "library.less"</li>
  134. * </ul>
  135. *
  136. * @public
  137. * @function default
  138. * @static
  139. *
  140. * @param {object} parameters Parameters
  141. * @param {@ui5/fs/Resource[]} parameters.resources List of <code>library.source.less</code>
  142. * resources
  143. * @param {fs|module:@ui5/fs/fsInterface} parameters.fs Node fs or custom
  144. * [fs interface]{@link module:@ui5/fs/fsInterface}
  145. * @returns {Promise<@ui5/fs/Resource[]>} Promise resolving with library.less resources
  146. */
  147. async function createLibraryLess({resources, fs}) {
  148. const generator = new LibraryLessGenerator({fs});
  149. return Promise.all(resources.map(async (librarySourceLessResource) => {
  150. const filePath = librarySourceLessResource.getPath();
  151. const fileContent = await librarySourceLessResource.getString();
  152. log.verbose(`Generating library.less file based on ${filePath}`);
  153. const libraryLessFileContent = await generator.generate({filePath, fileContent});
  154. const libraryLessFilePath = posixPath.join(posixPath.dirname(filePath), "library.less");
  155. return new Resource({
  156. path: libraryLessFilePath,
  157. string: libraryLessFileContent
  158. });
  159. }));
  160. }
  161. export default createLibraryLess;
  162. let myLibraryLessGenerator;
  163. // Export class for testing only
  164. /* istanbul ignore else */
  165. if (process.env.NODE_ENV === "test") {
  166. myLibraryLessGenerator = LibraryLessGenerator;
  167. }
  168. export const _LibraryLessGenerator = myLibraryLessGenerator;