Skip to content
Snippets Groups Projects
Commit a72ed063 authored by Matthias Konitzny's avatar Matthias Konitzny :fire:
Browse files

Added collapsible component

parent 9eb56cf6
No related branches found
No related tags found
1 merge request!3Master into new editor
Pipeline #56790 passed
.collapsible-content {
overflow: hidden;
transition: height 0.2s ease-in-out;
border-bottom: 1px solid #dee2e6 !important;
}
.collapsible-content-padding {
padding: 20px 20px 20px 30px;
}
.collapsible-rotate-center {
-moz-transition: all 0.2s linear;
-webkit-transition: all 0.2s linear;
font-size: 4em;
transition: all 0.2s linear;
}
.collapsible-rotate-center.down {
transform: rotate(-90deg);
}
.collapsible-rotate-center.up {
transform: rotate(90deg);
}
.collapsible-button {
position: relative;
display: flex;
align-items: center;
width: 100%;
font-size: 1.2rem;
background-color: transparent;
border-bottom: 1px solid gray;
color: black;
padding-left: 25px;
}
.collapsible-button:not(.collapsed) {
background-color: aliceblue;
color: royalblue;
}
.collapsible-button.collapsed:hover {
background-color: transparent;
}
.collapsible-button:hover {
background-color: aliceblue;
}
.collapsible-button.collapsed:focus {
border-color: lightblue;
background-color: white;
}
.collapsible-button.collapsed:focus {
border-color: lightblue;
}
.collapsible-button::after {
content: "\276E";
margin-left: auto;
transition: all 0.2s linear;
transform: rotate(90deg);
}
.collapsible-button:not(.collapsed)::after {
transform: rotate(-90deg);
}
.collapsible-card {
box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2);
transition: 0.3s;
}
.collapsible-card:hover {
box-shadow: 0 8px 16px 0 rgba(0, 0, 0, 0.2);
}
import React, { useEffect, useRef, useState } from "react";
import "./collapsible.css";
interface CollapsibleProps {
open?: boolean;
header: string | React.ReactNode;
children?: React.ReactNode | React.ReactNode[];
}
// Implementation details at nhttps://medium.com/edonec/build-a-react-collapsible-component-from-scratch-using-react-hooks-typescript-73dfd02c9208
function Collapsible({ open, header, children }: CollapsibleProps) {
const [isOpen, setIsOpen] = useState(open);
const [height, setHeight] = useState<number | undefined>(
open ? undefined : 0
);
const toggleOpen = () => {
setIsOpen((prev) => !prev);
};
const ref = useRef<HTMLDivElement>(null);
useEffect(() => {
if (!height || !isOpen || !ref.current) return undefined;
const resizeObserver = new ResizeObserver((el) => {
setHeight(el[0].contentRect.height);
});
resizeObserver.observe(ref.current);
return () => {
resizeObserver.disconnect();
};
}, [height, isOpen]);
useEffect(() => {
if (isOpen) setHeight(ref.current?.getBoundingClientRect().height);
else setHeight(0);
}, [isOpen]);
return (
<>
<div className={"collapsible-card"}>
<div>
<button
type="button"
className={`collapsible-button ${
isOpen ? "" : "collapsed"
}`}
onClick={toggleOpen}
>
{header}
</button>
</div>
<div className={"collapsible-content"} style={{ height }}>
<div ref={ref}>
<div className={"collapsible-content-padding"}>
{children}
</div>
</div>
</div>
</div>
</>
);
}
export default Collapsible;
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment