diff --git a/frontend/src/metabase/components/Icon.jsx b/frontend/src/metabase/components/Icon.jsx index a90a88ec8a1a590383116f7ad85ed55dfb889c65..d47c6d8d899bd4f4b17dd6c0bd005ddd0b7c258b 100644 --- a/frontend/src/metabase/components/Icon.jsx +++ b/frontend/src/metabase/components/Icon.jsx @@ -84,12 +84,11 @@ class BaseIcon extends Component { delete props.size, props.scale; if (icon.img) { + // avoid passing `role="img"` to an actual image file + // eslint-disable-next-line no-unused-vars + const { role, ...rest } = props; return ( - <RetinaImage - forceOriginalDimensions={false} - src={icon.img} - {...props} - /> + <RetinaImage forceOriginalDimensions={false} src={icon.img} {...rest} /> ); } else if (icon.svg) { return <svg {...props} dangerouslySetInnerHTML={{ __html: icon.svg }} />; diff --git a/frontend/src/metabase/icon_paths.js b/frontend/src/metabase/icon_paths.js index c05887db5c497b0ffa9ab81c9a128d517212c178..286719ee4b7f9174e8fd83230891f77a14016c1e 100644 --- a/frontend/src/metabase/icon_paths.js +++ b/frontend/src/metabase/icon_paths.js @@ -414,7 +414,7 @@ ICON_PATHS["scalar"] = ICON_PATHS["number"]; export function parseViewBox(viewBox: string): Array<number> { // a viewBox is a string that takes the form 'min-x, min-y, width, height' // grab the values and return just width and height since we currently don't - // tend to card about min-x or min-y + // tend to care about min-x or min-y // we cast to numbers so we can do math-y stuff with the width and height return viewBox @@ -440,6 +440,8 @@ export function loadIcon(name: string) { width: 16, height: 16, fill: "currentcolor", + role: "img", + "aria-label": name + " icon", }, svg: undefined, path: undefined, diff --git a/frontend/test/metabase/components/Icon.unit.spec.js b/frontend/test/metabase/components/Icon.unit.spec.js index 73efead971e158bf0ef3c91bd8ade49ad27b5fb0..d4f2ccc1cb96d466459f76a4061640736ec5d49a 100644 --- a/frontend/test/metabase/components/Icon.unit.spec.js +++ b/frontend/test/metabase/components/Icon.unit.spec.js @@ -1,11 +1,14 @@ +import React from "react"; +import "@testing-library/jest-dom/extend-expect"; +import { render, screen } from "@testing-library/react"; + import { ICON_PATHS, loadIcon, parseViewBox } from "metabase/icon_paths"; +import Icon from "metabase/components/Icon"; // find the first icon with a non standard viewBox -const NON_STANDARD_VIEWBOX_ICON = Object.keys(ICON_PATHS).filter(key => { - if (ICON_PATHS[key].attrs && ICON_PATHS[key].attrs.viewBox) { - return ICON_PATHS[key]; - } -})[0]; +const NON_STANDARD_VIEWBOX_ICON = Object.keys(ICON_PATHS).find(key => { + return ICON_PATHS[key].attrs && ICON_PATHS[key].attrs.viewBox; +}); describe("Icon", () => { describe("parseViewBox", () => { @@ -27,4 +30,33 @@ describe("Icon", () => { expect(height).toEqual(parsedHeight / 2); }); }); + + describe("component", () => { + it("should render correct icon for the valid `name`", () => { + // used "google" to cover the use of `dangerouslySetInnerHTML` + render(<Icon name="google" />); + const icon = screen.getByRole("img", { name: /google icon/i }); + + expect(icon).toHaveClass("Icon-google"); + }); + + it("should render `unknown` icon given non-existing `name`", () => { + render(<Icon name="404-icon-not-found" />); + const icon = screen.getByRole("img", { name: /unknown icon/i }); + + expect(icon).toHaveClass("Icon-unknown"); + }); + + it("should render `unknown` icon when `name` is not provided", () => { + render(<Icon />); + const icon = screen.getByRole("img", { name: /unknown icon/i }); + + expect(icon).toHaveClass("Icon-unknown"); + }); + + it("should render an image if `img` attribute is present in `ICON_PATHS`", () => { + render(<Icon name="slack" />); + expect(screen.getByRole("img")).toHaveAttribute("src"); + }); + }); }); diff --git a/frontend/test/metabase/components/__snapshots__/Button.unit.spec.js.snap b/frontend/test/metabase/components/__snapshots__/Button.unit.spec.js.snap index 44644853788cc6d83bacc27cbc95913ceac9b16b..dd8275762712255a0990637035250bb0bbdbacf7 100644 --- a/frontend/test/metabase/components/__snapshots__/Button.unit.spec.js.snap +++ b/frontend/test/metabase/components/__snapshots__/Button.unit.spec.js.snap @@ -26,9 +26,11 @@ exports[`Button should render correctly with an icon 1`] = ` style={null} > <svg + aria-label="star icon" className="Icon Icon-star Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={14} + role="img" viewBox="0 0 32 32" width={14} > diff --git a/frontend/test/metabase/components/__snapshots__/Calendar.unit.spec.js.snap b/frontend/test/metabase/components/__snapshots__/Calendar.unit.spec.js.snap index 103ec83247eaca9d1217c6616ed7bfcc03946411..879480652096ed7c7461d2530535a160214bf474 100644 --- a/frontend/test/metabase/components/__snapshots__/Calendar.unit.spec.js.snap +++ b/frontend/test/metabase/components/__snapshots__/Calendar.unit.spec.js.snap @@ -12,9 +12,11 @@ exports[`Calendar should render correctly 1`] = ` onClick={[Function]} > <svg + aria-label="chevronleft icon" className="Icon Icon-chevronleft Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={10} + role="img" viewBox="0 0 32 32" width={10} > @@ -37,9 +39,11 @@ exports[`Calendar should render correctly 1`] = ` onClick={[Function]} > <svg + aria-label="chevronright icon" className="Icon Icon-chevronright Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={10} + role="img" viewBox="0 0 32 32" width={10} > diff --git a/frontend/test/metabase/internal/__snapshots__/components.unit.spec.js.snap b/frontend/test/metabase/internal/__snapshots__/components.unit.spec.js.snap index 0fff2fd57b0c326f6650031040b3be6384fb8d6b..032cb8593c16cf1a437b189047613bacd0423488 100644 --- a/frontend/test/metabase/internal/__snapshots__/components.unit.spec.js.snap +++ b/frontend/test/metabase/internal/__snapshots__/components.unit.spec.js.snap @@ -43,9 +43,11 @@ exports[`Button should render "with an icon" correctly 1`] = ` style={null} > <svg + aria-label="star icon" className="Icon Icon-star Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={14} + role="img" viewBox="0 0 32 32" width={14} > @@ -259,9 +261,11 @@ exports[`CheckBox should render "Green" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -291,9 +295,11 @@ exports[`CheckBox should render "On - Default blue" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -323,9 +329,11 @@ exports[`CheckBox should render "Purple" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -355,9 +363,11 @@ exports[`CheckBox should render "Red" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -387,9 +397,11 @@ exports[`CheckBox should render "Yellow" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -420,9 +432,11 @@ exports[`EntityMenu should render "Edit menu" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="pencil icon" className="Icon Icon-pencil Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" height={18} + role="img" viewBox="0 0 32 32" width={18} > @@ -454,9 +468,11 @@ exports[`EntityMenu should render "More menu" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="burger icon" className="Icon Icon-burger Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" height={18} + role="img" viewBox="0 0 32 32" width={18} > @@ -488,9 +504,11 @@ exports[`EntityMenu should render "Multiple menus" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="pencil icon" className="Icon Icon-pencil Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" height={18} + role="img" viewBox="0 0 32 32" width={18} > @@ -511,10 +529,12 @@ exports[`EntityMenu should render "Multiple menus" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="share icon" className="Icon Icon-share Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" fillRule="evenodd" height={18} + role="img" viewBox="0 0 32 41" width={18} > @@ -535,9 +555,11 @@ exports[`EntityMenu should render "Multiple menus" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="burger icon" className="Icon Icon-burger Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" height={18} + role="img" viewBox="0 0 32 32" width={18} > @@ -569,10 +591,12 @@ exports[`EntityMenu should render "Share menu" correctly 1`] = ` onClick={[Function]} > <svg + aria-label="share icon" className="Icon Icon-share Icon-sc-1x67v3y-1 bcaYRA" fill="currentcolor" fillRule="evenodd" height={18} + role="img" viewBox="0 0 32 41" width={18} > @@ -598,10 +622,12 @@ exports[`ModalContent should render "default" correctly 1`] = ` id={undefined} > <svg + aria-label="close icon" className="Icon Icon-close text-light text-medium-hover cursor-pointer absolute z2 m2 p2 top right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={16} onClick={[Function]} + role="img" viewBox="0 0 32 32" width={16} > @@ -987,9 +1013,11 @@ exports[`Select should render "default" correctly 1`] = ` </span> </span> <svg + aria-label="chevrondown icon" className="Icon Icon-chevrondown AdminSelect-chevron flex-align-right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" viewBox="0 0 32 32" width={12} > @@ -1028,9 +1056,11 @@ exports[`Select should render "kitchen_sink" correctly 1`] = ` </span> </span> <svg + aria-label="chevrondown icon" className="Icon Icon-chevrondown AdminSelect-chevron flex-align-right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" viewBox="0 0 32 32" width={12} > @@ -1069,9 +1099,11 @@ exports[`Select should render "multiple" correctly 1`] = ` </span> </span> <svg + aria-label="chevrondown icon" className="Icon Icon-chevrondown AdminSelect-chevron flex-align-right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" viewBox="0 0 32 32" width={12} > @@ -1106,9 +1138,11 @@ exports[`Select should render "search" correctly 1`] = ` </span> </span> <svg + aria-label="chevrondown icon" className="Icon Icon-chevrondown AdminSelect-chevron flex-align-right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" viewBox="0 0 32 32" width={12} > @@ -1143,9 +1177,11 @@ exports[`Select should render "sections" correctly 1`] = ` </span> </span> <svg + aria-label="chevrondown icon" className="Icon Icon-chevrondown AdminSelect-chevron flex-align-right Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" viewBox="0 0 32 32" width={12} > @@ -1182,9 +1218,11 @@ exports[`StackedCheckBox should render "Checked with color" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white", @@ -1238,9 +1276,11 @@ exports[`StackedCheckBox should render "Checked" correctly 1`] = ` } > <svg + aria-label="check icon" className="Icon Icon-check Icon-sc-1x67v3y-1 hFNRPZ" fill="currentcolor" height={12} + role="img" style={ Object { "color": "white",