diff --git a/packages/dnb-design-system-portal/src/docs/uilib/components/height-animation/events.md b/packages/dnb-design-system-portal/src/docs/uilib/components/height-animation/events.md
index 21cdb369a28..77935d55e98 100644
--- a/packages/dnb-design-system-portal/src/docs/uilib/components/height-animation/events.md
+++ b/packages/dnb-design-system-portal/src/docs/uilib/components/height-animation/events.md
@@ -4,7 +4,8 @@ showTabs: true
## Events
-| Events | Description |
-| -------- | ----------------------------------------------------------------------------------------------------- |
-| `onOpen` | _(optional)_ Is called when fully opened or closed. Returns `true` or `false` depending on the state. |
-| `onOpen` | _(optional)_ Is called when fully opened or closed. Returns `true` or `false` depending on the state. |
+| Events | Description |
+| ---------------- | ---------------------------------------------------------------------------------------------------------------------------------- |
+| `onOpen` | _(optional)_ Is called when fully opened or closed. Returns `true` or `false` depending on the state. |
+| `onAnimationEnd` | _(optional)_ Is called when animation is done and the full height is reached. |
+| `onInit` | _(optional)_ Is called once before mounting the component (useLayoutEffect). Returns the instance of the internal animation class. |
diff --git a/packages/dnb-eufemia/src/components/height-animation/HeightAnimation.tsx b/packages/dnb-eufemia/src/components/height-animation/HeightAnimation.tsx
index 8b7a57b5c54..568865f1247 100644
--- a/packages/dnb-eufemia/src/components/height-animation/HeightAnimation.tsx
+++ b/packages/dnb-eufemia/src/components/height-animation/HeightAnimation.tsx
@@ -44,6 +44,7 @@ export default function HeightAnimation({
className,
innerRef,
children,
+ onInit = null,
onOpen = null,
onAnimationEnd = null,
...props
@@ -55,6 +56,7 @@ export default function HeightAnimation({
open,
animate,
children,
+ onInit,
onOpen,
onAnimationEnd,
})
diff --git a/packages/dnb-eufemia/src/components/height-animation/__tests__/HeightAnimation.test.tsx b/packages/dnb-eufemia/src/components/height-animation/__tests__/HeightAnimation.test.tsx
index 322d90c8b3f..08c901d6561 100644
--- a/packages/dnb-eufemia/src/components/height-animation/__tests__/HeightAnimation.test.tsx
+++ b/packages/dnb-eufemia/src/components/height-animation/__tests__/HeightAnimation.test.tsx
@@ -8,6 +8,7 @@ import { render, act, fireEvent } from '@testing-library/react'
import ToggleButton from '../../ToggleButton'
import { wait } from '@testing-library/user-event/dist/utils'
import HeightAnimation, { HeightAnimationProps } from '../HeightAnimation'
+import AnimateHeight from '../../../shared/AnimateHeight'
beforeEach(() => {
global.IS_TEST = false
@@ -160,6 +161,75 @@ describe('HeightAnimation', () => {
})
})
+ it('should call onOpen', () => {
+ const onOpen = jest.fn()
+ const { rerender } = render()
+
+ expect(document.querySelector('.dnb-height-animation')).toBeFalsy()
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onOpen).toHaveBeenCalledTimes(1)
+ expect(onOpen).toHaveBeenCalledWith(true)
+ })
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onOpen).toHaveBeenCalledTimes(2)
+ expect(onOpen).toHaveBeenCalledWith(false)
+ })
+ })
+
+ it('should call onAnimationEnd', () => {
+ const onAnimationEnd = jest.fn()
+ const { rerender } = render(
+
+ )
+
+ expect(document.querySelector('.dnb-height-animation')).toBeFalsy()
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onAnimationEnd).toHaveBeenCalledTimes(1)
+ expect(onAnimationEnd).toHaveBeenCalledWith('opened')
+ })
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onAnimationEnd).toHaveBeenCalledWith('closed')
+ })
+ })
+
+ it('should call onInit', () => {
+ const onInit = jest.fn()
+ const { rerender } = render()
+
+ expect(document.querySelector('.dnb-height-animation')).toBeFalsy()
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onInit).toHaveBeenCalledTimes(1)
+ expect(onInit).toHaveBeenCalledWith(expect.any(AnimateHeight))
+ })
+
+ rerender()
+
+ act(() => {
+ simulateAnimationEnd()
+ expect(onInit).toHaveBeenCalledTimes(1)
+ })
+ })
+
it('should have content in DOM when keepInDOM is true', async () => {
const { rerender } = render()
diff --git a/packages/dnb-eufemia/src/components/height-animation/useHeightAnimation.tsx b/packages/dnb-eufemia/src/components/height-animation/useHeightAnimation.tsx
index 5f084cd347f..f52c99ffad2 100644
--- a/packages/dnb-eufemia/src/components/height-animation/useHeightAnimation.tsx
+++ b/packages/dnb-eufemia/src/components/height-animation/useHeightAnimation.tsx
@@ -19,13 +19,18 @@ export type useHeightAnimationOptions = {
*/
children?: React.ReactNode | HTMLElement
+ /**
+ * Is called once before mounting the component (useLayoutEffect)
+ */
+ onInit?: (instance: AnimateHeight) => void
+
/**
* Is called when fully opened or closed
*/
onOpen?: (isOpen: boolean) => void
/**
- * Is called when animation is done and the full height has reached
+ * Is called when animation is done and the full height is reached.
*/
onAnimationEnd?: (state: HeightAnimationOnEndTypes) => void
}
@@ -43,11 +48,12 @@ export function useHeightAnimation(
open = null,
animate = true,
children = null,
+ onInit = null,
onOpen = null,
onAnimationEnd = null,
}: useHeightAnimationOptions = {}
) {
- const animRef = React.useRef(null)
+ const animRef = React.useRef(null)
const [isOpen, setIsOpen] = React.useState(open)
const [isVisible, setIsVisible] = React.useState(false)
const [isAnimating, setIsAnimating] = React.useState(false)
@@ -63,6 +69,10 @@ export function useHeightAnimation(
React.useLayoutEffect(() => {
animRef.current = new AnimateHeight({ animate })
+ if (isInitialRender && onInit) {
+ onInit(animRef.current)
+ }
+
if (isInitialRender && isOpen) {
onOpen?.(true)
}