Skip to content

Commit 1cc24c2

Browse files
authored
[Darkside] Added hook for renaming className-prefix from navds-> aksel- (#3585)
* ✨ MVP replacement function * ✨ Updated className replace function
1 parent 636eb05 commit 1cc24c2

File tree

3 files changed

+128
-9
lines changed

3 files changed

+128
-9
lines changed

@navikt/core/react/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -620,7 +620,7 @@
620620
"build": "yarn i18n-jsdoc && concurrently \"tsc -p tsconfig.build.json\" \"tsc -p tsconfig.esm.json && tsc-alias -p tsconfig.esm.json && yarn write-packagejson\" && yarn i18n-jsdoc --cleanup",
621621
"watch": "tsc --watch -p tsconfig.esm.json",
622622
"test": "TZ=UTC vitest run -c tests/vitest.config.ts",
623-
"test:watch": "vitest watch"
623+
"test:watch": "TZ=UTC vitest watch -c tests/vitest.config.ts"
624624
},
625625
"dependencies": {
626626
"@floating-ui/react": "0.25.4",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
import { describe, expect, test } from "vitest";
2+
import { compositeClassFunction } from "./Theme";
3+
4+
const start = "navds-accordion testclass navds-heading endclass";
5+
const end =
6+
"startclass navds-accordion navds-heading endclass navds-accordion__header";
7+
8+
const both = start + " " + end;
9+
10+
describe("RenameCSS", () => {
11+
test("String starts with navds", () => {
12+
expect(compositeClassFunction(start)).toBe(
13+
"aksel-accordion testclass aksel-heading endclass",
14+
);
15+
});
16+
17+
test("String ends with navds", () => {
18+
expect(compositeClassFunction(end)).toBe(
19+
"startclass aksel-accordion aksel-heading endclass aksel-accordion__header",
20+
);
21+
});
22+
23+
test("String starts and ends with navds", () => {
24+
expect(compositeClassFunction(both)).toBe(
25+
"aksel-accordion testclass aksel-heading endclass startclass aksel-accordion aksel-heading endclass aksel-accordion__header",
26+
);
27+
});
28+
29+
test("String does not contain navds", () => {
30+
expect(compositeClassFunction("startclass endclass")).toBe(
31+
"startclass endclass",
32+
);
33+
});
34+
35+
test("String is empty", () => {
36+
expect(compositeClassFunction("")).toBe("");
37+
});
38+
39+
test("String contains navds as a substring", () => {
40+
expect(compositeClassFunction("startclass test-navds-class endclass")).toBe(
41+
"startclass test-navds-class endclass",
42+
);
43+
});
44+
45+
test("String contains multiple navds occurrences", () => {
46+
const input = "navds-button navds-icon navds-label";
47+
const expected = "aksel-button aksel-icon aksel-label";
48+
expect(compositeClassFunction(input)).toBe(expected);
49+
});
50+
51+
test("String contains navds with special characters", () => {
52+
const input = "navds-button! navds-icon@ navds-#label navds#-label";
53+
const expected = "aksel-button! aksel-icon@ aksel-#label navds#-label";
54+
expect(compositeClassFunction(input)).toBe(expected);
55+
});
56+
57+
test("String contains mixed navds and non-navds classes", () => {
58+
const input = "navds-button custom-class navds-icon";
59+
const expected = "aksel-button custom-class aksel-icon";
60+
expect(compositeClassFunction(input)).toBe(expected);
61+
});
62+
63+
test("String contains only navds", () => {
64+
const input = "navds";
65+
const expected = "navds";
66+
expect(compositeClassFunction(input)).toBe(expected);
67+
});
68+
69+
test("String contains navds with numbers", () => {
70+
const input = "navds-1 navds-2 navds-3";
71+
const expected = "aksel-1 aksel-2 aksel-3";
72+
expect(compositeClassFunction(input)).toBe(expected);
73+
});
74+
75+
test("String contains different casings", () => {
76+
const input = "Navds-button NAVds-icon NAVDS-label navDS-component";
77+
const expected = "Navds-button NAVds-icon NAVDS-label navDS-component";
78+
expect(compositeClassFunction(input)).toBe(expected);
79+
});
80+
});

@navikt/core/react/src/theme/Theme.tsx

+47-8
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,43 @@ import { Slot } from "../slot/Slot";
44
import { createContext } from "../util/create-context";
55
import { AsChildProps } from "../util/types";
66

7+
/* -------------------------------------------------------------------------- */
8+
/* CSS Trsnalation */
9+
/* -------------------------------------------------------------------------- */
10+
type RenameCSSContext = {
11+
cn: (...inputs: Parameters<typeof cl>) => ReturnType<typeof cl>;
12+
};
13+
14+
const [RenameCSSProvider, useRenameCSS] = createContext<RenameCSSContext>({
15+
hookName: "useRenameCSS",
16+
name: "RenameCSS",
17+
providerName: "RenameCSSProvider",
18+
defaultValue: { cn: cl },
19+
});
20+
21+
export const compositeClassFunction = (
22+
...inputs: Parameters<typeof cl>
23+
): string => {
24+
const classes = cl(inputs)
25+
/* Replaces only if start of string "navds- navds-"*/
26+
.replace(/^navds-/g, "aksel-")
27+
/* Replaces all " navds-" hits */
28+
.replace(/\snavds-/g, " aksel-");
29+
30+
return classes.trim();
31+
};
32+
33+
const RenameCSS = ({ children }: { children: React.ReactNode }) => {
34+
return (
35+
<RenameCSSProvider cn={compositeClassFunction}>
36+
{children}
37+
</RenameCSSProvider>
38+
);
39+
};
40+
41+
/* -------------------------------------------------------------------------- */
42+
/* Theme provider */
43+
/* -------------------------------------------------------------------------- */
744
type ThemeContext = {
845
/**
946
* Color theme
@@ -45,16 +82,18 @@ const Theme = forwardRef<HTMLDivElement, ThemeProps>(
4582

4683
return (
4784
<ThemeProvider theme={theme}>
48-
<SlotElement
49-
ref={ref}
50-
className={cl("navds-theme", className, theme)}
51-
data-background={hasBackground}
52-
>
53-
{children}
54-
</SlotElement>
85+
<RenameCSS>
86+
<SlotElement
87+
ref={ref}
88+
className={cl("navds-theme", className, theme)}
89+
data-background={hasBackground}
90+
>
91+
{children}
92+
</SlotElement>
93+
</RenameCSS>
5594
</ThemeProvider>
5695
);
5796
},
5897
);
5998

60-
export { Theme, useThemeInternal };
99+
export { Theme, useRenameCSS, useThemeInternal };

0 commit comments

Comments
 (0)