diff --git a/packages/ffe-message-box-react/README.md b/packages/ffe-message-box-react/README.md index ba9d0731c0..e858c85bba 100644 --- a/packages/ffe-message-box-react/README.md +++ b/packages/ffe-message-box-react/README.md @@ -10,11 +10,6 @@ npm install --save @sb1/ffe-message-box-react Full documentation on message box usage is available at https://design.sparebank1.no/komponenter/meldinger/#messagebox. -## TypeScript definition files - -This component supports TypeScript - please update `index.d.ts` if you change any -of the external methods or properties in this component. - ## Development To start a local development server, run the following from the designsystem root folder: diff --git a/packages/ffe-message-box-react/package.json b/packages/ffe-message-box-react/package.json index 11de7d7ae2..d396af8fec 100644 --- a/packages/ffe-message-box-react/package.json +++ b/packages/ffe-message-box-react/package.json @@ -17,10 +17,10 @@ "url": "ssh://git@github.com:SpareBank1/designsystem.git" }, "scripts": { - "build": "ffe-buildtool babel", + "build": "ffe-buildtool tsc", "watch": "ffe-buildtool babel-watch", - "lint": "eslint src", - "lint:fix": "eslint src --fix", + "lint": "eslint src --ext ts,tsx", + "lint:fix": "eslint src --fix --ext ts,tsx", "test": "ffe-buildtool jest", "test:watch": "ffe-buildtool jest --watch" }, diff --git a/packages/ffe-message-box-react/src/BaseMessage.spec.js b/packages/ffe-message-box-react/src/BaseMessage.spec.js deleted file mode 100644 index a82becac25..0000000000 --- a/packages/ffe-message-box-react/src/BaseMessage.spec.js +++ /dev/null @@ -1,56 +0,0 @@ -import React from 'react'; -import BaseMessage from './BaseMessage'; -import { Icon } from '@sb1/ffe-icons-react'; -const defaultProps = { - type: 'info', - icon: , -}; -const getWrapper = props => - shallow(); - -describe('', () => { - it('renders without exploding', () => { - const wrapper = getWrapper(); - expect(wrapper.exists()).toBe(true); - }); - it('renders the correct classes by default', () => { - const wrapper = getWrapper({ title: 'test title' }); - expect(wrapper.hasClass('ffe-message-box')).toBe(true); - - ['info', 'error', 'success', 'tips'].forEach(type => { - const typedWrapper = getWrapper({ type, title: 'test' }); - expect( - typedWrapper.find(`.ffe-message-box--${type}`).exists(), - ).toBe(true); - expect(typedWrapper.find('.ffe-message-box__icon').exists()).toBe( - true, - ); - }); - }); - it('renders with correct title class', () => { - const wrapper = getWrapper({ title: 'test title' }); - expect(wrapper.find(`.ffe-message-box__title`).exists()).toBe(true); - }); - it('renders an icon by default', () => { - const wrapper = getWrapper(); - expect(wrapper.find('.ffe-message-box__icon').exists()).toBe(true); - }); - it('renders the container with role="group" by default', () => { - const wrapper = getWrapper(); - expect(wrapper.prop('role')).toBe('group'); - }); - it('renders a title if specified', () => { - const wrapper = getWrapper({ title: 'test title' }); - expect(wrapper.find('.ffe-message-box__title').text()).toBe( - 'test title', - ); - }); - it('renders children if specified', () => { - const wrapper = getWrapper({ children:

children

}); - expect(wrapper.text()).toContain('children'); - }); - it('renders onColoredBg styling if specified', () => { - const wrapper = getWrapper({ onColoredBg: true }); - expect(wrapper.find('.ffe-message-box--coloredbg').exists()).toBe(true); - }); -}); diff --git a/packages/ffe-message-box-react/src/BaseMessage.spec.tsx b/packages/ffe-message-box-react/src/BaseMessage.spec.tsx new file mode 100644 index 0000000000..4da69f8d90 --- /dev/null +++ b/packages/ffe-message-box-react/src/BaseMessage.spec.tsx @@ -0,0 +1,81 @@ +import React from 'react'; +import { BaseMessage, BaseMessageProps } from './BaseMessage'; +import { Icon } from '@sb1/ffe-icons-react'; +import { render, screen } from '@testing-library/react'; + +const defaultProps = { + type: 'info' as const, + icon: , +}; + +const TEST_ID = 'TEST_ID'; + +const renderBaseMessage = (props?: Partial) => + render(); + +describe('', () => { + it('renders the correct classes by default', () => { + const { rerender, container } = render( + , + ); + const baseMessage = screen.getByTestId(TEST_ID); + expect(baseMessage.classList.contains('ffe-message-box')).toBe(true); + + const types = ['info', 'error', 'success', 'tips'] as const; + + types.forEach(type => { + rerender( + , + ); + expect( + baseMessage.classList.contains(`ffe-message-box--${type}`), + ).toBe(true); + expect( + container.querySelector(`.ffe-message-box__icon`), + ).toBeTruthy(); + }); + }); + + it('renders with correct title class', () => { + const { container } = renderBaseMessage({ title: 'test title' }); + expect(container.querySelector(`.ffe-message-box__title`)).toBeTruthy(); + }); + it('renders an icon by default', () => { + const { container } = renderBaseMessage(); + expect(container.querySelector(`.ffe-message-box__icon`)).toBeTruthy(); + }); + it('renders the container with role="group" by default', () => { + renderBaseMessage(); + const baseMessage = screen.getByTestId(TEST_ID); + + expect(baseMessage.getAttribute('role')).toBe('group'); + }); + it('renders a title if specified', () => { + const { container } = renderBaseMessage({ title: 'test title' }); + + expect( + container.querySelector(`.ffe-message-box__title`)?.textContent, + ).toBe('test title'); + }); + it('renders children if specified', () => { + renderBaseMessage({ children:

children

}); + const baseMessage = screen.getByTestId(TEST_ID); + + expect(baseMessage.textContent).toContain('children'); + }); + it('renders onColoredBg styling if specified', () => { + const { container } = renderBaseMessage({ onColoredBg: true }); + expect( + container.querySelector('.ffe-message-box--coloredbg'), + ).toBeTruthy(); + }); +}); diff --git a/packages/ffe-message-box-react/src/BaseMessage.js b/packages/ffe-message-box-react/src/BaseMessage.tsx similarity index 54% rename from packages/ffe-message-box-react/src/BaseMessage.js rename to packages/ffe-message-box-react/src/BaseMessage.tsx index 1efbb8057a..1622069569 100644 --- a/packages/ffe-message-box-react/src/BaseMessage.js +++ b/packages/ffe-message-box-react/src/BaseMessage.tsx @@ -1,21 +1,35 @@ import React from 'react'; -import { bool, node, oneOf, string } from 'prop-types'; import classNames from 'classnames'; -const BaseMessage = props => { - const { - type, - title, - titleElement = 'h2', - icon, - children, - className = '', - onColoredBg = false, - ariaLabel, - role = 'group', - ...rest - } = props; +export interface BaseMessageProps + extends Omit, 'title'> { + icon?: React.ReactNode; + title?: React.ReactNode; + /** HTML element for the title */ + titleElement?: string; + /** + * Internal type enum for specifying the style of message box. Should not be used directly + * @ignore + */ + type: 'success' | 'error' | 'tips' | 'info'; + /** Adds alternative styling for better contrast on certain backgrounds */ + onColoredBg?: boolean; + ariaLabel?: string; + role?: 'alert' | 'group'; +} +export const BaseMessage: React.FC = ({ + type, + title, + titleElement = 'h2', + icon, + children, + className = '', + onColoredBg = false, + ariaLabel, + role = 'group', + ...rest +}) => { return (
{ {...rest} > - {React.cloneElement(icon, { - ...icon.props, - size: 'xl', - weight: 300, - })} + {React.isValidElement(icon) && + React.cloneElement(icon, { + ...icon.props, + size: 'xl', + weight: 300, + })}
{title && @@ -49,23 +64,3 @@ const BaseMessage = props => {
); }; - -BaseMessage.propTypes = { - children: node, - className: string, - icon: node.isRequired, - title: node, - /** HTML element for the title */ - titleElement: string, - /** - * Internal type enum for specifying the style of message box. Should not be used directly - * @ignore - */ - type: oneOf(['success', 'error', 'tips', 'info']).isRequired, - /* Adds alternative styling for better contrast on certain backgrounds */ - onColoredBg: bool, - ariaLabel: string, - role: oneOf(['alert', 'group']), -}; - -export default BaseMessage; diff --git a/packages/ffe-message-box-react/src/ErrorMessage.js b/packages/ffe-message-box-react/src/ErrorMessage.tsx similarity index 65% rename from packages/ffe-message-box-react/src/ErrorMessage.js rename to packages/ffe-message-box-react/src/ErrorMessage.tsx index b6919a7142..1eade48f19 100644 --- a/packages/ffe-message-box-react/src/ErrorMessage.js +++ b/packages/ffe-message-box-react/src/ErrorMessage.tsx @@ -1,12 +1,20 @@ import React from 'react'; -import { node, string, bool, oneOf } from 'prop-types'; import { Icon } from '@sb1/ffe-icons-react'; -import BaseMessage from './BaseMessage'; -import texts from './texts'; +import { BaseMessage, BaseMessageProps } from './BaseMessage'; +import { texts } from './texts'; -const ErrorMessage = props => { - const { alert = true, locale = 'nb', ...rest } = props; +export interface ErrorMessageProps + extends Omit { + /** When false, role is not set to alert, avoids message from being read up immediately after page load. Default value is true. */ + alert?: boolean; + locale?: 'nb' | 'nn' | 'en'; +} +export const ErrorMessage: React.FC = ({ + alert = true, + locale = 'nb', + ...rest +}) => { const priorityHighIcon = ''; @@ -20,20 +28,3 @@ const ErrorMessage = props => { /> ); }; - -ErrorMessage.propTypes = { - /** The content of the message box */ - children: node, - /** Any extra class names to the wrapping DOM node */ - className: string, - /** The icon to show. Has a default value, but can be overridden */ - icon: node, - /** An optional title for the message */ - title: node, - /** When false, role is not set to alert, avoids message from being read up immediately after page load. Default value is true. */ - alert: bool, - /** 'nb', 'nn', or 'en' */ - locale: oneOf(['en', 'nb', 'nn']), -}; - -export default ErrorMessage; diff --git a/packages/ffe-message-box-react/src/InfoMessage.js b/packages/ffe-message-box-react/src/InfoMessage.tsx similarity index 64% rename from packages/ffe-message-box-react/src/InfoMessage.js rename to packages/ffe-message-box-react/src/InfoMessage.tsx index d17c92cf03..ef4746ca68 100644 --- a/packages/ffe-message-box-react/src/InfoMessage.js +++ b/packages/ffe-message-box-react/src/InfoMessage.tsx @@ -1,10 +1,17 @@ import React from 'react'; -import { node, oneOf, string } from 'prop-types'; import { Icon } from '@sb1/ffe-icons-react'; -import BaseMessage from './BaseMessage'; -import texts from './texts'; +import { BaseMessage, BaseMessageProps } from './BaseMessage'; +import { texts } from './texts'; -const InfoMessage = ({ locale = 'nb', ...rest }) => { +export interface InfoMessageProps + extends Omit { + locale?: 'nb' | 'nn' | 'en'; +} + +export const InfoMessage: React.FC = ({ + locale = 'nb', + ...rest +}) => { const infoIcon = ''; return ( @@ -16,18 +23,3 @@ const InfoMessage = ({ locale = 'nb', ...rest }) => { /> ); }; - -InfoMessage.propTypes = { - /** The content of the message box */ - children: node, - /** Any extra class names to the wrapping DOM node */ - className: string, - /** The icon to show. Has a default value, but can be overridden */ - icon: node, - /** An optional title for the message */ - title: node, - /** 'nb', 'nn', or 'en' */ - locale: oneOf(['en', 'nb', 'nn']), -}; - -export default InfoMessage; diff --git a/packages/ffe-message-box-react/src/InfoMessageList.js b/packages/ffe-message-box-react/src/InfoMessageList.js deleted file mode 100644 index c20253aacc..0000000000 --- a/packages/ffe-message-box-react/src/InfoMessageList.js +++ /dev/null @@ -1,25 +0,0 @@ -import React from 'react'; -import { string, node } from 'prop-types'; - -export const InfoMessageListItem = props => ( -
  • - {props.href && ( - - {props.children} - - )} - {!props.href && props.children} -
  • -); -InfoMessageListItem.propTypes = { - href: string, - children: string.isRequired, -}; - -export const InfoMessageList = props => ( -
      {props.children}
    -); - -InfoMessageList.propTypes = { - children: node.isRequired, -}; diff --git a/packages/ffe-message-box-react/src/InfoMessageList.tsx b/packages/ffe-message-box-react/src/InfoMessageList.tsx new file mode 100644 index 0000000000..b8e21387ee --- /dev/null +++ b/packages/ffe-message-box-react/src/InfoMessageList.tsx @@ -0,0 +1,29 @@ +import React from 'react'; + +export interface InfoMessageListItemProps { + href: string; + children: NonNullable; +} + +export const InfoMessageListItem: React.FC = ({ + children, + href, +}) => ( +
  • + {href ? ( + + {children} + + ) : ( + children + )} +
  • +); + +export interface InfoMessageListProps { + children: NonNullable; +} + +export const InfoMessageList: React.FC = ({ + children, +}) =>
      {children}
    ; diff --git a/packages/ffe-message-box-react/src/SuccessMessage.js b/packages/ffe-message-box-react/src/SuccessMessage.tsx similarity index 60% rename from packages/ffe-message-box-react/src/SuccessMessage.js rename to packages/ffe-message-box-react/src/SuccessMessage.tsx index e0b9dceb80..017387caac 100644 --- a/packages/ffe-message-box-react/src/SuccessMessage.js +++ b/packages/ffe-message-box-react/src/SuccessMessage.tsx @@ -1,10 +1,17 @@ import React from 'react'; -import { node, oneOf, string } from 'prop-types'; import { Icon } from '@sb1/ffe-icons-react'; -import BaseMessage from './BaseMessage'; -import texts from './texts'; +import { BaseMessage, BaseMessageProps } from './BaseMessage'; +import { texts } from './texts'; -const SuccessMessage = ({ locale = 'nb', ...rest }) => { +export interface SuccessMessageProps + extends Omit { + locale?: 'nb' | 'nn' | 'en'; +} + +export const SuccessMessage: React.FC = ({ + locale = 'nb', + ...rest +}) => { const checkIcon = ''; @@ -17,18 +24,3 @@ const SuccessMessage = ({ locale = 'nb', ...rest }) => { /> ); }; - -SuccessMessage.propTypes = { - /** The content of the message box */ - children: node, - /** Any extra class names to the wrapping DOM node */ - className: string, - /** The icon to show. Has a default value, but can be overridden */ - icon: node, - /** An optional title for the message */ - title: node, - /** 'nb', 'nn', or 'en' */ - locale: oneOf(['en', 'nb', 'nn']), -}; - -export default SuccessMessage; diff --git a/packages/ffe-message-box-react/src/TipsMessage.js b/packages/ffe-message-box-react/src/TipsMessage.tsx similarity index 70% rename from packages/ffe-message-box-react/src/TipsMessage.js rename to packages/ffe-message-box-react/src/TipsMessage.tsx index 734fedcfd4..865700afc8 100644 --- a/packages/ffe-message-box-react/src/TipsMessage.js +++ b/packages/ffe-message-box-react/src/TipsMessage.tsx @@ -1,12 +1,17 @@ import React from 'react'; -import { node, oneOf, string } from 'prop-types'; import { Icon } from '@sb1/ffe-icons-react'; -import BaseMessage from './BaseMessage'; -import texts from './texts'; +import { BaseMessage, BaseMessageProps } from './BaseMessage'; +import { texts } from './texts'; -const TipsMessage = props => { - const { locale = 'nb', ...rest } = props; +export interface TipsMessageProps + extends Omit { + locale?: 'nb' | 'nn' | 'en'; +} +export const TipsMessage: React.FC = ({ + locale = 'nb', + ...rest +}) => { const lightbulbIcon = ''; return ( @@ -18,18 +23,3 @@ const TipsMessage = props => { /> ); }; - -TipsMessage.propTypes = { - /** The content of the message box */ - children: node, - /** Any extra class names to the wrapping DOM node */ - className: string, - /** The icon to show. Has a default value, but can be overridden */ - icon: node, - /** An optional title for the message */ - title: node, - /** 'nb', 'nn', or 'en' */ - locale: oneOf(['en', 'nb', 'nn']), -}; - -export default TipsMessage; diff --git a/packages/ffe-message-box-react/src/index.d.ts b/packages/ffe-message-box-react/src/index.d.ts deleted file mode 100644 index 039deea258..0000000000 --- a/packages/ffe-message-box-react/src/index.d.ts +++ /dev/null @@ -1,38 +0,0 @@ -import * as React from 'react'; - -export interface MessageBoxProps - extends Omit, 'title' | 'content'> { - children?: React.ReactNode; - className?: string; - icon?: React.ReactNode; - title?: React.ReactNode; - titleElement?: string; - onColoredBg?: boolean; - locale?: 'nb' | 'nn' | 'en'; -} - -export interface ErrorMessageBoxProps extends MessageBoxProps { - alert?: boolean; -} - -export interface InfoMessageListItemProps { - children: string; - href?: string; -} - -export interface InfoMessageListProps { - children: React.ReactNode; -} - -declare class SuccessMessage extends React.Component {} -declare class ErrorMessage extends React.Component {} -declare class InfoMessage extends React.Component {} -declare class TipsMessage extends React.Component {} -declare class InfoMessageList extends React.Component< - InfoMessageListProps, - any -> {} -declare class InfoMessageListItem extends React.Component< - InfoMessageListItemProps, - any -> {} diff --git a/packages/ffe-message-box-react/src/index.js b/packages/ffe-message-box-react/src/index.js deleted file mode 100644 index 1f8d26110d..0000000000 --- a/packages/ffe-message-box-react/src/index.js +++ /dev/null @@ -1,14 +0,0 @@ -import SuccessMessage from './SuccessMessage'; -import ErrorMessage from './ErrorMessage'; -import InfoMessage from './InfoMessage'; -import TipsMessage from './TipsMessage'; -import { InfoMessageList, InfoMessageListItem } from './InfoMessageList'; - -export { - SuccessMessage, - ErrorMessage, - InfoMessage, - TipsMessage, - InfoMessageList, - InfoMessageListItem, -}; diff --git a/packages/ffe-message-box-react/src/index.ts b/packages/ffe-message-box-react/src/index.ts new file mode 100644 index 0000000000..1f135e22d6 --- /dev/null +++ b/packages/ffe-message-box-react/src/index.ts @@ -0,0 +1,10 @@ +export { SuccessMessage, SuccessMessageProps } from './SuccessMessage'; +export { ErrorMessage, ErrorMessageProps } from './ErrorMessage'; +export { InfoMessage, InfoMessageProps } from './InfoMessage'; +export { TipsMessage, TipsMessageProps } from './TipsMessage'; +export { + InfoMessageList, + InfoMessageListProps, + InfoMessageListItem, + InfoMessageListItemProps, +} from './InfoMessageList'; diff --git a/packages/ffe-message-box-react/src/texts.js b/packages/ffe-message-box-react/src/texts.ts similarity index 90% rename from packages/ffe-message-box-react/src/texts.js rename to packages/ffe-message-box-react/src/texts.ts index f61c96eb91..14321a2c1d 100644 --- a/packages/ffe-message-box-react/src/texts.js +++ b/packages/ffe-message-box-react/src/texts.ts @@ -11,7 +11,7 @@ const nb = { tip: { ariaLabel: 'Tipsmelding', }, -}; +} as const; const nn = { error: { ariaLabel: 'Feilmelding', @@ -25,7 +25,7 @@ const nn = { tip: { ariaLabel: 'Tipsmelding', }, -}; +} as const; const en = { error: { ariaLabel: 'Error message', @@ -39,6 +39,6 @@ const en = { tip: { ariaLabel: 'Tip message', }, -}; +} as const; -export default { nb, nn, en }; +export const texts = { nb, nn, en }; diff --git a/packages/ffe-message-box-react/tsconfig.cjs.json b/packages/ffe-message-box-react/tsconfig.cjs.json new file mode 100644 index 0000000000..6579fd2246 --- /dev/null +++ b/packages/ffe-message-box-react/tsconfig.cjs.json @@ -0,0 +1,9 @@ +{ + "extends": "../../tsconfig.json", + "compilerOptions": { + "outDir": "./lib", + "module": "commonjs" + }, + "include": ["src/**/*"], + "exclude": ["node_modules", "src/**/*.spec.ts*"] +} diff --git a/packages/ffe-message-box-react/tsconfig.esm.json b/packages/ffe-message-box-react/tsconfig.esm.json new file mode 100644 index 0000000000..8e577796bf --- /dev/null +++ b/packages/ffe-message-box-react/tsconfig.esm.json @@ -0,0 +1,9 @@ +{ + "extends": "../../tsconfig.json", + "compilerOptions": { + "outDir": "./es", + "module": "esnext" + }, + "include": ["src/**/*"], + "exclude": ["node_modules", "src/**/*.spec.ts*"] +} diff --git a/packages/ffe-message-box-react/tsconfig.types.json b/packages/ffe-message-box-react/tsconfig.types.json new file mode 100644 index 0000000000..3499c0be03 --- /dev/null +++ b/packages/ffe-message-box-react/tsconfig.types.json @@ -0,0 +1,10 @@ +{ + "extends": "../../tsconfig.json", + "compilerOptions": { + "outDir": "./types", + "declaration": true, + "emitDeclarationOnly": true + }, + "include": ["src/**/*"], + "exclude": ["node_modules", "src/**/*.spec.ts*"] +}