@fremtind/jokul 5.0.0-next.1 → 5.0.0-next.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (258) hide show
  1. package/README.md +29 -6
  2. package/bin/jokul.mjs +10 -0
  3. package/bin/run-jokul-cli.mjs +63 -0
  4. package/build/build-stats.html +1 -1
  5. package/build/cjs/components/card/Card.cjs +1 -1
  6. package/build/cjs/components/card/Card.cjs.map +1 -1
  7. package/build/cjs/components/card/types.cjs +1 -1
  8. package/build/cjs/components/card/types.cjs.map +1 -1
  9. package/build/cjs/components/card/types.d.cts +2 -2
  10. package/build/cjs/components/checkbox/Checkbox.cjs +1 -1
  11. package/build/cjs/components/checkbox/Checkbox.cjs.map +1 -1
  12. package/build/cjs/components/cookie-consent/CookieConsent.cjs +1 -1
  13. package/build/cjs/components/cookie-consent/CookieConsent.cjs.map +1 -1
  14. package/build/cjs/components/cookie-consent/CookieConsent.d.cts +1 -1
  15. package/build/cjs/components/cookie-consent/types.d.cts +4 -0
  16. package/build/cjs/components/datepicker/DatePicker.cjs +1 -1
  17. package/build/cjs/components/datepicker/DatePicker.cjs.map +1 -1
  18. package/build/cjs/components/expander/ExpandablePanel.cjs +1 -1
  19. package/build/cjs/components/expander/ExpandablePanel.cjs.map +1 -1
  20. package/build/cjs/components/link/Link.cjs +1 -1
  21. package/build/cjs/components/link/Link.cjs.map +1 -1
  22. package/build/cjs/components/link/Link.d.cts +2 -2
  23. package/build/cjs/components/link/types.d.cts +1 -4
  24. package/build/cjs/components/modal/Modal.cjs +1 -1
  25. package/build/cjs/components/modal/Modal.cjs.map +1 -1
  26. package/build/cjs/components/modal/Modal.d.cts +2 -9
  27. package/build/cjs/components/modal/index.d.cts +1 -1
  28. package/build/cjs/components/modal/types.d.cts +18 -0
  29. package/build/cjs/components/popover/Popover.cjs +1 -1
  30. package/build/cjs/components/popover/Popover.cjs.map +1 -1
  31. package/build/cjs/components/radio-button/BaseRadioButton.cjs +1 -1
  32. package/build/cjs/components/radio-button/BaseRadioButton.cjs.map +1 -1
  33. package/build/cjs/core/tokens.cjs +2 -0
  34. package/build/cjs/core/tokens.cjs.map +1 -0
  35. package/build/cjs/core/tokens.d.cts +593 -0
  36. package/build/cjs/tailwind/colors.cjs +2 -0
  37. package/build/cjs/tailwind/colors.cjs.map +1 -0
  38. package/build/cjs/tailwind/colors.d.cts +39 -0
  39. package/build/cjs/tokens.cjs +1 -1
  40. package/build/cjs/tokens.cjs.map +1 -1
  41. package/build/cjs/tokens.d.cts +98 -39
  42. package/build/es/components/card/Card.js +1 -1
  43. package/build/es/components/card/Card.js.map +1 -1
  44. package/build/es/components/card/types.d.ts +2 -2
  45. package/build/es/components/card/types.js +1 -1
  46. package/build/es/components/card/types.js.map +1 -1
  47. package/build/es/components/checkbox/Checkbox.js +1 -1
  48. package/build/es/components/checkbox/Checkbox.js.map +1 -1
  49. package/build/es/components/cookie-consent/CookieConsent.d.ts +1 -1
  50. package/build/es/components/cookie-consent/CookieConsent.js +1 -1
  51. package/build/es/components/cookie-consent/CookieConsent.js.map +1 -1
  52. package/build/es/components/cookie-consent/types.d.ts +4 -0
  53. package/build/es/components/datepicker/DatePicker.js +1 -1
  54. package/build/es/components/datepicker/DatePicker.js.map +1 -1
  55. package/build/es/components/expander/ExpandablePanel.js +1 -1
  56. package/build/es/components/expander/ExpandablePanel.js.map +1 -1
  57. package/build/es/components/link/Link.d.ts +2 -2
  58. package/build/es/components/link/Link.js +1 -1
  59. package/build/es/components/link/Link.js.map +1 -1
  60. package/build/es/components/link/types.d.ts +1 -4
  61. package/build/es/components/modal/Modal.d.ts +2 -9
  62. package/build/es/components/modal/Modal.js +1 -1
  63. package/build/es/components/modal/Modal.js.map +1 -1
  64. package/build/es/components/modal/index.d.ts +1 -1
  65. package/build/es/components/modal/types.d.ts +18 -0
  66. package/build/es/components/popover/Popover.js +1 -1
  67. package/build/es/components/popover/Popover.js.map +1 -1
  68. package/build/es/components/radio-button/BaseRadioButton.js +1 -1
  69. package/build/es/components/radio-button/BaseRadioButton.js.map +1 -1
  70. package/build/es/core/tokens.d.ts +593 -0
  71. package/build/es/core/tokens.js +2 -0
  72. package/build/es/core/tokens.js.map +1 -0
  73. package/build/es/tailwind/colors.d.ts +39 -0
  74. package/build/es/tailwind/colors.js +2 -0
  75. package/build/es/tailwind/colors.js.map +1 -0
  76. package/build/es/tokens.d.ts +98 -39
  77. package/build/es/tokens.js +1 -1
  78. package/build/es/tokens.js.map +1 -1
  79. package/codemods/__tests__/import-paths.test.mjs +84 -0
  80. package/codemods/import-paths.mjs +393 -0
  81. package/package.json +6 -1
  82. package/styles/base.css +483 -66
  83. package/styles/base.min.css +1 -1
  84. package/styles/components/autosuggest/autosuggest.css +2 -2
  85. package/styles/components/autosuggest/autosuggest.min.css +1 -1
  86. package/styles/components/autosuggest/autosuggest.scss +2 -2
  87. package/styles/components/beta/description-list/description-list.css +1 -1
  88. package/styles/components/beta/description-list/description-list.min.css +1 -1
  89. package/styles/components/beta/description-list/description-list.scss +1 -1
  90. package/styles/components/beta/nav-link/navlink.css +2 -2
  91. package/styles/components/beta/nav-link/navlink.min.css +1 -1
  92. package/styles/components/beta/nav-link/navlink.scss +2 -2
  93. package/styles/components/beta/select/select.css +9 -9
  94. package/styles/components/beta/select/select.min.css +1 -1
  95. package/styles/components/beta/select/select.scss +8 -7
  96. package/styles/components/breadcrumb/breadcrumb.css +1 -1
  97. package/styles/components/breadcrumb/breadcrumb.min.css +1 -1
  98. package/styles/components/button/button.css +6 -2
  99. package/styles/components/button/button.min.css +1 -1
  100. package/styles/components/button/button.scss +12 -12
  101. package/styles/components/card/card.css +6 -9
  102. package/styles/components/card/card.min.css +1 -1
  103. package/styles/components/card/card.scss +6 -10
  104. package/styles/components/checkbox/checkbox.css +43 -9
  105. package/styles/components/checkbox/checkbox.min.css +1 -1
  106. package/styles/components/checkbox/checkbox.scss +38 -21
  107. package/styles/components/checkbox-panel/checkbox-panel.css +65 -26
  108. package/styles/components/checkbox-panel/checkbox-panel.min.css +1 -1
  109. package/styles/components/checkbox-panel/checkbox-panel.scss +4 -0
  110. package/styles/components/chip/chip.css +2 -2
  111. package/styles/components/chip/chip.min.css +1 -1
  112. package/styles/components/chip/chip.scss +1 -1
  113. package/styles/components/combobox/combobox.css +18 -15
  114. package/styles/components/combobox/combobox.min.css +1 -1
  115. package/styles/components/combobox/combobox.scss +9 -6
  116. package/styles/components/countdown/countdown.css +4 -4
  117. package/styles/components/countdown/countdown.min.css +1 -1
  118. package/styles/components/datepicker/_calendar-date-button.scss +7 -7
  119. package/styles/components/datepicker/_calendar-navigation-dropdown.scss +1 -1
  120. package/styles/components/datepicker/_calendar.scss +6 -7
  121. package/styles/components/datepicker/datepicker.css +19 -11
  122. package/styles/components/datepicker/datepicker.min.css +1 -1
  123. package/styles/components/description-list/description-list.css +1 -1
  124. package/styles/components/description-list/description-list.min.css +1 -1
  125. package/styles/components/description-list/description-list.scss +1 -1
  126. package/styles/components/expander/expandable.css +17 -17
  127. package/styles/components/expander/expandable.min.css +1 -1
  128. package/styles/components/expander/expandable.scss +15 -19
  129. package/styles/components/feedback/feedback.css +6 -8
  130. package/styles/components/feedback/feedback.min.css +1 -1
  131. package/styles/components/feedback/feedback.scss +4 -9
  132. package/styles/components/file/file.css +5 -5
  133. package/styles/components/file/file.min.css +1 -1
  134. package/styles/components/file/file.scss +5 -5
  135. package/styles/components/file-input/file-input.css +26 -22
  136. package/styles/components/file-input/file-input.min.css +1 -1
  137. package/styles/components/file-input/file-input.scss +3 -3
  138. package/styles/components/help/help.css +2 -2
  139. package/styles/components/help/help.min.css +1 -1
  140. package/styles/components/help/help.scss +2 -2
  141. package/styles/components/icon/icon.css +4 -4
  142. package/styles/components/icon/icon.min.css +1 -1
  143. package/styles/components/icon/icon.scss +4 -4
  144. package/styles/components/icon-button/icon-button.css +1 -1
  145. package/styles/components/icon-button/icon-button.min.css +1 -1
  146. package/styles/components/input-group/input-group.css +2 -2
  147. package/styles/components/input-group/input-group.min.css +1 -1
  148. package/styles/components/input-panel/input-panel.css +19 -17
  149. package/styles/components/input-panel/input-panel.min.css +1 -1
  150. package/styles/components/input-panel/input-panel.scss +20 -18
  151. package/styles/components/link/link.css +1 -1
  152. package/styles/components/link/link.min.css +1 -1
  153. package/styles/components/link-list/link-list.css +6 -2
  154. package/styles/components/link-list/link-list.min.css +1 -1
  155. package/styles/components/link-list/link-list.scss +6 -2
  156. package/styles/components/loader/loader.css +6 -6
  157. package/styles/components/loader/loader.min.css +1 -1
  158. package/styles/components/loader/skeleton-loader.css +4 -4
  159. package/styles/components/loader/skeleton-loader.min.css +1 -1
  160. package/styles/components/loader/skeleton-loader.scss +1 -1
  161. package/styles/components/menu/_menu-divider.scss +1 -1
  162. package/styles/components/menu/menu.css +3 -3
  163. package/styles/components/menu/menu.min.css +1 -1
  164. package/styles/components/menu/menu.scss +2 -2
  165. package/styles/components/message/message.css +19 -9
  166. package/styles/components/message/message.min.css +1 -1
  167. package/styles/components/message/message.scss +16 -6
  168. package/styles/components/modal/_layout.scss +22 -0
  169. package/styles/components/modal/_modal-base.scss +32 -0
  170. package/styles/components/modal/_motion.scss +45 -0
  171. package/styles/components/modal/_overlay.scss +20 -0
  172. package/styles/components/modal/_parts.scss +33 -0
  173. package/styles/components/modal/_placement.scss +59 -0
  174. package/styles/components/modal/modal.css +118 -34
  175. package/styles/components/modal/modal.min.css +1 -1
  176. package/styles/components/modal/modal.scss +6 -95
  177. package/styles/components/nav-link/nav-link.css +1 -1
  178. package/styles/components/nav-link/nav-link.min.css +1 -1
  179. package/styles/components/pagination/pagination.css +1 -1
  180. package/styles/components/pagination/pagination.min.css +1 -1
  181. package/styles/components/popover/popover.css +12 -1
  182. package/styles/components/popover/popover.min.css +1 -1
  183. package/styles/components/popover/popover.scss +15 -1
  184. package/styles/components/progress-bar/progress-bar.css +27 -3
  185. package/styles/components/progress-bar/progress-bar.min.css +1 -1
  186. package/styles/components/progress-bar/progress-bar.scss +5 -0
  187. package/styles/components/radio-button/radio-button.css +41 -6
  188. package/styles/components/radio-button/radio-button.min.css +1 -1
  189. package/styles/components/radio-button/radio-button.scss +35 -16
  190. package/styles/components/radio-panel/radio-panel.css +22 -17
  191. package/styles/components/radio-panel/radio-panel.min.css +1 -1
  192. package/styles/components/radio-panel/radio-panel.scss +4 -0
  193. package/styles/components/search/search-with-submit-button.css +1 -1
  194. package/styles/components/search/search-with-submit-button.min.css +1 -1
  195. package/styles/components/search/search-with-submit-button.scss +1 -1
  196. package/styles/components/search/search.css +2 -2
  197. package/styles/components/search/search.min.css +1 -1
  198. package/styles/components/search/search.scss +1 -1
  199. package/styles/components/segmented-control/segmented-control.css +54 -19
  200. package/styles/components/segmented-control/segmented-control.min.css +1 -1
  201. package/styles/components/segmented-control/segmented-control.scss +4 -4
  202. package/styles/components/select/select.css +15 -15
  203. package/styles/components/select/select.min.css +1 -1
  204. package/styles/components/select/select.scss +12 -12
  205. package/styles/components/summary-table/summary-table.css +2 -2
  206. package/styles/components/summary-table/summary-table.min.css +1 -1
  207. package/styles/components/summary-table/summary-table.scss +2 -2
  208. package/styles/components/system-message/system-message.css +20 -10
  209. package/styles/components/system-message/system-message.min.css +1 -1
  210. package/styles/components/system-message/system-message.scss +17 -7
  211. package/styles/components/table/_table-head.scss +1 -1
  212. package/styles/components/table/_table-row.scss +2 -2
  213. package/styles/components/table/table.css +3 -3
  214. package/styles/components/table/table.min.css +1 -1
  215. package/styles/components/table-of-contents/table-of-contents.css +1 -1
  216. package/styles/components/table-of-contents/table-of-contents.min.css +1 -1
  217. package/styles/components/table-of-contents/table-of-contents.scss +1 -1
  218. package/styles/components/tabs/tabs.css +3 -3
  219. package/styles/components/tabs/tabs.min.css +1 -1
  220. package/styles/components/tabs/tabs.scss +2 -2
  221. package/styles/components/tag/tag.css +16 -6
  222. package/styles/components/tag/tag.min.css +1 -1
  223. package/styles/components/tag/tag.scss +16 -6
  224. package/styles/components/text-area/text-area.css +8 -8
  225. package/styles/components/text-area/text-area.min.css +1 -1
  226. package/styles/components/text-input/text-input.css +8 -8
  227. package/styles/components/text-input/text-input.min.css +1 -1
  228. package/styles/components/toast/toast.css +19 -12
  229. package/styles/components/toast/toast.min.css +1 -1
  230. package/styles/components/toast/toast.scss +15 -12
  231. package/styles/components/toggle-switch/_toggle-slider.scss +4 -4
  232. package/styles/components/toggle-switch/toggle-switch.css +14 -19
  233. package/styles/components/toggle-switch/toggle-switch.min.css +2 -2
  234. package/styles/components/toggle-switch/toggle-switch.scss +9 -16
  235. package/styles/components/tooltip/tooltip.css +3 -3
  236. package/styles/components/tooltip/tooltip.min.css +1 -1
  237. package/styles/components/tooltip/tooltip.scss +3 -3
  238. package/styles/components.css +536 -294
  239. package/styles/components.min.css +2 -2
  240. package/styles/core/jkl/_tokens.scss +59 -0
  241. package/styles/core/jkl/legacy/_dynamic-colors.scss +40 -0
  242. package/styles/core/jkl/legacy/_tokens.scss +742 -0
  243. package/styles/core/theme/_color-tokens.scss +73 -0
  244. package/styles/core/theme/_legacy-color-tokens.scss +81 -0
  245. package/styles/core/theme/_legacy-tokens.scss +279 -0
  246. package/styles/core/theme/_spacing-tokens.scss +33 -0
  247. package/styles/core/theme/_tokens.scss +33 -0
  248. package/styles/hooks/stories/styles.scss +2 -2
  249. package/styles/jkl/_ornaments.scss +1 -1
  250. package/styles/jkl/_tokens.scss +151 -71
  251. package/styles/shared/input/shared-input-styles.scss +6 -6
  252. package/styles/shared/track/track.scss +2 -2
  253. package/styles/tailwind.css +108 -66
  254. package/styles/theme/_color-scheme.scss +135 -66
  255. package/styles/theme/_index.scss +3 -0
  256. package/styles/theme/brands/dnb/_color-scheme.scss +119 -0
  257. package/styles/theme/brands/eika/_color-scheme.scss +119 -0
  258. package/styles/theme/brands/sparebank1/_color-scheme.scss +119 -0
@@ -0,0 +1,393 @@
1
+ import { readdir, readFile, stat, writeFile } from "node:fs/promises";
2
+ import path from "node:path";
3
+
4
+ const TEXT_EXTENSIONS = new Set([
5
+ ".cjs",
6
+ ".css",
7
+ ".cts",
8
+ ".js",
9
+ ".jsx",
10
+ ".md",
11
+ ".mdx",
12
+ ".mjs",
13
+ ".mts",
14
+ ".sass",
15
+ ".scss",
16
+ ".ts",
17
+ ".tsx",
18
+ ]);
19
+
20
+ const IGNORED_DIRECTORIES = new Set([
21
+ ".changeset",
22
+ ".git",
23
+ ".github",
24
+ ".next",
25
+ ".turbo",
26
+ "build",
27
+ "coverage",
28
+ "dist",
29
+ "node_modules",
30
+ "storybook-static",
31
+ ]);
32
+
33
+ const IGNORED_FILE_PATTERNS = [
34
+ "/packages/jokul/CHANGELOG.md",
35
+ "/packages/jokul/MIGRATION.md",
36
+ "/packages/jokul/build-styles.mjs",
37
+ "/packages/jokul/bin/",
38
+ "/packages/jokul/codemods/",
39
+ ];
40
+
41
+ const DIRECT_REPLACEMENTS = [
42
+ [
43
+ "@fremtind/jokul/styles/core/core.min.css",
44
+ "@fremtind/jokul/styles/base.min.css",
45
+ ],
46
+ [
47
+ "@fremtind/jokul/styles/core/core.css",
48
+ "@fremtind/jokul/styles/base.css",
49
+ ],
50
+ [
51
+ "@fremtind/jokul/styles/core/core.scss",
52
+ "@fremtind/jokul/styles/base.scss",
53
+ ],
54
+ ["@fremtind/jokul/styles/core/core", "@fremtind/jokul/styles/base"],
55
+ [
56
+ "@fremtind/jokul/styles/styles.min.css",
57
+ "@fremtind/jokul/styles/components.min.css",
58
+ ],
59
+ ["@fremtind/jokul/styles/styles.css", "@fremtind/jokul/styles/components.css"],
60
+ [
61
+ "@fremtind/jokul/styles/styles.scss",
62
+ "@fremtind/jokul/styles/components.scss",
63
+ ],
64
+ ["@fremtind/jokul/styles/styles", "@fremtind/jokul/styles/components"],
65
+ [
66
+ "@fremtind/jokul/styles/core/jkl/index",
67
+ "@fremtind/jokul/styles/jkl",
68
+ ],
69
+ ["@fremtind/jokul/styles/core/jkl", "@fremtind/jokul/styles/jkl"],
70
+ [
71
+ "@fremtind/jokul/styles/fonts/webfonts.scss",
72
+ "@fremtind/jokul/styles/theme/fonts",
73
+ ],
74
+ [
75
+ "@fremtind/jokul/styles/fonts/webfonts",
76
+ "@fremtind/jokul/styles/theme/fonts",
77
+ ],
78
+ ["@fremtind/jokul/styles/fonts", "@fremtind/jokul/styles/theme/fonts"],
79
+ ["../../../core/jkl/index", "../../../styles/jkl"],
80
+ ["../../../core/jkl", "../../../styles/jkl"],
81
+ ["../../core/jkl/index", "../../styles/jkl"],
82
+ ["../../core/jkl", "../../styles/jkl"],
83
+ ["../core/jkl/index", "../styles/jkl"],
84
+ ["../core/jkl", "../styles/jkl"],
85
+ ["@fremtind/jokul/tailwind/v4", "@fremtind/jokul/styles/tailwind"],
86
+ ["@fremtind/jokul/styles/core", "@fremtind/jokul/styles/base.scss"],
87
+ ["@fremtind/jokul/styles", "@fremtind/jokul/styles/components.scss"],
88
+ ["@fremtind/jokul/core", "@fremtind/jokul/utilities"],
89
+ ].sort(([a], [b]) => b.length - a.length);
90
+
91
+ const BETA_STYLE_MIGRATIONS = [
92
+ {
93
+ component: "DescriptionList",
94
+ betaIdentifiers: ["BETA_DescriptionList", "BETA_DescriptionListItem"],
95
+ replacements: [
96
+ [
97
+ "@fremtind/jokul/styles/components/description-list/_index.scss",
98
+ "@fremtind/jokul/styles/components/beta/description-list/_index.scss",
99
+ ],
100
+ [
101
+ "@fremtind/jokul/styles/components/description-list/description-list.scss",
102
+ "@fremtind/jokul/styles/components/beta/description-list/description-list.scss",
103
+ ],
104
+ [
105
+ "@fremtind/jokul/styles/components/description-list",
106
+ "@fremtind/jokul/styles/components/beta/description-list",
107
+ ],
108
+ ],
109
+ },
110
+ {
111
+ component: "NavLink",
112
+ betaIdentifiers: ["BETA_NavLink"],
113
+ replacements: [
114
+ [
115
+ "@fremtind/jokul/styles/components/nav-link/_index.scss",
116
+ "@fremtind/jokul/styles/components/beta/nav-link/_index.scss",
117
+ ],
118
+ [
119
+ "@fremtind/jokul/styles/components/nav-link/nav-link.scss",
120
+ "@fremtind/jokul/styles/components/beta/nav-link/navlink.scss",
121
+ ],
122
+ [
123
+ "@fremtind/jokul/styles/components/nav-link",
124
+ "@fremtind/jokul/styles/components/beta/nav-link",
125
+ ],
126
+ ],
127
+ },
128
+ {
129
+ component: "Select",
130
+ betaIdentifiers: ["BETA_Select"],
131
+ replacements: [
132
+ [
133
+ "@fremtind/jokul/styles/components/select/_index.scss",
134
+ "@fremtind/jokul/styles/components/beta/select/_index.scss",
135
+ ],
136
+ [
137
+ "@fremtind/jokul/styles/components/select/select.scss",
138
+ "@fremtind/jokul/styles/components/beta/select/select.scss",
139
+ ],
140
+ [
141
+ "@fremtind/jokul/styles/components/select",
142
+ "@fremtind/jokul/styles/components/beta/select",
143
+ ],
144
+ ],
145
+ },
146
+ ];
147
+
148
+ function shouldIgnoreFile(filePath) {
149
+ const normalizedPath = filePath.split(path.sep).join("/");
150
+
151
+ return IGNORED_FILE_PATTERNS.some((pattern) =>
152
+ normalizedPath.includes(pattern),
153
+ );
154
+ }
155
+
156
+ function shouldIgnoreDirectory(directoryPath) {
157
+ return IGNORED_DIRECTORIES.has(path.basename(directoryPath));
158
+ }
159
+
160
+ function escapeRegExp(value) {
161
+ return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
162
+ }
163
+
164
+ function replaceSpecifier(text, from, to) {
165
+ const pattern = new RegExp(
166
+ `(?<![A-Za-z0-9_./-])${escapeRegExp(from)}(?![A-Za-z0-9_./-])`,
167
+ "g",
168
+ );
169
+
170
+ let count = 0;
171
+ const next = text.replace(pattern, () => {
172
+ count += 1;
173
+ return to;
174
+ });
175
+
176
+ return { text: next, count };
177
+ }
178
+
179
+ function applyDirectReplacements(text) {
180
+ let next = text;
181
+ let replacements = 0;
182
+
183
+ for (const [from, to] of DIRECT_REPLACEMENTS) {
184
+ const result = replaceSpecifier(next, from, to);
185
+ next = result.text;
186
+ replacements += result.count;
187
+ }
188
+
189
+ return { text: next, replacements };
190
+ }
191
+
192
+ function applyBetaStyleReplacements(text) {
193
+ let next = text;
194
+ let replacements = 0;
195
+ const warnings = [];
196
+
197
+ for (const migration of BETA_STYLE_MIGRATIONS) {
198
+ const mentionsBeta = migration.betaIdentifiers.some((identifier) =>
199
+ next.includes(identifier),
200
+ );
201
+
202
+ const hasOldSpecifier = migration.replacements.some(([from]) =>
203
+ new RegExp(
204
+ `(?<![A-Za-z0-9_./-])${escapeRegExp(from)}(?![A-Za-z0-9_./-])`,
205
+ ).test(next),
206
+ );
207
+
208
+ if (!hasOldSpecifier) {
209
+ continue;
210
+ }
211
+
212
+ if (!mentionsBeta) {
213
+ warnings.push(
214
+ `Manuell vurdering: gammel stilimport for ${migration.component} kan peke på enten stabil eller beta-variant.`,
215
+ );
216
+ continue;
217
+ }
218
+
219
+ for (const [from, to] of migration.replacements) {
220
+ const result = replaceSpecifier(next, from, to);
221
+ next = result.text;
222
+ replacements += result.count;
223
+ }
224
+ }
225
+
226
+ return { text: next, replacements, warnings };
227
+ }
228
+
229
+ function reorderConfiguredFontImport(text) {
230
+ const fontImportPattern =
231
+ /^@use\s+["']@fremtind\/jokul\/styles\/theme\/fonts["'][\s\S]*?;\s*/m;
232
+ const baseImportPattern =
233
+ /^@use\s+["']@fremtind\/jokul\/styles\/base(?:\.scss)?["'][^;]*;\s*/m;
234
+
235
+ const fontMatch = fontImportPattern.exec(text);
236
+ const baseMatch = baseImportPattern.exec(text);
237
+
238
+ if (!fontMatch || !baseMatch || fontMatch.index < baseMatch.index) {
239
+ return { text, reordered: false };
240
+ }
241
+
242
+ const withoutFontImport =
243
+ text.slice(0, fontMatch.index) +
244
+ text.slice(fontMatch.index + fontMatch[0].length);
245
+ const nextBaseMatch = baseImportPattern.exec(withoutFontImport);
246
+
247
+ if (!nextBaseMatch) {
248
+ return { text, reordered: false };
249
+ }
250
+
251
+ const nextText =
252
+ withoutFontImport.slice(0, nextBaseMatch.index) +
253
+ fontMatch[0] +
254
+ withoutFontImport.slice(nextBaseMatch.index);
255
+
256
+ return { text: nextText, reordered: true };
257
+ }
258
+
259
+ export function transformImportPaths(text, filePath = "") {
260
+ const direct = applyDirectReplacements(text);
261
+ const beta = applyBetaStyleReplacements(direct.text);
262
+ let next = beta.text;
263
+ let reordered = false;
264
+
265
+ if (/\.(sass|scss)$/i.test(filePath)) {
266
+ const reorderedResult = reorderConfiguredFontImport(next);
267
+ next = reorderedResult.text;
268
+ reordered = reorderedResult.reordered;
269
+ }
270
+
271
+ return {
272
+ text: next,
273
+ changed: next !== text,
274
+ replacements: direct.replacements + beta.replacements,
275
+ warnings: beta.warnings,
276
+ reordered,
277
+ };
278
+ }
279
+
280
+ async function collectFiles(targetPath, collected) {
281
+ const stats = await stat(targetPath);
282
+
283
+ if (stats.isDirectory()) {
284
+ if (shouldIgnoreDirectory(targetPath)) {
285
+ return collected;
286
+ }
287
+
288
+ const entries = await readdir(targetPath, { withFileTypes: true });
289
+ const sortedEntries = [...entries].sort((a, b) =>
290
+ a.name.localeCompare(b.name),
291
+ );
292
+
293
+ for (const entry of sortedEntries) {
294
+ if (entry.isDirectory() && IGNORED_DIRECTORIES.has(entry.name)) {
295
+ continue;
296
+ }
297
+
298
+ await collectFiles(path.join(targetPath, entry.name), collected);
299
+ }
300
+
301
+ return collected;
302
+ }
303
+
304
+ if (TEXT_EXTENSIONS.has(path.extname(targetPath)) && !shouldIgnoreFile(targetPath)) {
305
+ collected.push(targetPath);
306
+ }
307
+
308
+ return collected;
309
+ }
310
+
311
+ function parseArguments(rawArgs) {
312
+ const options = {
313
+ dryRun: false,
314
+ verbose: false,
315
+ };
316
+ const targets = [];
317
+
318
+ for (const arg of rawArgs) {
319
+ if (arg === "--dry-run") {
320
+ options.dryRun = true;
321
+ continue;
322
+ }
323
+
324
+ if (arg === "--verbose") {
325
+ options.verbose = true;
326
+ continue;
327
+ }
328
+
329
+ if (arg === "--help" || arg === "-h") {
330
+ throw new Error(
331
+ "Bruk: jokul codemod [import-paths] [sti ...] [--dry-run] [--verbose]",
332
+ );
333
+ }
334
+
335
+ targets.push(arg);
336
+ }
337
+
338
+ return {
339
+ options,
340
+ targets: targets.length > 0 ? targets : ["."],
341
+ };
342
+ }
343
+
344
+ export async function runImportPathsCodemod(rawArgs = []) {
345
+ const { options, targets } = parseArguments(rawArgs);
346
+ const files = [];
347
+
348
+ for (const target of targets) {
349
+ await collectFiles(path.resolve(target), files);
350
+ }
351
+
352
+ const uniqueFiles = [...new Set(files)].sort((a, b) => a.localeCompare(b));
353
+ let changedFiles = 0;
354
+ let changedSpecifiers = 0;
355
+ let reorderedFiles = 0;
356
+ const warnings = [];
357
+
358
+ for (const filePath of uniqueFiles) {
359
+ const source = await readFile(filePath, "utf-8");
360
+ const result = transformImportPaths(source, filePath);
361
+
362
+ for (const warning of result.warnings) {
363
+ warnings.push(`${filePath}: ${warning}`);
364
+ }
365
+
366
+ if (!result.changed) {
367
+ continue;
368
+ }
369
+
370
+ changedFiles += 1;
371
+ changedSpecifiers += result.replacements;
372
+ reorderedFiles += result.reordered ? 1 : 0;
373
+
374
+ if (!options.dryRun) {
375
+ await writeFile(filePath, result.text, "utf-8");
376
+ }
377
+
378
+ if (options.verbose || options.dryRun) {
379
+ console.log(`${options.dryRun ? "Ville endret" : "Endret"} ${filePath}`);
380
+ }
381
+ }
382
+
383
+ if (warnings.length > 0) {
384
+ console.warn("\nFiler som trenger manuell oppfølging:");
385
+ for (const warning of warnings) {
386
+ console.warn(`- ${warning}`);
387
+ }
388
+ }
389
+
390
+ console.log(
391
+ `\n${options.dryRun ? "Dry run ferdig" : "Codemod ferdig"}: ${changedFiles} filer, ${changedSpecifiers} erstattede imports${reorderedFiles > 0 ? `, ${reorderedFiles} justerte font-importrekkefølger` : ""}.`,
392
+ );
393
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@fremtind/jokul",
3
- "version": "5.0.0-next.1",
3
+ "version": "5.0.0-next.2",
4
4
  "type": "module",
5
5
  "publishConfig": {
6
6
  "access": "public"
@@ -14,11 +14,16 @@
14
14
  "*.scss"
15
15
  ],
16
16
  "files": [
17
+ "./bin",
17
18
  "./build",
19
+ "./codemods",
18
20
  "./styles",
19
21
  "./src/fonts",
20
22
  "./package.json"
21
23
  ],
24
+ "bin": {
25
+ "jokul": "./bin/jokul.mjs"
26
+ },
22
27
  "exports": {
23
28
  "./package.json": "./package.json",
24
29
  "./styles/*": "./styles/*",