builder/lib/tasks/generateThemeDesignerResources.js

  1. import posixPath from "node:path/posix";
  2. import {getLogger} from "@ui5/logger";
  3. const log = getLogger("builder:tasks:generateThemeDesignerResources");
  4. import libraryLessGenerator from "../processors/libraryLessGenerator.js";
  5. import {updateLibraryDotTheming} from "./utils/dotTheming.js";
  6. import ReaderCollectionPrioritized from "@ui5/fs/ReaderCollectionPrioritized";
  7. import Resource from "@ui5/fs/Resource";
  8. import fsInterface from "@ui5/fs/fsInterface";
  9. /**
  10. * Returns a relative path from the given themeFolder to the root namespace.
  11. *
  12. * When combining the given themeFolder with the returned relative path it
  13. * resolves to "/resources/". However the "/resources/" part is not important
  14. * here as it doesn't exist within the theming engine environment where the
  15. * UI5 resources are part of a "UI5" folder (e.g. "UI5/sap/ui/core/") that
  16. * is next to a "Base" folder.
  17. *
  18. * @example
  19. * getPathToRoot("/resources/sap/ui/foo/themes/base")
  20. * > "../../../../../"
  21. *
  22. * @param {string} themeFolder Virtual path including /resources/
  23. * @returns {string} Relative path to root namespace
  24. */
  25. function getPathToRoot(themeFolder) {
  26. // -2 for initial "/"" and "resources/"
  27. return "../".repeat(themeFolder.split("/").length - 2);
  28. }
  29. /**
  30. * Generates an less import statement for the given <code>filePath</code>
  31. *
  32. * @param {string} filePath The path to the desired file
  33. * @returns {string} The less import statement
  34. */
  35. function lessImport(filePath) {
  36. return `@import "${filePath}";\n`;
  37. }
  38. function generateLibraryDotTheming({namespace, version, hasThemes}) {
  39. const dotTheming = {
  40. sEntity: "Library",
  41. sId: namespace,
  42. sVersion: version
  43. };
  44. // Note that with sap.ui.core version 1.127.0 the .theming file has been put into
  45. // the library sources so that "aFiles" can be maintained from there.
  46. // The below configuration is still needed for older versions of sap.ui.core which do not
  47. // contain the file.
  48. if (namespace === "sap/ui/core") {
  49. dotTheming.aFiles = [
  50. "library",
  51. "global", // Additional entry compared to UI5 root .theming
  52. "css_variables",
  53. ];
  54. }
  55. if (!hasThemes) {
  56. // Set ignore flag when there are no themes at all
  57. // This is important in case a library used to contain themes that have been removed
  58. // in a later version of the library.
  59. dotTheming.bIgnore = true;
  60. }
  61. return new Resource({
  62. path: `/resources/${namespace}/.theming`,
  63. string: JSON.stringify(dotTheming, null, 2)
  64. });
  65. }
  66. async function generateThemeDotTheming({workspace, combo, themeFolder}) {
  67. const themeName = posixPath.basename(themeFolder);
  68. const libraryMatchPattern = /^\/resources\/(.*)\/themes\/[^/]*$/i;
  69. const libraryMatch = libraryMatchPattern.exec(themeFolder);
  70. let libraryName;
  71. if (libraryMatch) {
  72. libraryName = libraryMatch[1].replace(/\//g, ".");
  73. } else {
  74. throw new Error(`Failed to extract library name from theme folder path: ${themeFolder}`);
  75. }
  76. const dotThemingTargetPath = posixPath.join(themeFolder, ".theming");
  77. if (libraryName === "sap.ui.core") {
  78. // sap.ui.core should always have a .theming file for all themes
  79. if (await workspace.byPath(dotThemingTargetPath)) {
  80. // .theming file present, skip further processing
  81. return;
  82. } else {
  83. throw new Error(`.theming file for theme ${themeName} missing in sap.ui.core library source`);
  84. }
  85. }
  86. let newDotThemingResource;
  87. const coreDotThemingResource = await combo.byPath(`/resources/sap/ui/core/themes/${themeName}/.theming`);
  88. if (coreDotThemingResource) {
  89. // Copy .theming file from core
  90. newDotThemingResource = await coreDotThemingResource.clone();
  91. newDotThemingResource.setPath(dotThemingTargetPath);
  92. } else {
  93. // No core .theming file found for this theme => Generate a .theming file
  94. const dotTheming = {
  95. sEntity: "Theme",
  96. sId: themeName,
  97. sVendor: "SAP"
  98. };
  99. if (themeName !== "base") {
  100. dotTheming.oExtends = "base";
  101. }
  102. newDotThemingResource = new Resource({
  103. path: dotThemingTargetPath,
  104. string: JSON.stringify(dotTheming, null, 2)
  105. });
  106. }
  107. return newDotThemingResource;
  108. }
  109. async function createCssVariablesLessResource({workspace, combo, themeFolder}) {
  110. const pathToRoot = getPathToRoot(themeFolder);
  111. const cssVariablesSourceLessFile = "css_variables.source.less";
  112. const cssVariablesLessFile = "css_variables.less";
  113. // posix as it is a virtual path (separated with /)
  114. const themeName = posixPath.basename(themeFolder);
  115. // The "base" theme of the baseLib is called "baseTheme"
  116. const baseLibThemeName = themeName === "base" ? "baseTheme" : themeName;
  117. // Some themes do not have a base.less file (e.g. sap_hcb)
  118. const hasBaseLess = !!(await combo.byPath(`/resources/sap/ui/core/themes/${themeName}/base.less`));
  119. let cssVariablesLess =
  120. `/* NOTE: This file was generated as an optimized version of "${cssVariablesSourceLessFile}" \
  121. for the Theme Designer. */\n\n`;
  122. if (themeName !== "base") {
  123. const cssVariablesSourceLessResource = await workspace.byPath(
  124. posixPath.join(themeFolder, cssVariablesSourceLessFile)
  125. );
  126. if (!cssVariablesSourceLessResource) {
  127. throw new Error(`Could not find file "${cssVariablesSourceLessFile}" in theme "${themeFolder}"`);
  128. }
  129. const cssVariablesSourceLess = await cssVariablesSourceLessResource.getString();
  130. cssVariablesLess += lessImport(`../base/${cssVariablesLessFile}`);
  131. cssVariablesLess += `
  132. /* START "${cssVariablesSourceLessFile}" */
  133. ${cssVariablesSourceLess}
  134. /* END "${cssVariablesSourceLessFile}" */
  135. `;
  136. }
  137. if (hasBaseLess) {
  138. cssVariablesLess += lessImport(`${pathToRoot}../Base/baseLib/${baseLibThemeName}/base.less`);
  139. }
  140. cssVariablesLess += lessImport(`${pathToRoot}sap/ui/core/themes/${themeName}/global.less`);
  141. return new Resource({
  142. path: posixPath.join(themeFolder, cssVariablesLessFile),
  143. string: cssVariablesLess
  144. });
  145. }
  146. async function generateCssVariablesLess({workspace, combo, namespace}) {
  147. let cssVariablesSourceLessResourcePattern;
  148. if (namespace) {
  149. // In case of a library only check for themes directly below the namespace
  150. cssVariablesSourceLessResourcePattern = `/resources/${namespace}/themes/*/css_variables.source.less`;
  151. } else {
  152. // In case of a theme-library check for all "themes"
  153. cssVariablesSourceLessResourcePattern = `/resources/**/themes/*/css_variables.source.less`;
  154. }
  155. const cssVariablesSourceLessResource = await workspace.byGlob(cssVariablesSourceLessResourcePattern);
  156. const hasCssVariables = cssVariablesSourceLessResource.length > 0;
  157. if (hasCssVariables) {
  158. await Promise.all(
  159. cssVariablesSourceLessResource.map(async (cssVariableSourceLess) => {
  160. const themeFolder = posixPath.dirname(cssVariableSourceLess.getPath());
  161. log.verbose(`Generating css_variables.less for theme ${themeFolder}`);
  162. const r = await createCssVariablesLessResource({
  163. workspace, combo, themeFolder
  164. });
  165. return await workspace.write(r);
  166. })
  167. );
  168. }
  169. }
  170. /**
  171. * @public
  172. * @module @ui5/builder/tasks/generateThemeDesignerResources
  173. */
  174. /* eslint "jsdoc/check-param-names": ["error", {"disableExtraPropertyReporting":true}] */
  175. /**
  176. * Generates resources required for integration with the SAP Theme Designer.
  177. *
  178. * @public
  179. * @function default
  180. * @static
  181. *
  182. * @param {object} parameters Parameters
  183. * @param {@ui5/fs/DuplexCollection} parameters.workspace DuplexCollection to read and write files
  184. * @param {@ui5/fs/AbstractReader} parameters.dependencies Reader or Collection to read dependency files
  185. * @param {object} parameters.options Options
  186. * @param {string} parameters.options.projectName Project name
  187. * @param {string} parameters.options.version Project version
  188. * @param {string} [parameters.options.projectNamespace] If the project is of type <code>library</code>,
  189. * provide its namespace.
  190. * Omit for type <code>theme-library</code>
  191. * @returns {Promise<undefined>} Promise resolving with <code>undefined</code> once data has been written
  192. */
  193. export default async function({workspace, dependencies, options}) {
  194. const {projectName, version} = options;
  195. // Backward compatibility: "namespace" option got renamed to "projectNamespace"
  196. const namespace = options.projectNamespace || options.namespace;
  197. // Skip sap.ui.documentation since it is not intended to be available in SAP Theme Designer to create custom themes
  198. if (namespace === "sap/ui/documentation") {
  199. return;
  200. }
  201. let librarySourceLessPattern;
  202. if (namespace) {
  203. // In case of a library only check for themes directly below the namespace
  204. librarySourceLessPattern = `/resources/${namespace}/themes/*/library.source.less`;
  205. } else {
  206. // In case of a theme-library check for all "themes"
  207. librarySourceLessPattern = `/resources/**/themes/*/library.source.less`;
  208. }
  209. const librarySourceLessResources = await workspace.byGlob(librarySourceLessPattern);
  210. const hasThemes = librarySourceLessResources.length > 0;
  211. // library .theming file
  212. // Only for type "library". Type "theme-library" does not provide a namespace
  213. // Also needs to be created in case a library does not have any themes (see bIgnore flag)
  214. if (namespace) {
  215. let libraryDotThemingResource;
  216. // Do not generate a .theming file for the sap.ui.core library
  217. if (namespace === "sap/ui/core") {
  218. // Check if the .theming file already exists
  219. libraryDotThemingResource = await workspace.byPath(`/resources/${namespace}/.theming`);
  220. if (libraryDotThemingResource) {
  221. // Update the existing .theming resource
  222. log.verbose(`Updating .theming for namespace ${namespace}`);
  223. await updateLibraryDotTheming({
  224. resource: libraryDotThemingResource,
  225. namespace,
  226. version,
  227. hasThemes
  228. });
  229. }
  230. }
  231. if (!libraryDotThemingResource) {
  232. log.verbose(`Generating .theming for namespace ${namespace}`);
  233. libraryDotThemingResource = generateLibraryDotTheming({
  234. namespace,
  235. version,
  236. hasThemes
  237. });
  238. }
  239. await workspace.write(libraryDotThemingResource);
  240. }
  241. if (!hasThemes) {
  242. // Skip further processing as there are no themes
  243. return;
  244. }
  245. const combo = new ReaderCollectionPrioritized({
  246. name: `generateThemeDesignerResources - prioritize workspace over dependencies: ${projectName}`,
  247. readers: [workspace, dependencies]
  248. });
  249. // theme .theming files
  250. const themeDotThemingFiles = await Promise.all(
  251. librarySourceLessResources.map((librarySourceLess) => {
  252. const themeFolder = posixPath.dirname(librarySourceLess.getPath());
  253. log.verbose(`Generating .theming for theme ${themeFolder}`);
  254. return generateThemeDotTheming({
  255. workspace, combo, themeFolder
  256. });
  257. })
  258. );
  259. await Promise.all(
  260. themeDotThemingFiles.map(async (resource) => {
  261. if (resource) {
  262. await workspace.write(resource);
  263. }
  264. })
  265. );
  266. // library.less files
  267. const libraryLessResources = await libraryLessGenerator({
  268. resources: librarySourceLessResources,
  269. fs: fsInterface(combo),
  270. });
  271. await Promise.all(
  272. libraryLessResources.map((resource) => workspace.write(resource))
  273. );
  274. // css_variables.less
  275. await generateCssVariablesLess({workspace, combo, namespace});
  276. }