diff --git a/package.json b/package.json index ab34a2846..f16fc2889 100644 --- a/package.json +++ b/package.json @@ -104,6 +104,7 @@ "@types/jest": "^29.5.12", "@types/node": "^20.12.2", "@types/react": "^18.2.75", + "@types/react-bootstrap": "^0.32.37", "@types/react-dom": "^18.2.24", "@types/react-transition-group": "^4.4.10", "bootstrap-italia": "2.12.1", @@ -135,6 +136,7 @@ "dependencies": { "classnames": "^2.3.1", "is-number": "^7.0.0", + "react-bootstrap": "^2.10.6", "react-stickup": "^1.12.1", "react-toastify": "^7.0.4", "react-transition-group": "^4.4.5", diff --git a/src/Alert/Alert.tsx b/src/Alert/Alert.tsx index 6aeeac5b5..b8451039d 100644 --- a/src/Alert/Alert.tsx +++ b/src/Alert/Alert.tsx @@ -28,7 +28,6 @@ export interface AlertProps extends HTMLAttributes { testId?: string; } - export const Alert: FC = ({ color = 'success', isOpen = true, fade = true, testId, ...props }) => { const baseProps = { color, @@ -36,4 +35,4 @@ export const Alert: FC = ({ color = 'success', isOpen = true, fade = fade }; return ; -}; \ No newline at end of file +}; diff --git a/src/Autocomplete/Autocomplete.tsx b/src/Autocomplete/Autocomplete.tsx index e9d3c7241..f1949ac2f 100644 --- a/src/Autocomplete/Autocomplete.tsx +++ b/src/Autocomplete/Autocomplete.tsx @@ -1,80 +1,81 @@ -import React, { FC} from 'react'; +import React, { FC } from 'react'; // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore non ci sono i types import BaseAutocomplete from 'accessible-autocomplete/react'; // Reference to https://www.npmjs.com/package/accessible-autocomplete export interface AutocompleteAttributes { - /** Identificativo */ - id?: string; - /** Valori chiave - valore all'interno della select */ - source: { value: string; label: string }[]; - /** Placeholder (default: ``) */ - placeholder?: string; - /** Valore di default (default: ``) */ - defaultValue?: string; - /** Modalità display menu (default: `inline`) */ - displayMenu?: string; - /** Funzione ritornante stringa in caso di nessun risultato */ - tNoResults?: () => string; - /** Funzione ritornante stringa di suggerimento */ - tAssistiveHint?: () => string; - /** Funzione ritornante stringa se la query è troppo corta */ - tStatusQueryTooShort?: () => string; - /** Funzione ritornante stringa se non ci sono risultati di ricerca */ - tStatusNoResults?: () => string; - /** Funzione ritornante stringa che identifica l'opzione selezionata */ - tStatusSelectedOption?: () => string; - /** Funzione ritornante stringa che identifica i risultati */ - tStatusResults?: () => string; - /** Classi aggiuntive da usare per il componente Button */ - className?: string; - testId?: string; + /** Identificativo */ + id?: string; + /** Valori chiave - valore all'interno della select */ + source: { value: string; label: string }[]; + /** Placeholder (default: ``) */ + placeholder?: string; + /** Valore di default (default: ``) */ + defaultValue?: string; + /** Modalità display menu (default: `inline`) */ + displayMenu?: string; + /** Funzione ritornante stringa in caso di nessun risultato */ + tNoResults?: () => string; + /** Funzione ritornante stringa di suggerimento */ + tAssistiveHint?: () => string; + /** Funzione ritornante stringa se la query è troppo corta */ + tStatusQueryTooShort?: () => string; + /** Funzione ritornante stringa se non ci sono risultati di ricerca */ + tStatusNoResults?: () => string; + /** Funzione ritornante stringa che identifica l'opzione selezionata */ + tStatusSelectedOption?: () => string; + /** Funzione ritornante stringa che identifica i risultati */ + tStatusResults?: () => string; + /** Classi aggiuntive da usare per il componente Button */ + className?: string; + testId?: string; } - const tAssistiveHintDefault = () => - 'Quando i risultati del completamento automatico sono disponibili, usa le frecce su e giù per rivedere e Invio per selezionare. Utenti di dispositivi touch, esplora tramite tocco o con gesti di scorrimento' -const tNoResultsDefault = () => 'Nessun risultato trovato' -const tStatusQueryTooShortDefault = (minQueryLength: number) => `Digita ${minQueryLength} o più caratteri per mostrare le opzioni di ricerca` -const tStatusNoResultsDefault = () => 'Nessun risultato di ricerca' -const tStatusSelectedOptionDefault = (selectedOption: number, length: number, index: number) => `${selectedOption} ${index + 1} di ${length} è sottolineato` + 'Quando i risultati del completamento automatico sono disponibili, usa le frecce su e giù per rivedere e Invio per selezionare. Utenti di dispositivi touch, esplora tramite tocco o con gesti di scorrimento'; +const tNoResultsDefault = () => 'Nessun risultato trovato'; +const tStatusQueryTooShortDefault = (minQueryLength: number) => + `Digita ${minQueryLength} o più caratteri per mostrare le opzioni di ricerca`; +const tStatusNoResultsDefault = () => 'Nessun risultato di ricerca'; +const tStatusSelectedOptionDefault = (selectedOption: number, length: number, index: number) => + `${selectedOption} ${index + 1} di ${length} è sottolineato`; const tStatusResultsDefault = (length: number, contentSelectedOption: number) => { - const words = { - result: length === 1 ? 'risultato' : 'risultati', - is: length === 1 ? 'è' : 'sono', - available: length === 1 ? 'disponibile' : 'disponibili', - } + const words = { + result: length === 1 ? 'risultato' : 'risultati', + is: length === 1 ? 'è' : 'sono', + available: length === 1 ? 'disponibile' : 'disponibili' + }; - return `${length} ${words.result} ${words.is} ${words.available}. ${contentSelectedOption}` -} + return `${length} ${words.result} ${words.is} ${words.available}. ${contentSelectedOption}`; +}; -export const Autocomplete: FC = ({ - tAssistiveHint = tAssistiveHintDefault, - tNoResults = tNoResultsDefault, - tStatusQueryTooShort = tStatusQueryTooShortDefault, - tStatusNoResults = tStatusNoResultsDefault, - tStatusSelectedOption = tStatusSelectedOptionDefault, - tStatusResults = tStatusResultsDefault, - placeholder = '', - defaultValue = '', - displayMenu = 'inline', - source, - ...attributes - }) => { - - return ; - }; - \ No newline at end of file +export const Autocomplete: FC = ({ + tAssistiveHint = tAssistiveHintDefault, + tNoResults = tNoResultsDefault, + tStatusQueryTooShort = tStatusQueryTooShortDefault, + tStatusNoResults = tStatusNoResultsDefault, + tStatusSelectedOption = tStatusSelectedOptionDefault, + tStatusResults = tStatusResultsDefault, + placeholder = '', + defaultValue = '', + displayMenu = 'inline', + source, + ...attributes +}) => { + return ( + + ); +}; diff --git a/src/BackToTop/BackToTop.tsx b/src/BackToTop/BackToTop.tsx index e4d7b34cd..ee52ed93a 100644 --- a/src/BackToTop/BackToTop.tsx +++ b/src/BackToTop/BackToTop.tsx @@ -34,7 +34,7 @@ const backToTop = () => { }; export const BackToTop = ({ - ariaLabel='Torna su', + ariaLabel = 'Torna su', className, dark = false, small = false, diff --git a/src/Card/CardCategory.tsx b/src/Card/CardCategory.tsx index 16045d640..f448127cd 100644 --- a/src/Card/CardCategory.tsx +++ b/src/Card/CardCategory.tsx @@ -21,7 +21,18 @@ export interface CardCategoryProps extends HTMLAttributes { testId?: string; } -export const CardCategory: FC = ({ iconName, iconTitle, date, href, onClick, testId, children, textDescription, dateDescription, ...rest }) => { +export const CardCategory: FC = ({ + iconName, + iconTitle, + date, + href, + onClick, + testId, + children, + textDescription, + dateDescription, + ...rest +}) => { const classes = classNames({ 'category-top': date || ' ', 'categoryicon-top': iconName @@ -29,7 +40,7 @@ export const CardCategory: FC = ({ iconName, iconTitle, date, // Simple category link const categoryLink = !iconName && ( <> - {textDescription && {textDescription}} + {textDescription && {textDescription}} {children} @@ -44,8 +55,10 @@ export const CardCategory: FC = ({ iconName, iconTitle, date,
{categoryLink} {categoryIcon} - {categoryText && textDescription && {textDescription}}{categoryText} - {dateDescription && {dateDescription}}{categoryDate} + {categoryText && textDescription && {textDescription}} + {categoryText} + {dateDescription && {dateDescription}} + {categoryDate}
); }; diff --git a/src/Collapse/Collapse.tsx b/src/Collapse/Collapse.tsx index 74ddc2153..3b340d2ed 100644 --- a/src/Collapse/Collapse.tsx +++ b/src/Collapse/Collapse.tsx @@ -6,7 +6,6 @@ import { CSSModule } from 'reactstrap/types/lib/utils'; import { Icon } from '../Icon/Icon'; - // Copy over from reactstrap and add new ones export interface CollapseProps extends HTMLAttributes { /** Indica se il menu HeaderNav sia aperto o meno. Usato unicamente nel caso della HeaderNav, ovvero con navbar e header entrambi true */ @@ -43,7 +42,7 @@ export interface CollapseProps extends HTMLAttributes { /** Da utilizzare per impostare un riferimento all'elemento DOM */ innerRef?: Ref; /** Testo pulsante di chiusura per screen reader */ - closeSrText?: string, + closeSrText?: string; testId?: string; } @@ -57,7 +56,7 @@ export const Collapse: FC = ({ onOverlayClick, cssModule, testId, - closeSrText='Nascondi la navigazione', + closeSrText = 'Nascondi la navigazione', ...attributes }) => { const newCssModule = { @@ -69,13 +68,10 @@ export const Collapse: FC = ({ expanded: isOpen }); const style = { display: isOpen ? 'block' : 'none' }; - const overlayClasses = classNames( - 'overlay', - { - 'fade' : isOpen, - 'show' : isOpen - } - ) + const overlayClasses = classNames('overlay', { + fade: isOpen, + show: isOpen + }); return ( = ({
{megamenu ?
{children}
: <>{children}} diff --git a/src/Dropdown/Dropdown.tsx b/src/Dropdown/Dropdown.tsx index e9a398327..0905d1ec2 100644 --- a/src/Dropdown/Dropdown.tsx +++ b/src/Dropdown/Dropdown.tsx @@ -1,6 +1,6 @@ import classNames from 'classnames'; import React, { ElementType, FC } from 'react'; -import { Dropdown as BSDRopdown, DropdownProps as BSDRopdownProps} from 'reactstrap'; +import { Dropdown as BSDRopdown, DropdownProps as BSDRopdownProps } from 'reactstrap'; export interface DropdownProps extends BSDRopdownProps { tag?: ElementType; inNavbar?: boolean; @@ -21,7 +21,7 @@ export const Dropdown: FC = ({ ...attributes }) => { const classes = classNames(className, { - 'text-center': textCenter, + 'text-center': textCenter }); const [isOpen, setIsOpen] = React.useState(false); @@ -41,10 +41,8 @@ export const Dropdown: FC = ({ > { // eslint-disable-next-line @typescript-eslint/no-explicit-any - React.Children.map(children, (child: any) => - React.cloneElement(child, { inNavbar: inNavbar }) - ) - } + React.Children.map(children, (child: any) => React.cloneElement(child, { inNavbar: inNavbar })) + } ); }; diff --git a/src/Forward/Forward.tsx b/src/Forward/Forward.tsx index 4239a7dda..dd99c0843 100644 --- a/src/Forward/Forward.tsx +++ b/src/Forward/Forward.tsx @@ -15,14 +15,14 @@ export const Forward: FC = ({ className, children, testId, ...attr { - e.preventDefault() + e.preventDefault(); if (attributes.href) { - const scrollToRef = document.querySelector(attributes.href) + const scrollToRef = document.querySelector(attributes.href); if (scrollToRef) { scrollToRef.scrollIntoView({ behavior: 'smooth', block: 'start' - }) + }); } } }} diff --git a/src/Grid/GridItemTextWrapper.tsx b/src/Grid/GridItemTextWrapper.tsx index cf12cbe47..7f2f73657 100644 --- a/src/Grid/GridItemTextWrapper.tsx +++ b/src/Grid/GridItemTextWrapper.tsx @@ -9,7 +9,13 @@ export interface GridItemTextWrapperProps extends HTMLAttributes = ({ className, children, testId, tag='span', ...attributes }) => { +export const GridItemTextWrapper: FC = ({ + className, + children, + testId, + tag = 'span', + ...attributes +}) => { const classes = classname('it-griditem-text-wrapper', className); const Tag = tag; return ( diff --git a/src/Header/HeaderToggler.tsx b/src/Header/HeaderToggler.tsx index 0e5c84c38..611b547f8 100644 --- a/src/Header/HeaderToggler.tsx +++ b/src/Header/HeaderToggler.tsx @@ -21,7 +21,7 @@ export interface HeaderTogglerProps extends ButtonHTMLAttributes { +export const HeaderToggler = ({ className, tag, type, isOpen = false, testId, ...attributes }: HeaderTogglerProps) => { const HeaderType = useHeaderContext(); const defaultTag = HeaderType === SLIM ? 'a' : BUTTON; const defaultType = HeaderType === SLIM ? undefined : BUTTON; @@ -32,7 +32,7 @@ export const HeaderToggler = ({ className, tag, type, isOpen=false, testId, ...a }, className ); - const expanded = isOpen ? "true" : "false" + const expanded = isOpen ? 'true' : 'false'; useEffect(() => { document.querySelectorAll('.container-fluid').forEach((element) => { element.classList.remove('container-fluid'); diff --git a/src/Icon/assets/index.ts b/src/Icon/assets/index.ts index 125e3f3b8..846230e48 100644 --- a/src/Icon/assets/index.ts +++ b/src/Icon/assets/index.ts @@ -698,4 +698,4 @@ export const allIcons = Object.keys(iconList); export interface SVGRProps { title?: string; titleId?: string; -} \ No newline at end of file +} diff --git a/src/Input/Input.tsx b/src/Input/Input.tsx index e6d5b2f50..637d8b1a3 100644 --- a/src/Input/Input.tsx +++ b/src/Input/Input.tsx @@ -416,4 +416,4 @@ export const Input = ({ } return ; -}; \ No newline at end of file +}; diff --git a/src/Input/InputContainer.tsx b/src/Input/InputContainer.tsx index 5b135f048..d46afd705 100644 --- a/src/Input/InputContainer.tsx +++ b/src/Input/InputContainer.tsx @@ -38,7 +38,6 @@ export const InputContainer: FC = ({ iconLeft, children }) => { - if (hasButtonRight || hasIconLeft) { return (
diff --git a/src/Input/TextArea.tsx b/src/Input/TextArea.tsx index 793a3b8e8..76da3cfc3 100644 --- a/src/Input/TextArea.tsx +++ b/src/Input/TextArea.tsx @@ -57,10 +57,7 @@ export const TextArea = ({ const extraAttributes: { ['aria-describedby']?: string } = {}; //Chiamo questa funzione per impostare classNames a 'form-control' - const formControlClass = getFormControlClass( - {}, - cssModule - ); + const formControlClass = getFormControlClass({}, cssModule); // associate the input field with the help text const infoId = id ? `${id}Description` : undefined; if (id) { diff --git a/src/Input/utils.tsx b/src/Input/utils.tsx index 7d9bd99cb..6551422a6 100644 --- a/src/Input/utils.tsx +++ b/src/Input/utils.tsx @@ -117,8 +117,7 @@ export function getClasses( // we can model here only if stylings 'form-control-plaintext': normalizedOnlyCondition, 'form-control': passwordOnlyCondition, - 'input-password': passwordOnlyCondition, - 'focus--mouse': passwordOnlyCondition || normalizedOnlyCondition + 'input-password': passwordOnlyCondition } ), cssModule diff --git a/src/List/List.tsx b/src/List/List.tsx index 72761e02a..b868149e2 100644 --- a/src/List/List.tsx +++ b/src/List/List.tsx @@ -16,14 +16,7 @@ export interface ListProps extends HTMLAttributes { testId?: string; } -export const List: FC = ({ - className, - wrapperClassName, - tag = 'div', - noWrapper, - testId, - ...attributes -}) => { +export const List: FC = ({ className, wrapperClassName, tag = 'div', noWrapper, testId, ...attributes }) => { const Tag = tag; const wrapperClasses = classNames('it-list-wrapper', wrapperClassName); const classes = classNames(className, 'it-list'); diff --git a/src/List/ListItem.tsx b/src/List/ListItem.tsx index 23aabd9a3..8d349d28c 100644 --- a/src/List/ListItem.tsx +++ b/src/List/ListItem.tsx @@ -39,46 +39,37 @@ export const ListItem: FC & { children, ...attributes }) => { - const Tag = tag; - const classes = classNames( - className, - { active }, - 'list-item' - ), - classesItem = classNames(className, { - 'it-rounded-icon': icon, - 'avatar size-lg': avatar, - 'it-thumb': img - }), - leftItem = icon || avatar || img; - - if (href) { - return ( -
  • - -
    {children}
    -
    -
  • - ); - } + const Tag = tag; + const classes = classNames(className, { active }, 'list-item'), + classesItem = classNames(className, { + 'it-rounded-icon': icon, + 'avatar size-lg': avatar, + 'it-thumb': img + }), + leftItem = icon || avatar || img; + if (href) { return (
  • - - {leftItem &&
    {leftItem}
    } -
    {children}
    -
    + +
    {children}
    +
  • ); - }; + } + + return ( +
  • + + {leftItem &&
    {leftItem}
    } +
    {children}
    +
    +
  • + ); +}; const MultipleAction: FC = ({ children }) => { - return {children} + return {children}; }; ListItem.MultipleAction = MultipleAction; diff --git a/src/NavScroll/types.ts b/src/NavScroll/types.ts index e5e678ed1..50c80d8a2 100644 --- a/src/NavScroll/types.ts +++ b/src/NavScroll/types.ts @@ -87,8 +87,8 @@ export type useNavScrollResult = { */ getActiveRef: () => RefObject | null; /** - * A list of active ids (the full hierarchy). - */ + * A list of active ids (the full hierarchy). + */ percentage: number; }; diff --git a/src/ResponsiveImage/ResponsiveImage.tsx b/src/ResponsiveImage/ResponsiveImage.tsx index ebba273be..fe8befeb4 100644 --- a/src/ResponsiveImage/ResponsiveImage.tsx +++ b/src/ResponsiveImage/ResponsiveImage.tsx @@ -12,26 +12,32 @@ export interface ResponsiveImageProps extends HTMLAttributes { proportioned?: boolean; } -export const ResponsiveImage: FC = ({ alt, testId, proportioned=false, children, ...attributes }) => { +export const ResponsiveImage: FC = ({ + alt, + testId, + proportioned = false, + children, + ...attributes +}) => { if (children) { if (proportioned) { - return( + return (
    - {alt} + {alt} {children}
    - ) + ); } else { - return( + return (
    - {alt} + {alt} {children}
    - ) + ); } } else { return ( diff --git a/src/Skiplink/Skiplink.tsx b/src/Skiplink/Skiplink.tsx index 268917c7c..f04146b14 100644 --- a/src/Skiplink/Skiplink.tsx +++ b/src/Skiplink/Skiplink.tsx @@ -12,18 +12,23 @@ export interface SkiplinkProps extends HTMLAttributes { testId?: string; } -export const Skiplink: FC = ({ ariaLabel=null, className, tag = 'div', nav=false, testId, children, ...attributes }) => { +export const Skiplink: FC = ({ + ariaLabel = null, + className, + tag = 'div', + nav = false, + testId, + children, + ...attributes +}) => { const Tag = nav ? 'nav' : tag; const classes = classNames(className, 'skiplinks'); if (nav) { return ( - -
      - {children} -
    + +
      {children}
    - ) - + ); } else { return ; } diff --git a/src/Skiplink/SkiplinkItem.tsx b/src/Skiplink/SkiplinkItem.tsx index 2b88d18e5..ed122b131 100644 --- a/src/Skiplink/SkiplinkItem.tsx +++ b/src/Skiplink/SkiplinkItem.tsx @@ -31,7 +31,7 @@ export const SkiplinkItem: FC = ({ if (navItem) { return ( -
  • +
  • ); diff --git a/src/Tab/TabNav.tsx b/src/Tab/TabNav.tsx new file mode 100644 index 000000000..e596df3db --- /dev/null +++ b/src/Tab/TabNav.tsx @@ -0,0 +1,111 @@ +import classNames from 'classnames'; +import React, { ElementType, FC, useRef } from 'react'; +import { Nav, NavProps } from 'react-bootstrap'; + +export interface TabNavProps extends NavProps { + /** Utilizzarlo in caso di utilizzo di componenti personalizzati + * @default ul + */ + tag?: ElementType; + /** Classi aggiuntive da usare per il componente Tab */ + className?: string; + /** Imposta l'orientameno delle tab in verticale + * @default false + */ + vertical?: boolean; + /** Imposta la tab con sfondo scuro + * @default false + */ + dark?: boolean; + /** Imposta la tab con design tipo card + * @default false + */ + card?: boolean; + testId?: string; +} + +export const TabNav: FC = ({ + className, + vertical = false, + dark = false, + card = false, + tag = 'ul', + testId, + ...attributes +}) => { + const Tag = tag; + const rootRef = useRef(); + + const classes = classNames( + className, + 'nav-tabs', + { 'nav-tabs-vertical': vertical }, + { 'nav-dark': dark }, + { 'nav-tabs-cards': card } + ); + let currentTabIndex = 0; + let activeTabIndex = -1; + + // Ugly workaround to keep Bootstrap Italia behaviour + + const handleKeyDown = (event: React.KeyboardEvent, disabled: boolean = false) => { + const queriedElements = rootRef.current?.querySelectorAll('.nav-link'); + if (queriedElements) { + for (let i = 0; i < queriedElements.length; i++) { + if (queriedElements[i].ariaSelected === 'true') { + activeTabIndex = i; + } + // Disabled elements ignore current focused tab + if (!disabled && document.activeElement === queriedElements[i]) { + currentTabIndex = i; + } + queriedElements[i].ariaSelected = 'false'; + } + switch (event.key) { + case 'ArrowLeft': + case 'ArrowUp': + event.stopPropagation(); + event.preventDefault(); + if (currentTabIndex - 1 < 0) { + currentTabIndex = queriedElements.length; + } + currentTabIndex = (currentTabIndex - 1) % queriedElements.length; + break; + case 'ArrowRight': + case 'ArrowDown': + event.stopPropagation(); + event.preventDefault(); + currentTabIndex = (currentTabIndex + 1) % queriedElements.length; + break; + case 'Enter': + event.stopPropagation(); + event.preventDefault(); + (queriedElements[currentTabIndex] as HTMLElement).click(); + break; + case 'Tab': + break; + default: + return; + } + if (queriedElements[currentTabIndex].ariaDisabled === 'true') { + handleKeyDown(event, true); + } else { + (queriedElements[currentTabIndex] as HTMLElement).focus({ preventScroll: true }); + setTimeout(() => { + queriedElements[activeTabIndex].ariaSelected = 'true'; + }, 300); + } + } + }; + + return ( + + ); +}; diff --git a/src/Tab/TabNavItem.tsx b/src/Tab/TabNavItem.tsx new file mode 100644 index 000000000..2d6bc9efd --- /dev/null +++ b/src/Tab/TabNavItem.tsx @@ -0,0 +1,14 @@ +import React, { ElementType, FC } from 'react'; +import { Nav, NavItemProps } from 'react-bootstrap'; + +export interface TabNavItemProps extends NavItemProps { + /** Utilizzarlo in caso di utilizzo di componenti personalizzati */ + tag?: ElementType; + testId?: string; +} + +export const TabNavItem: FC = ({ tag = 'li', testId, ...attributes }) => { + const Tag = tag; + + return ; +}; diff --git a/src/Tab/TabNavLink.tsx b/src/Tab/TabNavLink.tsx new file mode 100644 index 000000000..bc1c7d06f --- /dev/null +++ b/src/Tab/TabNavLink.tsx @@ -0,0 +1,14 @@ +import React, { ElementType, FC } from 'react'; +import { Nav, NavLinkProps } from 'react-bootstrap'; + +export interface TabNavLinkProps extends NavLinkProps { + /** Utilizzarlo in caso di utilizzo di componenti personalizzati */ + tag?: ElementType; + testId?: string; +} + +export const TabNavLink: FC = ({ tag, testId, ...attributes }) => { + const Tag = tag; + + return ; +}; diff --git a/src/Tab/TabPanel.tsx b/src/Tab/TabPanel.tsx new file mode 100644 index 000000000..beaa5ebda --- /dev/null +++ b/src/Tab/TabPanel.tsx @@ -0,0 +1,14 @@ +import React, { ElementType, FC } from 'react'; +import { NavProps, Tab } from 'react-bootstrap'; + +export interface TabsProps extends NavProps { + /** Utilizzarlo in caso di utilizzo di componenti personalizzati */ + tag?: ElementType; + testId?: string; +} + +export const TabContainer: FC = ({ tag, testId, ...attributes }) => { + const Tag = tag; + + return ; +}; diff --git a/src/Timeline/TimelinePin.tsx b/src/Timeline/TimelinePin.tsx index d1b99d503..7309792fa 100644 --- a/src/Timeline/TimelinePin.tsx +++ b/src/Timeline/TimelinePin.tsx @@ -40,7 +40,7 @@ export const TimelinePin: FC = ({ nowText, testId, className, - tag='h3', + tag = 'h3', ...attributes }) => { const { children, ...rest } = attributes; @@ -51,7 +51,7 @@ export const TimelinePin: FC = ({ }); const pinIcon = (
    - +
    ); const pinLabel = ( diff --git a/src/Toggle/Toggle.tsx b/src/Toggle/Toggle.tsx index a424d2e4a..4cc80a219 100644 --- a/src/Toggle/Toggle.tsx +++ b/src/Toggle/Toggle.tsx @@ -23,7 +23,7 @@ export const Toggle: FC = ({ label, testId, ...rest }) => {
    diff --git a/src/index.ts b/src/index.ts index 6f429ee77..504272cfe 100644 --- a/src/index.ts +++ b/src/index.ts @@ -37,8 +37,6 @@ export { PopoverBody, PopoverHeader, Row, - TabContent, - TabPane, Table, Tooltip, UncontrolledAlert, @@ -47,6 +45,7 @@ export { Util } from 'reactstrap'; +export { TabContainer, TabContent, TabPane } from 'react-bootstrap'; export { Autocomplete } from './Autocomplete/Autocomplete'; export { Accordion } from './Accordion/Accordion'; export { AccordionBody } from './Accordion/AccordionBody'; @@ -136,6 +135,9 @@ export { StepperDots } from './Stepper/StepperDots'; export { StepperHeader } from './Stepper/StepperHeader'; export { StepperHeaderElement } from './Stepper/StepperHeaderElement'; export { StepperNav } from './Stepper/StepperNav'; +export { TabNav } from './Tab/TabNav'; +export { TabNavItem } from './Tab/TabNavItem'; +export { TabNavLink } from './Tab/TabNavLink'; export { ThumbNav } from './ThumbNav/ThumbNav'; export { ThumbNavItem } from './ThumbNav/ThumbNavItem'; export { TimelinePin } from './Timeline/TimelinePin'; @@ -242,6 +244,9 @@ export type { StepperDotsProps } from './Stepper/StepperDots'; export type { StepperHeaderProps } from './Stepper/StepperHeader'; export type { StepperHeaderElementProps } from './Stepper/StepperHeaderElement'; export type { StepperNavProps } from './Stepper/StepperNav'; +export type { TabNavProps } from './Tab/TabNav'; +export type { TabNavItemProps } from './Tab/TabNavItem'; +export type { TabNavLinkProps } from './Tab/TabNavLink'; export type { ThumbNavProps } from './ThumbNav/ThumbNav'; export type { ThumbNavItemProps } from './ThumbNav/ThumbNavItem'; export type { TimelinePinProps } from './Timeline/TimelinePin'; @@ -290,8 +295,6 @@ export type { PopoverHeaderProps, PopoverProps, RowProps, - TabContentProps, - TabPaneProps, TableProps, TooltipProps, UncontrolledAlertProps, @@ -299,4 +302,5 @@ export type { UncontrolledTooltipProps } from 'reactstrap'; -import "./track-focus.js" \ No newline at end of file +export type { TabContainerProps, TabContentProps, TabPaneProps } from 'react-bootstrap'; +import './track-focus.js'; diff --git a/src/track-focus.js b/src/track-focus.js index 7883aefb6..3d4c7c688 100644 --- a/src/track-focus.js +++ b/src/track-focus.js @@ -1,4 +1,3 @@ - // Focus Management /** @@ -9,41 +8,41 @@ * -------------------------------------------------------------------------- */ -const DATA_MOUSE_FOCUS = 'data-focus-mouse' -const CLASS_NAME_MOUSE_FOCUS = 'focus--mouse' +const DATA_MOUSE_FOCUS = 'data-focus-mouse'; +const CLASS_NAME_MOUSE_FOCUS = 'focus--mouse'; class TrackFocus { constructor() { - this._usingMouse = false + this._usingMouse = false; - this._bindEvents() + this._bindEvents(); } _bindEvents() { if (typeof document === 'undefined') { - return + return; } - const events = ['keydown', 'mousedown'] + const events = ['keydown', 'mousedown']; events.forEach((evtName) => { document.addEventListener(evtName, (evt) => { - this._usingMouse = evt.type === 'mousedown' - }) - }) + this._usingMouse = evt.type === 'mousedown'; + }); + }); document.addEventListener('focusin', (evt) => { if (this._usingMouse) { if (evt.target) { evt.target.classList.add(CLASS_NAME_MOUSE_FOCUS); - evt.target.setAttribute(DATA_MOUSE_FOCUS, 'true') + evt.target.setAttribute(DATA_MOUSE_FOCUS, 'true'); } } - }) + }); document.addEventListener('focusout', (evt) => { if (evt.target) { evt.target.classList.remove(CLASS_NAME_MOUSE_FOCUS); - evt.target.setAttribute(DATA_MOUSE_FOCUS, 'false') + evt.target.setAttribute(DATA_MOUSE_FOCUS, 'false'); } - }) + }); } } -new TrackFocus() \ No newline at end of file +new TrackFocus(); diff --git a/stories/Components/Tab.stories.tsx b/stories/Components/Tab.stories.tsx index b14289c5f..132cdd618 100644 --- a/stories/Components/Tab.stories.tsx +++ b/stories/Components/Tab.stories.tsx @@ -1,475 +1,1051 @@ -import { Meta, StoryObj } from "@storybook/react"; -import React, { useState } from "react"; -import { Col, Container, Icon, Nav, NavItem, NavLink, Row, TabContent, TabPane } from "../../src"; +import { Meta, StoryObj } from '@storybook/react'; +import React from 'react'; +import { Col, Icon, Row, TabContainer, TabContent, TabNav, TabNavItem, TabNavLink, TabPane } from '../../src'; -const meta: Meta = { - title: "Documentazione/Componenti/Tab", - component: Nav, +const meta: Meta = { + title: 'Documentazione/Componenti/Tab', + component: TabNav }; export default meta; -type Story = StoryObj; +type Story = StoryObj; -export const Esempi: Story = { +export const TestualeFull: Story = { render: () => ( -
    - + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + ) +}; - -
    +export const IconaFull: Story = { + render: () => ( + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + ) }; -const TestoWithHooks = () => { - const [activeTab, toggleTab] = useState("1"); - return ( -
    - +export const IconaGrandeFull: Story = { + render: () => ( + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + ) +}; - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim - veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in - voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia - deserunt mollit anim id est laborum. +export const TestoIconaFull: Story = { + render: () => ( + + + + + + Tab 1 + + + + + + Tab 2 + + + + + + Tab 3 + + + + + + Tab 4 + + + + + ) +}; + +export const Testuale: Story = { + render: () => ( + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + ) +}; + +export const Icona: Story = { + render: () => ( + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + ) +}; + +export const IconaGrande: Story = { + render: () => ( + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + + + Tab titolo 1 + + + + + + ) +}; + +export const TestoIcona: Story = { + render: () => ( + + + + + + Tab 1 + + + + + + Tab 2 + + + + + + Tab 3 + + + + + + Tab 4 + + + + + ) +}; + +export const Hidescroll: Story = { + render: () => ( +
    + + + + Attivo + + + Link + + + Link + + + Link + + + Link + + + Link + + + Link + + + Link + + + +
    + ) +}; + +export const TestualePannel: Story = { + render: () => ( + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + + Contenuto 1 - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, - consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud - exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum - dolore eu fugiat nulla pariatur. + + Contenuto 2 - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa - qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt - ut labore et dolore magna aliqua. + + Contenuto 3 + + + Contenuto 4 -
    - ); -} - -export const Testo: Story = { - render: () => , - parameters: { - docs: { - canvas: { sourceState: "none" }, - }, - }, + + ) }; -const IconaWithHooks = () => { - const [activeTab, toggleTab] = useState("1"); - return ( -
    - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim - veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in - voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia - deserunt mollit anim id est laborum. + + + + + + + Tab titolo 1 + + + + + + + Contenuto 1 + + + Contenuto 2 - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, - consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud - exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum - dolore eu fugiat nulla pariatur. + + Contenuto 3 - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa - qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt - ut labore et dolore magna aliqua. + + Contenuto 4 -
    - ); -} -export const Icona: Story = { - render: () => , - parameters: { - docs: { - canvas: { sourceState: "none" }, - }, - }, + + ) }; -const TestoIconaWithHooks = () => { - const [activeTab, toggleTab] = useState("1"); - return ( -
    - + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + ) +}; - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim - veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in - voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia - deserunt mollit anim id est laborum. +export const TestoIconaPannel: Story = { + render: () => ( + + + + + + Tab 1 + + + + + + Tab 2 + + + + + + Tab 3 + + + + + + Tab 4 + + + + + + Contenuto 1 + + + Contenuto 2 - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, - consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud - exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum - dolore eu fugiat nulla pariatur. + + Contenuto 3 - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa - qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt - ut labore et dolore magna aliqua. + + Contenuto 4 -
    - ); -} + + ) +}; -export const TestoIcona: Story = { - render: () => , - parameters: { - docs: { - canvas: { sourceState: "none" }, - }, - }, +export const TestualePannelVertical: Story = { + render: () => ( + + + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + + + ) +}; + +export const TestualePannelVerticalBackground: Story = { + render: () => ( + + + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + + + ) }; -const BottoneWithHooks = () => { - const [activeTab, toggleTab] = useState("1"); - return ( - +export const TestoIconaPannelVertical: Story = { + render: () => ( + - - + + + + + + Tab 4 + + + + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + + + ) +}; + +export const TestoPannelVertical: Story = { + render: () => ( + + + + + + + Tab 1 + + + + + + Tab 2 + + + + + + Tab 3 + + + + + + Tab 4 + + + + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + + + ) +}; + +export const TestualePannelReverseBottom: Story = { + render: () => ( + +
    + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + +
    +
    + ) +}; + +export const TestualePannelReverseRight: Story = { + render: () => ( + + + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim - ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt - in culpa qui officia deserunt mollit anim id est laborum. + + + + Contenuto 1 + + + Contenuto 2 - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit - amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis - nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate - velit esse cillum dolore eu fugiat nulla pariatur. + + Contenuto 3 - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt - in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod - tempor incididunt ut labore et dolore magna aliqua. + + Contenuto 4 -
    - ); -} + + ) +}; + +export const TestualeFullDark: Story = { + render: () => ( + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + ) +}; -export const Bottone: Story = { - render: () => , - parameters: { - docs: { - canvas: { sourceState: "none" }, - }, - }, +export const TestoIconaFullDark: Story = { + render: () => ( + + + + + + Tab 1 + + + + + + Tab 2 + + + + + + Tab 3 + + + + + + Tab 4 + + + + + ) }; -const VerticaleWithHooks = () => { - const [activeTab, toggleTab] = useState("1"); - return ( - +export const TestoIconaPannelVerticalDark: Story = { + render: () => ( + - - + + + + + + Tab 4 + + + - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim - ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt - in culpa qui officia deserunt mollit anim id est laborum. + + + + Contenuto 1 + + + Contenuto 2 - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit - amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis - nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate - velit esse cillum dolore eu fugiat nulla pariatur. + + Contenuto 3 - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in - reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt - in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod - tempor incididunt ut labore et dolore magna aliqua. + + Contenuto 4 - - ); -} + + ) +}; + +export const TestualePannelCard: Story = { + render: () => ( + + + + Attivo + + + Link + + + Link + + + + Disattivo + + + + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + + + ) +}; -export const Verticale: Story = { - render: () => , - parameters: { - docs: { - canvas: { sourceState: "none" }, - }, - }, +export const TestualePannelCardEditable: Story = { + render: () => ( + +
    + + + Tab 1 +
    + + Chiudi tab 1 +
    +
    + + Tab 2 +
    + + Chiudi tab 2 +
    +
    + + Tab 3 +
    + + Chiudi tab 3 +
    +
    + + + Disattivo + +
    + + Chiudi tab 4 +
    +
    +
  • + + + Aggiungi un tab + + +
    +
    + + + Contenuto 1 + + + Contenuto 2 + + + Contenuto 3 + + + Contenuto 4 + + +
    + ) }; diff --git a/stories/Documentation/NavScroll.mdx b/stories/Documentation/NavScroll.mdx index 699fe93dc..858d632b3 100644 --- a/stories/Documentation/NavScroll.mdx +++ b/stories/Documentation/NavScroll.mdx @@ -66,7 +66,7 @@ return ( toggleNavScroll(!isOpen)} > @@ -388,7 +388,7 @@ return ( toggleNavScroll(!isOpen)} > @@ -723,7 +723,7 @@ return ( className='it-navscroll-wrapper navbar-expand-lg it-top-navscroll it-right-side affix-top theme-dark-desk theme-dark-mobile' > toggleNavScroll(!isOpen)} > diff --git a/stories/Documentation/Tab.mdx b/stories/Documentation/Tab.mdx index 7a8d511df..ea3ff986a 100644 --- a/stories/Documentation/Tab.mdx +++ b/stories/Documentation/Tab.mdx @@ -1,5 +1,4 @@ import { ArgTypes, Canvas, Controls, Meta, Story } from '@storybook/blocks'; -import { Code } from '@storybook/components'; import { Alert, Callout, CalloutText, CalloutTitle } from '../../src'; import * as TabStories from '../Components/Tab.stories'; @@ -7,433 +6,190 @@ import * as TabStories from '../Components/Tab.stories'; # Tab -### Esempi - L'interfaccia a tab (o schede) di Bootstrap si basa sull'utilizzo del layout di navigazione `Nav`, con l'aggiunta della proprietà `tabs`. Per ottenere una versione con sfondo scuro e testo chiaro bisognerà aggiungere un'ulteriore classe: `.nav-dark`. - - -## Con Testo - -Gli esempi sopra riportati non hanno molto senso senza un contenuto che cambi al di sotto di essi; per rendere tali interfacce navigabili è necessario collegare Tab e Pannelli collegando il tag `` dei primi all’attributo id dei secondi, come mostrato di seguito: - - - -#### Source - -```tsx -const [activeTab, toggleTab] = useState('1'); -return ( -
    - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. - - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. - - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis - aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint - occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum - dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. - - -
    -); -``` - -## Icona - -Oppure al posto della label usare una icona. +
    + + + Info documentazione + + + La maggior parte degli esempi riguardanti il componente Tab non hanno pannelli di contenuto associati, pertanto la + funzionalità per selezionare il pannello risulterà disattivata. Per vedere il componente Tab in azione fare + riferimento alla sezione [Controllo dei pannelli associati](#controllo-dei-pannelli-associati). + + +
    + +
    + + + Accessibilità + + + Le interfacce a tab, come descritto nelle [pratiche di implementazione WAI ARIA](https://www.w3.org/TR/wai-aria-practices/#tabpanel), richiedono l’utilizzo di attributi `role="tablist"`, `role="tab"`, `role="tabpanel"`, e ulteriori attributi `aria-`, al fine di trasmettere la loro struttura, funzionalità e stato attuale agli utenti delle tecnologie assistive (come i lettori di schermo). + + Nota che le interfacce dinamiche a tab **non devono** contenere menu a discesa, poiché ciò causa problemi di usabilità e accessibilità. Dal punto di vista dell’usabilità, il fatto che l’elemento trigger del tab attualmente visualizzato non sia immediatamente visibile (dato che si trova all’interno del menu a discesa chiuso) può causare confusione. + + + +
    + +
    + + + Breaking change + + + Dalla versione 5.x.x il componente `Tab` è stato completamente ridisegnato per una migliore implementazione dei + pattern ARIA. + + +
    + +## Tab orizzontali a tutta larghezza + +Aggiungendo la classe `.auto` al contenitore `TabContainer` i tab occupano automaticamente l’intera larghezza disponibile. Se su schermi particolarme piccoli (es: smartphone) le dimensioni dei tab dovessero superare quelle dello schermo, verrà attivato lo scrolling orizzontale dei tab stessi. + +### Tab testuale + +Lo stato dei tab può essere attivo, non attivo e disabilitato. Utilizzare la proprietà `disabled` sul link dei tab da disabilitare. + + + +### Tab con icona + +Le label dei tab possono essere sostituite da icone con classi che ne indicano il tipo, avendo cura di includere all’interno del link un elemento `` con classe `.visually-hidden` contenente la descrizione dedicata agli screen reader. La classe .visually-hidden impedisce la visualizzazione del testo sui browser visuali. + + + +### Tab con icona grande + +Per ottenere icone più grandi utilizzare la classe `.icon-lg` alle icone. Aggiungere la classe `.nav-tabs-icon-lg` al contenitore `TabContainer` per ottimizzare i margini fra tab. + + + +### Tab con testo e icona + +Icone e testi possono convivere all’interno dei tab, l’allineamento verticale dei due elementi è automatico. Per ottenere il corretto margine fra testo e icona nei tab a sviluppo orizzontale è necessario aggiungere la classe `.nav-tabs-icon-text` al contenitore `TabContainer`. + + + +## Tab orizzontali + +In assenza della classe `.auto` i tab vengono dimensionati in base al contenuto. + +### Tab testuale + + + +### Tab con icona -#### Source - -```tsx -const [activeTab, toggleTab] = useState('1'); -return ( -
    - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. - - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. - - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis - aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint - occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum - dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. - - -
    -); -``` - -### Testo e icona - -Oppure con label e icona insieme. +### Tab con icona grande + + + +### Tab con testo e icona -#### Source - -```tsx -const [activeTab, toggleTab] = useState('1'); -return ( -
    - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. - - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore - magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo - consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla - pariatur. - - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis - aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint - occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum - dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. - - -
    -); -``` - -## Bottone - -I tab possono ereditare dalla navigazione l'utilizzo della proprietà `pills` per generare tab a bottoni. - - - -#### Source - -```tsx -const [activeTab, toggleTab] = useState('1'); -return ( - - - - - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et - dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex - ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat - nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit - anim id est laborum. - - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore - et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip - ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu - fugiat nulla pariatur. - - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. - Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore - et dolore magna aliqua. - - - - - -); -``` - -## Allineamento verticale - - - -#### Source - -```tsx -const [activeTab, toggleTab] = useState('1'); -return ( - - - - - - - - - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et - dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex - ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat - nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit - anim id est laborum. - - - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore - et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip - ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu - fugiat nulla pariatur. - - - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. - Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est - laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore - et dolore magna aliqua. - - - - - -); -``` +## Rimozione delle scrollbar su dispositivi touch + +Se per motivi estetici si volessero nascondere le scrollbar che appaiono sui dispositivi touch quando vengono scrollati orizzontalmente i tab sarà necessario inserire l’html dei tab all’interno di un wrapper con classe `.nav-tabs-hidescroll`. + +Se i tab contengono icone è necessario aggiungere un’ulteriore classe al wrapper: + +- `.hidescroll-ico` nel caso di icone normali +- `.hidescroll-ico-lg` nel caso di icone piccole + + + +## Controllo dei pannelli associati + +Gli esempi sopra riportati non hanno molto senso senza un contenuto che cambi al di sotto di essi; per rendere tali interfacce navigabili è necessario inserire i componenti `TabContent` e `TabPane`. + +## Tab testuale + + + +### Tab con icona + + + +### Tab con icona grande + + + +### Tab con testo e icona + + + +## Tab verticali + +Impostando il parametro `vertical` al componente `TabNav` e aggiungendo i dovuti componenti per la spaziatura verticale, è possibile realizzare tab a orientamento verticale. + +### Tab testuale + + + +### Tab testuale con colore di sfondo + +Aggiungendo la classe `.nav-tabs-vertical-background` al componente `TabNav` contenitore dei link i tab selezionati avranno un colore di sfondo. + + + +### Tab con testo e icona + + + +### Tab con icona + +Le label dei Tab possono essere sostituite da icone, avendo cura di includere all’interno del link un elemento `` contenente la descrizione per non gli screen reader del tab con classe `.visually-hidden` per nascondere la descrizione agli altri browser. Al componente `TabNavLink` contenente l’icona va aggiunta la classe `.justify-content-end` per allineare l’icona a destra. + + + +## Posizione dei Tab + +Per questioni di accessibilità è preferibile utilizzare le proprietà Flex di CSS a un cambio di posizione dei Tab nell’HTML. Per garantire un flusso di lettura naturale della pagina i tab di navigazione devono precedere il contenuto a loro associato. + +### Orizzontale in fondo + +Per posizionare i tab al di sotto del contenuto è necessario utilizzare un elemento contenitore (ad esempio un `
    `) con classi `.d-flex .flex-column-reverse`. + + + +### Verticale a destra + +Per posizionare i tab verticali a destra contenuto è necessario applicare la classe `.flex-row-reverse` all’elemento con classe `.row` che li contiene. + + + +## Tab con sfondo scuro + +### Tab orizzontali a tutta larghezza + + + +### Tab con testo e icona + + + +### Tab verticali + + + +### Tab verticali destra + + + +## Tab tipo Card + +Aggiungere la proprietà `card` al componente `TabNav` per ottenere un design tipo card. + + + +### Tab tipo Card con pulsanti aggiungi/elimina + + diff --git a/test/__snapshots__/Storybook.test.tsx.snap b/test/__snapshots__/Storybook.test.tsx.snap index 9dedc8a75..92d01aa1a 100644 --- a/test/__snapshots__/Storybook.test.tsx.snap +++ b/test/__snapshots__/Storybook.test.tsx.snap @@ -13555,170 +13555,149 @@ exports[`Stories Snapshots Documentazione/Componenti/Stepper/Mobile SaveButton 1
    `; -exports[`Stories Snapshots Documentazione/Componenti/Tab Bottone 1`] = ` +exports[`Stories Snapshots Documentazione/Componenti/Tab Hidescroll 1`] = `
    -`; - -exports[`Stories Snapshots Documentazione/Componenti/Tab Esempi 1`] = ` -
    -
    - @@ -13728,158 +13707,2846 @@ exports[`Stories Snapshots Documentazione/Componenti/Tab Esempi 1`] = ` exports[`Stories Snapshots Documentazione/Componenti/Tab Icona 1`] = `
    -
    - -
    + + +
    -
    + + + + Tab titolo 1 + + + + +
    `; -exports[`Stories Snapshots Documentazione/Componenti/Tab Testo 1`] = ` +exports[`Stories Snapshots Documentazione/Componenti/Tab IconaFull 1`] = `
    - +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab IconaGrande 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab IconaGrandeFull 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab IconaGrandePannel 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab IconaPannel 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIcona 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIconaFull 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIconaFullDark 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIconaPannel 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIconaPannelVertical 1`] = ` +
    +
    + +
    +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIconaPannelVerticalDark 1`] = ` +
    +
    + +
    +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestoPannelVertical 1`] = ` +
    +
    + +
    +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab Testuale 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualeFull 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualeFullDark 1`] = ` + +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannel 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelCard 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelCardEditable 1`] = ` +
    + +
    +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelReverseBottom 1`] = ` +
    +
    + @@ -13887,42 +16554,196 @@ exports[`Stories Snapshots Documentazione/Componenti/Tab Testo 1`] = ` class="tab-content" >
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. + Contenuto 3
    - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. + Contenuto 4
    +
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelReverseRight 1`] = ` +
    +
    + +
    - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    `; -exports[`Stories Snapshots Documentazione/Componenti/Tab TestoIcona 1`] = ` +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelReverseRightDark 1`] = `
    -
    - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    -
    +
    +
    +`; + +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelVertical 1`] = ` +
    +
    +
    +
    + + + + + +
    +
    - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. +
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4 +
    `; -exports[`Stories Snapshots Documentazione/Componenti/Tab Verticale 1`] = ` +exports[`Stories Snapshots Documentazione/Componenti/Tab TestualePannelVerticalBackground 1`] = `
    -
    - -
    + Link + + + + +
    +
    -
    - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. -
    -
    - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. -
    -
    - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. -
    + Contenuto 1 +
    +
    + Contenuto 2 +
    +
    + Contenuto 3 +
    +
    + Contenuto 4
    @@ -32994,7 +36033,7 @@ exports[`Stories Snapshots Documentazione/Organizzare i contenuti/Liste di Immag class="card-columns" >
    =16.9.11": version "18.2.75" resolved "https://registry.yarnpkg.com/@types/react/-/react-18.2.75.tgz#45d18f384939306d35312def1bf532eb38a68562" integrity sha512-+DNnF7yc5y0bHkBTiLKqXFe+L4B3nvOphiMY3tuA5X10esmjqk7smyBZzbGTy2vsiy/Bnzj8yFIBL8xhRacoOg== @@ -2639,6 +2682,11 @@ resolved "https://registry.yarnpkg.com/@types/uuid/-/uuid-9.0.8.tgz#7545ba4fc3c003d6c756f651f3bf163d8f0f29ba" integrity sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA== +"@types/warning@^3.0.0": + version "3.0.3" + resolved "https://registry.yarnpkg.com/@types/warning/-/warning-3.0.3.tgz#d1884c8cc4a426d1ac117ca2611bf333834c6798" + integrity sha512-D1XC7WK8K+zZEveUPY+cf4+kgauk8N4eHr/XIHXGlGYkHLud6hK9lYfZk1ry1TNh798cZUCgb6MqGEG8DkJt6Q== + "@types/yargs-parser@*": version "21.0.3" resolved "https://registry.yarnpkg.com/@types/yargs-parser/-/yargs-parser-21.0.3.tgz#815e30b786d2e8f0dcd85fd5bcf5e1a04d008f15" @@ -3385,7 +3433,7 @@ cjs-module-lexer@^1.0.0: resolved "https://registry.yarnpkg.com/cjs-module-lexer/-/cjs-module-lexer-1.4.1.tgz#707413784dbb3a72aa11c2f2b042a0bef4004170" integrity sha512-cuSVIHi9/9E/+821Qjdvngor+xpnlwnuwIyZOaLmHBVdXL+gP+I6QQB9VkO7RI77YIcTV+S1W9AreJ5eN63JBA== -classnames@^2.2.3, classnames@^2.3.1: +classnames@^2.2.3, classnames@^2.3.1, classnames@^2.3.2: version "2.5.1" resolved "https://registry.yarnpkg.com/classnames/-/classnames-2.5.1.tgz#ba774c614be0f016da105c858e7159eae8e7687b" integrity sha512-saHYOzhIQs6wy2sVxTM6bUDsQO4F50V9RQ22qBpEdCW+I+/Wmke2HOl6lS6dTpdxVhb88/I6+Hs+438c3lfUow== @@ -3737,7 +3785,7 @@ dom-accessibility-api@^0.6.3: resolved "https://registry.yarnpkg.com/dom-accessibility-api/-/dom-accessibility-api-0.6.3.tgz#993e925cc1d73f2c662e7d75dd5a5445259a8fd8" integrity sha512-7ZgogeTnjuHbo+ct10G9Ffp0mif17idi0IyWNVA/wcwcm7NPOD/WEHVP3n7n3MhXqxoIYm8d6MuZohYWIZ4T3w== -dom-helpers@^5.0.1: +dom-helpers@^5.0.1, dom-helpers@^5.2.0, dom-helpers@^5.2.1: version "5.2.1" resolved "https://registry.yarnpkg.com/dom-helpers/-/dom-helpers-5.2.1.tgz#d9400536b2bf8225ad98fe052e029451ac40e902" integrity sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA== @@ -4647,6 +4695,13 @@ ini@^4.1.2, ini@^4.1.3: resolved "https://registry.yarnpkg.com/ini/-/ini-4.1.3.tgz#4c359675a6071a46985eb39b14e4a2c0ec98a795" integrity sha512-X7rqawQBvfdjS10YU1y1YVreA3SsLrW9dX2CewP2EbBJM4ypVNLDkO5y04gejPwKIY9lR+7r9gn3rFPt/kmWFg== +invariant@^2.2.4: + version "2.2.4" + resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.4.tgz#610f3c92c9359ce1db616e538008d23ff35158e6" + integrity sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA== + dependencies: + loose-envify "^1.0.0" + is-alphabetical@^1.0.0: version "1.0.4" resolved "https://registry.yarnpkg.com/is-alphabetical/-/is-alphabetical-1.0.4.tgz#9e7d6b94916be22153745d184c298cbf986a686d" @@ -6744,7 +6799,15 @@ prompts@^2.0.1: kleur "^3.0.3" sisteransi "^1.0.5" -prop-types@^15.5.8, prop-types@^15.6.2: +prop-types-extra@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/prop-types-extra/-/prop-types-extra-1.1.1.tgz#58c3b74cbfbb95d304625975aa2f0848329a010b" + integrity sha512-59+AHNnHYCdiC+vMwY52WmvP5dM3QLeoumYuEyceQDi9aEhtwN9zIQ2ZNo25sMyXnbh32h+P1ezDsUpUH3JAew== + dependencies: + react-is "^16.3.2" + warning "^4.0.0" + +prop-types@^15.5.8, prop-types@^15.6.2, prop-types@^15.8.1: version "15.8.1" resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5" integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg== @@ -6785,6 +6848,24 @@ quick-lru@^4.0.1: resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-4.0.1.tgz#5b8878f113a58217848c6482026c73e1ba57727f" integrity sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g== +react-bootstrap@^2.10.6: + version "2.10.6" + resolved "https://registry.yarnpkg.com/react-bootstrap/-/react-bootstrap-2.10.6.tgz#cb8b6f3604480b99b1e3cfa09cf53446e760bba7" + integrity sha512-fNvKytSp0nHts1WRnRBJeBEt+I9/ZdrnhIjWOucEduRNvFRU1IXjZueDdWnBiqsTSJ7MckQJi9i/hxGolaRq+g== + dependencies: + "@babel/runtime" "^7.24.7" + "@restart/hooks" "^0.4.9" + "@restart/ui" "^1.9.0" + "@types/react-transition-group" "^4.4.6" + classnames "^2.3.2" + dom-helpers "^5.2.1" + invariant "^2.2.4" + prop-types "^15.8.1" + prop-types-extra "^1.1.0" + react-transition-group "^4.4.5" + uncontrollable "^7.2.1" + warning "^4.0.3" + react-confetti@^6.1.0: version "6.1.0" resolved "https://registry.yarnpkg.com/react-confetti/-/react-confetti-6.1.0.tgz#03dc4340d955acd10b174dbf301f374a06e29ce6" @@ -6826,7 +6907,7 @@ react-fast-compare@^3.0.1: resolved "https://registry.yarnpkg.com/react-fast-compare/-/react-fast-compare-3.2.2.tgz#929a97a532304ce9fee4bcae44234f1ce2c21d49" integrity sha512-nsO+KSNgo1SbJqJEYRE9ERzo7YtYbou/OqjSQKxV7jcKox7+usiUVZOAC+XnDOABXggQTno0Y1CpVnuWEc1boQ== -react-is@^16.13.1: +react-is@^16.13.1, react-is@^16.3.2: version "16.13.1" resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4" integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ== @@ -6846,6 +6927,11 @@ react-layout-masonry@^1.2.0: resolved "https://registry.yarnpkg.com/react-layout-masonry/-/react-layout-masonry-1.2.0.tgz#675e4032c11715a5b47da2de8e5c59ba0fd72a12" integrity sha512-Nauk/lE+1boEDFonOea6zWXq4QduCS3p+bOsH2iYVSqs5RBtn6yyeeJnPN++EAZsrUNT3Blvlac1CcoHlJ98Vg== +react-lifecycles-compat@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/react-lifecycles-compat/-/react-lifecycles-compat-3.0.4.tgz#4f1a273afdfc8f3488a8c516bfda78f872352362" + integrity sha512-fBASbA6LnOU9dOU2eW7aQ8xmYBSXUIWr+UmF9b1efZBazGNO+rcXT/icdKnYm2pTwcRylVUYwW7H1PHfLekVzA== + react-popper@^2.2.4: version "2.3.0" resolved "https://registry.yarnpkg.com/react-popper/-/react-popper-2.3.0.tgz#17891c620e1320dce318bad9fede46a5f71c70ba" @@ -7375,7 +7461,16 @@ string-length@^4.0.1: char-regex "^1.0.2" strip-ansi "^6.0.0" -"string-width-cjs@npm:string-width@^4.2.0", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: +"string-width-cjs@npm:string-width@^4.2.0": + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: version "4.2.3" resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== @@ -7426,7 +7521,14 @@ stringify-entities@^4.0.0: character-entities-html4 "^2.0.0" character-entities-legacy "^3.0.0" -"strip-ansi-cjs@npm:strip-ansi@^6.0.1", strip-ansi@^6.0.0, strip-ansi@^6.0.1: +"strip-ansi-cjs@npm:strip-ansi@^6.0.1": + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-ansi@^6.0.0, strip-ansi@^6.0.1: version "6.0.1" resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== @@ -7687,7 +7789,7 @@ tsconfig-paths@^4.2.0: minimist "^1.2.6" strip-bom "^3.0.0" -tslib@^2.0.1, tslib@^2.4.0, tslib@^2.6.2: +tslib@^2.0.1, tslib@^2.4.0, tslib@^2.6.2, tslib@^2.8.0: version "2.8.1" resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.8.1.tgz#612efe4ed235d567e8aba5f2a5fab70280ade83f" integrity sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w== @@ -7790,6 +7892,21 @@ typescript@^5.4.5: resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.7.2.tgz#3169cf8c4c8a828cde53ba9ecb3d2b1d5dd67be6" integrity sha512-i5t66RHxDvVN40HfDd1PsEThGNnlMCMT3jMUuoh9/0TaqWevNontacunWyN02LA9/fIbEWlcHZcgTKb9QoaLfg== +uncontrollable@^7.2.1: + version "7.2.1" + resolved "https://registry.yarnpkg.com/uncontrollable/-/uncontrollable-7.2.1.tgz#1fa70ba0c57a14d5f78905d533cf63916dc75738" + integrity sha512-svtcfoTADIB0nT9nltgjujTi7BzVmwjZClOmskKu/E8FW9BXzg9os8OLr4f8Dlnk0rYWJIWr4wv9eKUXiQvQwQ== + dependencies: + "@babel/runtime" "^7.6.3" + "@types/react" ">=16.9.11" + invariant "^2.2.4" + react-lifecycles-compat "^3.0.4" + +uncontrollable@^8.0.1: + version "8.0.4" + resolved "https://registry.yarnpkg.com/uncontrollable/-/uncontrollable-8.0.4.tgz#a0a8307f638795162fafd0550f4a1efa0f8c5eb6" + integrity sha512-ulRWYWHvscPFc0QQXvyJjY6LIXU56f0h8pQFvhxiKk5V1fcI8gp9Ht9leVAhrVjzqMw0BgjspBINx9r6oyJUvQ== + undici-types@~6.19.2: version "6.19.8" resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.19.8.tgz#35111c9d1437ab83a7cdc0abae2f26d88eda0a02" @@ -8127,7 +8244,7 @@ walker@^1.0.8: dependencies: makeerror "1.0.12" -warning@^4.0.2: +warning@^4.0.0, warning@^4.0.2, warning@^4.0.3: version "4.0.3" resolved "https://registry.yarnpkg.com/warning/-/warning-4.0.3.tgz#16e9e077eb8a86d6af7d64aa1e05fd85b4678ca3" integrity sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w== @@ -8213,7 +8330,16 @@ word-wrap@^1.2.5: resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34" integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA== -"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0", wrap-ansi@^7.0.0: +"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0": + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrap-ansi@^7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==
    -
    - -
    -
    -
    - Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. -
    -
    - Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. -
    -
    - Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. -
    -
    -
    -
    -
    -