diff --git a/.storybook/custom-styles.scss b/.storybook/custom-styles.scss index 937502af1f..42071df8c0 100644 --- a/.storybook/custom-styles.scss +++ b/.storybook/custom-styles.scss @@ -3,7 +3,7 @@ // Add custom styles for storybook implementation .custom-class { - background: #fa9441; + background: #bf5906; } .usa-color-text-visited { diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cf569ccb9..ff269d9fa8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,33 @@ All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. +## [5.5.0](https://github.com/trussworks/react-uswds/compare/5.4.0...5.5.0) (2023-10-20) + + +### Features + +* added 4 icons ([#2525](https://github.com/trussworks/react-uswds/issues/2525)) ([c489814](https://github.com/trussworks/react-uswds/commit/c48981442b4bb65b403dc2009ed3cd28a72037a9)) +* Added LanguageDefinition to exports ([#2570](https://github.com/trussworks/react-uswds/issues/2570)) ([0f4688a](https://github.com/trussworks/react-uswds/commit/0f4688ae06afadd595689055750e65624868b1bd)) +* Added required marker ([#2524](https://github.com/trussworks/react-uswds/issues/2524)) ([c5f2b40](https://github.com/trussworks/react-uswds/commit/c5f2b40b4b79784340f66d66fceea365922648f8)) +* Allow Tooltip label to be a string or React component ([#2596](https://github.com/trussworks/react-uswds/issues/2596)) ([bc01008](https://github.com/trussworks/react-uswds/commit/bc0100816a0bcba797562a02ea3b32543f850904)) +* changed template links to buttons ([#2526](https://github.com/trussworks/react-uswds/issues/2526)) ([a4b8423](https://github.com/trussworks/react-uswds/commit/a4b8423f4d3a4b8dcb89b8dab8dea4363c6b567a)) +* replaced memorable date month input with select ([#2527](https://github.com/trussworks/react-uswds/issues/2527)) ([d38b31b](https://github.com/trussworks/react-uswds/commit/d38b31be9454a03a4605f69335daf19d376020a7)) + + +### Bug Fixes + +* Language selector: add prop for display lang for bug fix ([#2622](https://github.com/trussworks/react-uswds/issues/2622)) ([df68d0d](https://github.com/trussworks/react-uswds/commit/df68d0d8bc273533dcb5a85a5a699aca12404d67)) +* Sass modules bug ([#2555](https://github.com/trussworks/react-uswds/issues/2555)) ([a1ed9f7](https://github.com/trussworks/react-uswds/commit/a1ed9f758c151b639726b16fb74327299f7c09a2)) +* **storybook:** password label set to password input field id on sign in ([#2618](https://github.com/trussworks/react-uswds/issues/2618)) ([eae195e](https://github.com/trussworks/react-uswds/commit/eae195e73b71adb9b7849a25f8836c244e49ad21)) +* **storybook:** sufficient color contrast set in custom class button ([#2619](https://github.com/trussworks/react-uswds/issues/2619)) ([2878822](https://github.com/trussworks/react-uswds/commit/28788220761dc92243302bac72c0bd2b97872192)) +* Updates file input stories to set appropriate html for value of label ([#2628](https://github.com/trussworks/react-uswds/issues/2628)) ([7f51150](https://github.com/trussworks/react-uswds/commit/7f51150f1a12df261b18eb1d6ecf9e8faa3c4c0e)) + + +### Documentation & Examples + +* add design PR review documentation ([#2615](https://github.com/trussworks/react-uswds/issues/2615)) ([2892424](https://github.com/trussworks/react-uswds/commit/289242407c718fcfc157488492b0a9a942de00c0)) +* adds in information on how to test PRs in GitHub Codespaces ([#2617](https://github.com/trussworks/react-uswds/issues/2617)) ([ef944f8](https://github.com/trussworks/react-uswds/commit/ef944f8054e15f374b07fa73c26be827a89f5ef1)) + ## [5.4.0](https://github.com/trussworks/react-uswds/compare/5.3.1...5.4.0) (2023-09-29) diff --git a/docs/codespace_pr_reviews.md b/docs/codespace_pr_reviews.md new file mode 100644 index 0000000000..08ed4d01bd --- /dev/null +++ b/docs/codespace_pr_reviews.md @@ -0,0 +1,41 @@ +# Using Codesapces to review PRs without pulling a repo to your machine + +1. go to the PR +2. click the `Code` button in the top right corner +3. click the `Codespaces` tab +4. `Create a new codespace on ...` + + ![Create a new codespace screenshot](https://github.com/trussworks/react-uswds/assets/59394696/9ebfff8a-353c-4064-bf70-8712bbfbbeda) + +5. **Wait** for the codespace to finish setting up, you should see stuff happening in the `Terminal`. This is the codespace installig the correct versions of packages etc will run in the environment, so it's not necessary to have those updated locally, or even have the repo on your machine locally at all. If you mess with it before it is done it might not work. If you mess this up, you might see an error about your node version. + + When it is complete, you should see a message like this in the terminal. If you don't see this, you should delete the codespace and start over. + + ![Github Codespace configuaration process](https://github.com/trussworks/react-uswds/assets/59394696/bdf6cd8d-22e2-4e6b-b778-65c83c51556a) + +6. if the yarn install in that process was not successful (should look like the image above), run `yarn install`. Wait until it is done before you do anything. takes ~5ish minutes +7. if it is successful run `yarn storybook`. this might also take a couple of minutes to complete. +8. when the link the storybook will be hosted on is available, you will get a message saying your application is running on port 9009 in the bottom right corner. click the `Open in Browser` button + + ![application running message](https://github.com/trussworks/react-uswds/assets/59394696/9d6ac91d-5ea2-4bbc-a64d-685076444354) + + If you don't see that message (it doesn't always happen) check the terminal for something like this: + ![storybook started message in the terminal](https://user-images.githubusercontent.com/13249580/272653168-b9d66c78-3659-4c18-a7b4-c6d493354135.png) + + If you see that, go to `Ports` tab in the top bar of that bottom section where your `Terminal` is. You can find a link under the column in the table labelled `Forwarded address`. Click that to open the Storybook site being hosted on Codespaces. + + ![Ports Tab table with forwarded address](https://github.com/trussworks/react-uswds/assets/59394696/a3075661-a1c2-476a-aeb9-974e6f9e036f) + +9. It should open a new tab that will have a blank window until the storybook is completely done running its build. Might take a couple of minutes. +10. When the build completes, should show you the ReactUSWDS Storybook with the changes in the PR! + + ![storybook running in codespaces dev environment](https://github.com/trussworks/react-uswds/assets/59394696/cdfc67de-711f-43c2-928a-284158744f1f) + +11. proceed as if you were running the PR on your own machine. hooray! +12. When your review is complete, you should delete the codespace. There are costs associated with storing codespaces. **You should therefore delete any codespaces you no longer need**. + +![how to delete the codespace screencap](https://github.com/trussworks/react-uswds/assets/59394696/45c1ea01-28f3-4b77-a6da-b6af81546bbc) + +## Things to watch out for + +If you disrupt the installation process when you start the Codespace, it can mess up the node installation. If this happens, your best bet is to go back to the PR, open the codespaces menu, delete the codespace, and make a new one. This will not mess up the PR. diff --git a/docs/design-pr-reviews.md b/docs/design-pr-reviews.md new file mode 100644 index 0000000000..fc2f398cbc --- /dev/null +++ b/docs/design-pr-reviews.md @@ -0,0 +1,128 @@ +# PR Review Process for react-uswds + +## Introduction + +When a developer completes a story, they create a PR (pull request) in Github. This notifies the team that their work is ready for review. This review involves making sure the work meets the acceptance criteria defined in the Issue it corresponds to, passes automated tests, and matches the expected visual design. PRs can require review from engineers, product managers and designers. This is a key step in the process to ensure we are only accepting clean, bug-free code that creates the value we intended and doesn’t introduce new problems. + +If the work meets all the criteria, reviewers will approve the PR and it will be merged into main and pushed to staging. + +Sometimes, a team member may notice something in the code that needs to be addressed, some unexpected behavior caught in testing the app, or something about the UI doesn’t look right. + +In that case, the reviewer who found the issue will leave a comment on the PR in Github or reach out directly to the engineer to talk through the issue. The engineer will make any necessary changes and request review again when ready. + +## Getting Set Up to Review PRs + +get added to the design group on github. This will tag designers for someone to review a PR if it has not been looked at by a designer. + +### Github/Repo Access + +As part of onboarding, you join the contributors of the project. When you are added, you will suddenly start to get 1 million emails (approximately) every day from Github. We recommend turning off email notifications and using the Slackbot to get a daily digest of PRs that need a design review. + +**Turn off Github email notifications** + +Navigate to your notification settings using this link: https://github.com/settings/notifications. Uncheck “email” under Participating, Watching, and Github Actions. You can also uncheck the email alerts under Dependabot alerts. + +**Set up the Github Slackbot** + +Use this link: https://github.com/settings/reminders/. Select the day/time that you want to receive a message in Slack with a list of PRs needing review (many of us like to receive this first thing in the morning to know what’s on our plate for the day). Check the boxes for “Review requests assigned to you” and “Review requests assigned to your team.” + +The messages will come through under “Apps” in the sidebar of Slack. + +**Joining the Design group on Github** + +If you are a designer joining the project to conduct reviews, you'll want to ask the maintainer to be added to the design codeowners group. This will make it easier for engineers to more easily request a review from the designers without having to check off all of our names individually or rememerbing who is currecntly in the group and willing to review PRs. This group is also automatically requested for review, and required, if any visual/design changes are detected. + +### Happo + +Happo is a visual regression testing platform we have integrated into the project to help keep track of the impact a code change might have on our design system in production. It will automatically run a test to detect any visual changes a PR might make to the components and design system elements we track in Storybook. Happo takes screenshots of all of our components in Storybook and compares differences against its collection of screenshots of what the main branch looks like with each PR. It is integrated with Github, so when an engineer creates a PR, it generates a Happo report. During onboarding, ask to pair with someone doing a PR review to get an introduction to Happo and how to use it. + +![happo diffs view in browser](https://lh4.googleusercontent.com/i_imRggob1k2hwPiJKczfZR9sHbbRYXGEUhA5ufqUtPOd2rVUjv0xDfDmVoWgSgtrLqeogg4qX0Q3GofFX0uS89EVzAAGrRb-9YMINiuOkyuOPcSVcIz0NqgYYYjNcfT2jK9K2ZZaGC3wAb7McvguKFy7aAdmVFs1hUxkpKxdhdhtC2T2Cvf8fm7_g) + +![happo changes in a PR in Github](https://lh4.googleusercontent.com/qXmkExulMtCIbUH-mw6QWQGu0MWhOS7638JymvAKmmpMfxEjh4KUsrWcjbRPEiiRdNB5Gv77kX7BNMsC6bdi1RIxYjHnKyxL-Hd8RK9MSFl5LcwOeL53lcSCegBMjzZEOf2ZFfrivjhwbRu-oIitEUIRISdCME1vnZcNqFSGzVF3N1QsUjI5hPAdww) + +**Good to know about Happo** + +Sometimes Happo flags components that are not related to the work done on the PR. these are called “[spurious diffs](https://docs.happo.io/docs/spurious-diffs)”. It may be caused by the way a screenshot was taken, showing a very slight difference in spacing, for example. Typically these can be ignored, but if you are unsure, ask the engineer who created the PR what might be causing these diffs. Generally, if you look at the before and after image and can’t find any differences, it’s okay to accept it. Happo is helpful, but it does deliver false positives sometimes. + +![meme of Pam Beesley shwoing Creed two identical photos on an episode of The Office (US) and asking him to spot the differences](https://lh3.googleusercontent.com/0CJGCVuHI2fqOhURfnV6xztELYpJUvb6iRgKYJ5uLd4HVDyWzNa9fClk74jegJ5PvTLZlrJ54OPu4OGFd7tm05OGp4DMnZdg2r-w8BWItCydmWdGZHPd8LpnzIAG0JUh9YQWCkygIIQk2ru69wLiN5pvUkXR2h-Oqr9NP5k1xiuj9ooBpaYjAmkDBA) + +![Pam informing the viewer that the photos are the same picture, so there are no differences](https://lh3.googleusercontent.com/XAvh9ARUnK-WXittJtrgDkIqns_ykebb4woQgjUJgHNmqqPA1oOOG9FXosbLSyOR-2vuCTH4DyZ1gXG2Bt9_oCQ_h4MgGY0TIk8YtveG_qdUbnaxA-2mNUJoTDlQhpZxfXJGueKbayXHo9q4KsEsmX5Maq7nYZtYQKzDRZ4ryKMK898RZeg6xqfvNQ) + +### Storybook + +Our storybook: [https://trussworks.github.io/react-uswds/](https://trussworks.github.io/react-uswds/) + +Storybook is our primary resource for everyone on the project outside of the design team for the USSF Design System’s reusable UI components. + +To run storybook locally, run the command  yarn storybook  from your command line. This will host the project storybook at localhost:9009. It can be helpful to save this to your bookmarks bar. + +**Running a PR branch for review in the browser without having the repo on your machine** + +See [Codespaces for PR reviews](docs/codespace_pr_reviews.md). + +**Reviewing a newly added component to storybook** + +You can run storybook locally to measure and evaluate the component to ensure it meets your intentions, but you can also get a preview of what’s been added in Happo. When you add to Happo, it won’t show up as a failure that needs approval, but you can still navigate in to preview the added images. + +![the location of the Added Images in Happo in the menu on the page](https://lh6.googleusercontent.com/nBN5pDS5ZVEaCt6ulj6ekgnEA_6-n8u4tPLatBkO0cyps27cSPE5c4Su0i-mSj3Byx1lK3dnn3jzJIXp7f62jLVzCdj2sQ0SH535HezOvQKl2p6KCd7WK15Oe-UGf22x_WRWsjHzOQR3Pp3-lvZ-1KfuKPyj5GQgGTNqHOWKG4RV-lxRUvhMpWkS5g) + +![What added components look like on the Added page](https://lh4.googleusercontent.com/1xepAdkBt17UZ-QG3NdVnXbtHbW64OGDire70DCeUe18XXiBHXII3Bg35nis5_BJx2WEVLiRjJQCakJ1tw8NcqQ8Qfj23XawAoN0gmlw_VgdYjNnDIxDnzpnsfwEQbDS2T1fSWKEkRVnAn9htaXmuPzf0s6nZnAikbkwuizp0lYN3GIc0CRAKnyYIg) + +**Recommended Browser Plugins for easier reviewing** + +Your browser’s developer tools are great for reviewing this kind of work, but if you aren’t very experienced with these kinds of tools, there are a few browser extensions that can make testing certain attributes on a component easier. + +![Whatfont](https://lh5.googleusercontent.com/TfMBWiP1wFAYPdJ6_-Vk09OqQXBcBe_Z7liV968MMxXBXIShgquUdf6mqBfTUb8G8qxJt0C7_Yi6SThfUq5UysNu9i3n7lLOSyacjEEEwiLK4w6T6UBWG7Ezrt0WAcWUrz7BpL6reG13emGu1quH76jxEXRdyVWP0I6HatGv7ODLezx4MyOiVnwadw) + +- **WhatFont** - when running, this tool will allow you to click on text in your browser and obtain the color, font weight, font style, calculated font size for that text. It doesn’t work in the storybook main view, but if you open the canvas in a new tab, it works very well there. Available from [Chrome](https://chrome.google.com/webstore/detail/whatfont/jabopobgcpjmedljpbcaablpmlmfcogm?hl=en) and [Firefox](https://addons.mozilla.org/en-US/firefox/addon/zjm-whatfont/) + +![Measure-it](https://lh4.googleusercontent.com/qajZqDU9zMceS3gJstr0wAXtxC9O02A0d9JKROYBsMjANp2oEOpdl8PwJ5IWVYD0qoh5gIX2_eAH9m9F1TDSIXFyDr409TBHibJFeFqFfw6rN4b6OPa_n2G8V5QzfgemQTQqKYleFBbRohJ5avlnfOowLs_xu6pSS1Tzo18fSXXx622zWmqLyy6TyQ) + +- **Measure-it** It’s an on-screen pixel ruler. It lets you draw boxes over the browser view to measure things. Handy if you have any specific values for padding, margin, etc. that you want to make sure are right. [Chrome](https://chrome.google.com/webstore/detail/measure-it/jocbgkoackihphodedlefohapackjmna?hl=en) & [Firefox](https://addons.mozilla.org/en-US/firefox/addon/measure-it/) + +![](https://lh4.googleusercontent.com/ap2Nd_CxBJHF0b9kgeK1C-nK1B6j-PX853XIcGPP8me3Cflkx9DGIjTN0f_4DvUc5IJHzpRiqr08i3RCBr4AOYi0ItlBFrfWG2MlrvBWnDVcqa3nyV40mr-n5FadbJPyiNjXJ68n-Y1jHUvamS8GtZT9VEVSyRlh-7FiejRTyFddc7It3XxYLneUtQ) + +- **ColorZilla** lets you sample colors out of the page like you can in any design tool. [Chrome](https://www.colorzilla.com/chrome/) & [Firefox](https://www.colorzilla.com/firefox/) + +## When a PR Needs Review + +For your first few PR reviews, you may want to ask another designer to pair with you to walk through the steps. + +### Who should do the review? + +Any designer in the group well versed in USWDS and the design patterns it uses can be helpful in reviewing react-uswds PRs. One designer review on frontend PRs is what we require, but if you see someone has not caught something, or they're unavailable, feel free to step in and request changes. + +### Expected turnaround + +✨**As soon as possible**✨. Engineers may be blocked waiting for review, or time may be needed for making changes after review. If you think it might take a while, it’s helpful to communicate or leave a comment saying when you expect to look at it. If you can't take it and don't see anyone talking about it, it might be helpful to start a conversation in the slack channel to get a conversation going on how to not block work merging. + +### Reviewing the PR in Github + +Read the PR description to understand the changes made and the expected impact to the UI. You may choose to pair with the engineer to discuss and understand what you are looking for. + +Review the acceptance criteria in the Shortcut ticket, any Figma links provided in the story, or find the relevant screens in Figma to compare to. + +### Reviewing the Happo Report + +Review the Happo report to ensure any changes matched expectations, and no unexpected changes occurred. If you aren’t sure about what you’re seeing in Happo, check with the engineer requesting the review. + +Make sure to hit Accept in the dropdown in the left sidebar. Ideally, it’s a designer that is the person clicking to approve these changes in Happo. When you do this, don’t forget to go back and approve the PR itself as well. + +### Go through the design review checklist + +It’s in the description of the PR and looks like this. Use your best judgement in terms of what steps are necessary. + +![design PR checklist](https://lh4.googleusercontent.com/DT68ooHqMQ4GecIX2huNPjHZlxR5Pir1tkEJFGusPYTvZRG0sonemeF_TkipbH9rFxH41HrqksgaF3-1tLaXVGqloNl-b1TaEiLiy90kN9UjCTJewq9ZKNQudpFOE2VGxRRgitIV3FahpKoGjg-VxK_nXW1IhE97GEYoalEv4PDweTXl6KfP2t2P) + +### If everything looks good: + +1. Check that the Happo test is passing, and approve the Happo report +2. Approve the PR in Github + +### If changes are needed: + +1. Leave a comment directly on the PR in Github explaining what is needed, and/or + 1. If you want your feedback to block the branch from getting merged without you re-reviewing, make sure to select the Request Changes box when you leave the review. The engineer will have to re-request your review, and you’ll have to go back and approve their changes in order for the branch to be allowed to merge. + 2. If you aren’t sure if your feedback is in scope for the branch, or there is some other reason you want to say something without blocking the branch from moving forward, use the Comment option. +2. Reach out directly to the engineer to talk through what you are seeing. +3. Repeat the review process as changes are made until everything looks good. diff --git a/docs/image.png b/docs/image.png new file mode 100644 index 0000000000..2f300d2549 Binary files /dev/null and b/docs/image.png differ diff --git a/package.json b/package.json index 2bcc4e1d04..5e9a4d9e72 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@trussworks/react-uswds", - "version": "5.4.0", + "version": "5.5.0", "description": "React USWDS 3.0 component library", "keywords": [ "react", @@ -76,7 +76,7 @@ "@types/react-test-renderer": "^18.0.0", "@typescript-eslint/eslint-plugin": "^5.40.0", "@typescript-eslint/parser": "^5.40.0", - "@uswds/uswds": "3.6.0", + "@uswds/uswds": "3.6.1", "all-contributors-cli": "^6.24.0", "babel-jest": "^29.4.3", "babel-loader": "^9.1.2", @@ -94,7 +94,7 @@ "eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-security": "^1.4.0", "focus-trap-react": "^8.8.1", - "fork-ts-checker-webpack-plugin": "^8.0.0", + "fork-ts-checker-webpack-plugin": "^9.0.0", "happo-plugin-storybook": "^3.0.0", "happo.io": "^8.3.1", "jest": "^26.1.0", diff --git a/src/components/LanguageSelector/LanguageSelector.stories.tsx b/src/components/LanguageSelector/LanguageSelector.stories.tsx index d09ab6dd61..8f75319fab 100644 --- a/src/components/LanguageSelector/LanguageSelector.stories.tsx +++ b/src/components/LanguageSelector/LanguageSelector.stories.tsx @@ -1,10 +1,12 @@ -import React from 'react' +import React, { useState } from 'react' import { LanguageSelector, LanguageDefinition } from './LanguageSelector' +import { ComponentStory } from '@storybook/react' export default { title: 'Components/LanguageSelector', component: LanguageSelector, argTypes: { + displayLang: { control: 'string' }, small: { control: 'boolean' }, }, parameters: { @@ -19,6 +21,78 @@ Source: https://designsystem.digital.gov/components/language-selector/ }, }, } + +const TwoLanguagesTemplate: ComponentStory = ( + args +) => { + const [lang, setLang] = useState(args.displayLang) + + const languagesDisplayProp: LanguageDefinition[] = [ + { + label: '简体字', + label_local: 'Chinese - Simplified', + attr: 'zh', + on_click: () => { + setLang(`en`) + }, + }, + { + label: 'English', + attr: 'en', + on_click: () => { + setLang(`zh`) + }, + }, + ] + + return +} + +export const LanguagesDisplayPropSandbox = TwoLanguagesTemplate.bind({}) +LanguagesDisplayPropSandbox.args = { + displayLang: `en`, +} + +const MoreThanTwoLanguagesTemplate: ComponentStory = ( + args +) => { + const [lang, setLang] = useState(args.displayLang) + + const languagesDisplayProp: LanguageDefinition[] = [ + { + label: 'العربية', + label_local: 'Arabic', + attr: 'ar', + on_click: () => { + setLang(`ar`) + }, + }, + { + label: '简体字', + label_local: 'Chinese - Simplified', + attr: 'zh', + on_click: () => { + setLang(`zh`) + }, + }, + { + label: 'English', + attr: 'en', + on_click: () => { + setLang(`en`) + }, + }, + ] + + return +} + +export const LanguagesDisplayMoreThanTwoLanguagesPropSandbox = + MoreThanTwoLanguagesTemplate.bind({}) +LanguagesDisplayPropSandbox.args = { + displayLang: `en`, +} + type StorybookArguments = { small?: boolean } diff --git a/src/components/LanguageSelector/LanguageSelector.tsx b/src/components/LanguageSelector/LanguageSelector.tsx index 976c4141db..f8a01e6349 100644 --- a/src/components/LanguageSelector/LanguageSelector.tsx +++ b/src/components/LanguageSelector/LanguageSelector.tsx @@ -1,8 +1,7 @@ import React, { useState } from 'react' import classnames from 'classnames' -import { Menu } from '../header/Menu/Menu' import { LanguageSelectorButton } from './LanguageSelectorButton' -import { Button } from '../Button/Button' +import LanguageSelectorDropdown from './LanguageSelectorDropdown' export type LanguageDefinition = { label: string @@ -11,11 +10,12 @@ export type LanguageDefinition = { on_click: string | (() => void) } -type LanguageSelectorProps = { +export type LanguageSelectorProps = { label?: string langs: LanguageDefinition[] small?: boolean className?: string + displayLang?: string } export const LanguageSelector = ({ @@ -23,6 +23,7 @@ export const LanguageSelector = ({ langs, small, className, + displayLang, ...divProps }: LanguageSelectorProps & JSX.IntrinsicElements['div']): React.ReactElement => { @@ -34,66 +35,19 @@ export const LanguageSelector = ({ className ) - const [isOpen, setIsOpen] = useState(false) const [langIndex, setLangIndex] = useState(false) if (langs.length > 2) { - const items = [] - for (let i = 0; i < langs.length; i++) { - // eslint-disable-next-line security/detect-object-injection - const lang: LanguageDefinition = langs[i] - if (typeof lang.on_click === 'string') { - items.push( - - - {lang.label} - - {lang.label_local && ` (${lang.label_local})`} - - ) - } else { - items.push( - - ) - } - } - return ( -
-
    -
  • - { - setIsOpen((prevIsOpen) => !prevIsOpen) - }} - /> - -
  • -
-
- ) + const dropdownProps = { label, langs, small, displayLang } + return } else { if (label) { console.warn( "LanguageSelector's label is not used when only two languages are available." ) } - const curLang = langs[Number(langIndex)] + const curLang = + langs.find((langDef) => langDef.attr === displayLang) || + langs[Number(langIndex)] const onClickString: string = typeof curLang.on_click === 'string' ? curLang.on_click : '' const onClick = @@ -110,7 +64,7 @@ export const LanguageSelector = ({ labelAttr={curLang.attr} onToggle={() => { onClick() - setLangIndex((prevLangIndex) => !prevLangIndex) + if (!displayLang) setLangIndex((prevLangIndex) => !prevLangIndex) }} /> diff --git a/src/components/LanguageSelector/LanguageSelectorDropdown.tsx b/src/components/LanguageSelector/LanguageSelectorDropdown.tsx new file mode 100644 index 0000000000..7b22236ca7 --- /dev/null +++ b/src/components/LanguageSelector/LanguageSelectorDropdown.tsx @@ -0,0 +1,80 @@ +import React, { useState } from 'react' +import { Menu } from '../header/Menu/Menu' +import { LanguageSelectorButton } from './LanguageSelectorButton' +import classnames from 'classnames' +import { LanguageDefinition, LanguageSelectorProps } from './LanguageSelector' +import { Button } from '../Button/Button' + +const generateMenuItems = (langs: LanguageDefinition[]) => { + return langs.map((lang, index) => { + const label = ( + <> + + {lang.label} + + {lang.label_local && ` (${lang.label_local})`} + + ) + if (typeof lang.on_click === 'string') { + return ( + + {label} + + ) + } else { + return ( + + ) + } + }) +} + +const LanguageSelectorDropdown: React.FC = ({ + label, + langs, + small, + className, + displayLang, + ...divProps +}) => { + const [isOpen, setIsOpen] = useState(false) + + const classes = classnames( + 'usa-language-container', + { + [`usa-language--small`]: small !== undefined, + }, + className + ) + const displayLabel = langs.find((langDef) => langDef.attr === displayLang) + + return ( +
+
    +
  • + setIsOpen((prevIsOpen) => !prevIsOpen)} + /> + +
  • +
+
+ ) +} + +export default LanguageSelectorDropdown diff --git a/src/components/Tooltip/Tooltip.tsx b/src/components/Tooltip/Tooltip.tsx index 22649e432e..ca177a3ae0 100644 --- a/src/components/Tooltip/Tooltip.tsx +++ b/src/components/Tooltip/Tooltip.tsx @@ -12,7 +12,8 @@ import classnames from 'classnames' import { isElementInViewport, calculateMarginOffset } from './utils' type TooltipProps = { - label: string + label: ReactNode + title?: string position?: 'top' | 'bottom' | 'left' | 'right' | undefined wrapperclasses?: string className?: string @@ -202,7 +203,7 @@ export function Tooltip( }) if (isCustomProps(props)) { - const { label, asCustom, children, ...remainingProps } = props + const { label, title, asCustom, children, ...remainingProps } = props const customProps: FCProps = remainingProps as unknown as FCProps const triggerClasses = classnames('usa-tooltip__trigger', className) @@ -232,7 +233,7 @@ export function Tooltip( {triggerElement} ( ) } else { - const { label, children, ...remainingProps } = props + const { label, title, children, ...remainingProps } = props const triggerClasses = classnames( 'usa-button', @@ -273,7 +274,7 @@ export function Tooltip( ( export const dateOfBirthExample = (): React.ReactElement => (
- For example: 4 28 1986 + For example: April 28 1986 - + + + + ( ) export const checkboxFieldsetWithDefaultLegend = (): React.ReactElement => ( -
- - - - -
+ <> +

+ Required fields are marked with an asterisk ( + ). +

+
+ + + + +
+ ) export const radioFieldset = (): React.ReactElement => ( diff --git a/src/components/forms/Fieldset/Fieldset.test.tsx b/src/components/forms/Fieldset/Fieldset.test.tsx index b7aee18534..814a0edad5 100644 --- a/src/components/forms/Fieldset/Fieldset.test.tsx +++ b/src/components/forms/Fieldset/Fieldset.test.tsx @@ -26,6 +26,17 @@ describe('Fieldset component', () => { expect(myFieldset).toHaveAttribute('aria-live', 'polite') }) + it('renders with required marker', () => { + const { queryByText } = render( +
+ My Fieldset +
+ ) + const marker = queryByText('*') + expect(marker).toBeInTheDocument() + expect(marker).toHaveClass('usa-hint--required') + }) + describe('renders uswds classes', () => { it('renders legend with class usa-legend by default', () => { const { queryByTestId, getByText } = render( diff --git a/src/components/forms/Fieldset/Fieldset.tsx b/src/components/forms/Fieldset/Fieldset.tsx index a052caec17..0edb11c3b2 100644 --- a/src/components/forms/Fieldset/Fieldset.tsx +++ b/src/components/forms/Fieldset/Fieldset.tsx @@ -1,11 +1,13 @@ import React from 'react' import classnames from 'classnames' +import { RequiredMarker } from '../Label/RequiredMarker' type FieldsetProps = { children: React.ReactNode legend?: React.ReactNode legendStyle?: 'default' | 'large' | 'srOnly' className?: string + requiredMarker?: boolean } export const Fieldset = ({ @@ -13,6 +15,7 @@ export const Fieldset = ({ legend, className, legendStyle = 'default', + requiredMarker, ...fieldsetProps }: FieldsetProps & JSX.IntrinsicElements['fieldset']): React.ReactElement => { const classes = classnames('usa-fieldset', className) @@ -25,7 +28,17 @@ export const Fieldset = ({ return (
- {legend && {legend}} + {legend && ( + + {legend} + {requiredMarker && ( + <> + {' '} + + + )} + + )} {children}
) diff --git a/src/components/forms/FileInput/FileInput.stories.tsx b/src/components/forms/FileInput/FileInput.stories.tsx index 65eca5ed14..550d3d5f47 100644 --- a/src/components/forms/FileInput/FileInput.stories.tsx +++ b/src/components/forms/FileInput/FileInput.stories.tsx @@ -88,7 +88,7 @@ export const multipleFilesInput = (): React.ReactElement => ( export const withError = (): React.ReactElement => (
-