Skip to content

Commit f339a7c

Browse files
committed
feat: add revalidateCookies
1 parent dcc7ed0 commit f339a7c

File tree

3 files changed

+40
-5
lines changed

3 files changed

+40
-5
lines changed

README.md

+8
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,14 @@ import { useCookie } from "@charlietango/hooks/use-cookie";
3737
const [value, setValue] = useCookie("mode");
3838
```
3939

40+
If the cookies is changed outside the `useCookie` hook, you can call the `revalidateCookies`, to get React to reevaluate the cookie values.
41+
42+
```ts
43+
import { revalidateCookies } from "@charlietango/hooks/use-cookie";
44+
45+
revalidateCookies();
46+
```
47+
4048
### `useDebouncedValue`
4149

4250
Debounce a value. The value will only be updated after the delay has passed without the value changing.

src/__tests__/useCookie.test.ts

+25-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { act, renderHook } from "@testing-library/react";
2-
import { useCookie } from "../hooks/useCookie";
2+
import { revalidateCookies, useCookie } from "../hooks/useCookie";
33

44
function setValue(
55
value: string | ((prevValue?: string) => string | undefined) | undefined,
@@ -14,8 +14,15 @@ function getValue(hook: { current: ReturnType<typeof useCookie> }) {
1414
return hook.current[0];
1515
}
1616

17+
afterEach(() => {
18+
// Clear all cookies after each test
19+
document.cookie.split(";").forEach((c) => {
20+
document.cookie = `${c.trim().split("=")[0]}=;expires=Thu, 01 Jan 1970 00:00:00 UTC;`;
21+
});
22+
});
23+
1724
test("should manage cookies", () => {
18-
const { result: hook } = renderHook(() => useCookie("test"));
25+
const { result: hook } = renderHook(() => useCookie("manage-test"));
1926

2027
setValue("custom value", hook);
2128

@@ -30,7 +37,7 @@ test("should manage cookies", () => {
3037

3138
test("should manage cookies with default value", () => {
3239
const { result: hook } = renderHook(() =>
33-
useCookie("test", { defaultValue: "default value" }),
40+
useCookie("default-value", { defaultValue: "default value" }),
3441
);
3542

3643
expect(getValue(hook)).toBe("default value");
@@ -43,11 +50,24 @@ test("should manage cookies with default value", () => {
4350
});
4451

4552
test("should sync values across hooks", () => {
46-
const { result: hook } = renderHook(() => useCookie("test"));
47-
const { result: hook2 } = renderHook(() => useCookie("test"));
53+
const { result: hook } = renderHook(() => useCookie("sync"));
54+
const { result: hook2 } = renderHook(() => useCookie("sync"));
4855

4956
setValue("new value", hook);
5057

5158
expect(getValue(hook)).toBe("new value");
5259
expect(getValue(hook2)).toBe("new value");
5360
});
61+
62+
test("should be able to revalidate cookies externally", () => {
63+
const { result: hook } = renderHook(() => useCookie("external"));
64+
document.cookie = "external=new value";
65+
expect(hook.current[0]).toBe(undefined);
66+
67+
act(() => {
68+
// Revalidate the cookies, trigger the external sync
69+
revalidateCookies();
70+
});
71+
72+
expect(hook.current[0]).toBe("new value");
73+
});

src/hooks/useCookie.ts

+7
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,13 @@ type Options = {
1313
cookieOptions?: CookieOptions;
1414
};
1515

16+
/**
17+
* Revalidate the cookies. Useful to update the cookies when they are changed outside the hook, e.g. by a server response.
18+
*/
19+
export function revalidateCookies() {
20+
trigger(SUBSCRIPTION_KEY);
21+
}
22+
1623
/**
1724
* Get or set a cookie, and update the value when it changes
1825
* @param key {string} The name of the cookie.

0 commit comments

Comments
 (0)