diff --git a/src/components/Button.scss b/src/components/Button.scss new file mode 100644 index 00000000..1ad22cb8 --- /dev/null +++ b/src/components/Button.scss @@ -0,0 +1,8 @@ +@import "../css/theme"; + +.excalidraw { + .excalidraw-button { + @include outlineButtonStyles; + overflow: hidden; + } +} diff --git a/src/components/Button.tsx b/src/components/Button.tsx new file mode 100644 index 00000000..3303c3eb --- /dev/null +++ b/src/components/Button.tsx @@ -0,0 +1,35 @@ +import "./Button.scss"; + +interface ButtonProps extends React.HTMLAttributes { + type?: "button" | "submit" | "reset"; + onSelect: () => any; + children: React.ReactNode; + className?: string; +} + +/** + * A generic button component that follows Excalidraw's design system. + * Style can be customised using `className` or `style` prop. + * Accepts all props that a regular `button` element accepts. + */ +export const Button = ({ + type = "button", + onSelect, + children, + className = "", + ...rest +}: ButtonProps) => { + return ( + + ); +}; diff --git a/src/components/CollabButton.scss b/src/components/CollabButton.scss index 4e09d11c..94e52d53 100644 --- a/src/components/CollabButton.scss +++ b/src/components/CollabButton.scss @@ -2,29 +2,22 @@ .excalidraw { .collab-button { - @include outlineButtonStyles; - width: var(--lg-button-size); - height: var(--lg-button-size); + --button-bg: var(--color-primary); + --button-color: white; + --button-border: var(--color-primary); + + --button-width: var(--lg-button-size); + --button-height: var(--lg-button-size); + + --button-hover-bg: var(--color-primary-darker); + --button-hover-border: var(--color-primary-darker); + + --button-active-bg: var(--color-primary-darker); - svg { - width: var(--lg-icon-size); - height: var(--lg-icon-size); - } - background-color: var(--color-primary); - border-color: var(--color-primary); - color: white; flex-shrink: 0; - &:hover { - background-color: var(--color-primary-darker); - border-color: var(--color-primary-darker); - } - - &:active { - background-color: var(--color-primary-darker); - } - - &.active { + // double .active to force specificity + &.active.active { background-color: #0fb884; border-color: #0fb884; diff --git a/src/components/CollabButton.tsx b/src/components/CollabButton.tsx index d63444a2..34521383 100644 --- a/src/components/CollabButton.tsx +++ b/src/components/CollabButton.tsx @@ -3,6 +3,7 @@ import { UsersIcon } from "./icons"; import "./CollabButton.scss"; import clsx from "clsx"; +import { Button } from "./Button"; const CollabButton = ({ isCollaborating, @@ -14,10 +15,10 @@ const CollabButton = ({ onClick: () => void; }) => { return ( - + ); }; diff --git a/src/components/dropdownMenu/DropdownMenu.scss b/src/components/dropdownMenu/DropdownMenu.scss index 28a81287..ff94f492 100644 --- a/src/components/dropdownMenu/DropdownMenu.scss +++ b/src/components/dropdownMenu/DropdownMenu.scss @@ -73,7 +73,7 @@ } &:hover { - background-color: var(--button-hover); + background-color: var(--button-hover-bg); text-decoration: none; } diff --git a/src/css/styles.scss b/src/css/styles.scss index df259bb5..42c111c3 100644 --- a/src/css/styles.scss +++ b/src/css/styles.scss @@ -408,7 +408,7 @@ pointer-events: all; &:hover { - background-color: var(--button-hover); + background-color: var(--button-hover-bg); } &:active { diff --git a/src/css/theme.scss b/src/css/theme.scss index aaa8da5b..ebf71398 100644 --- a/src/css/theme.scss +++ b/src/css/theme.scss @@ -35,7 +35,7 @@ --shadow-island: 0px 7px 14px rgba(0, 0, 0, 0.05), 0px 0px 3.12708px rgba(0, 0, 0, 0.0798), 0px 0px 0.931014px rgba(0, 0, 0, 0.1702); - --button-hover: var(--color-gray-10); + --button-hover-bg: var(--color-gray-10); --default-border-color: var(--color-gray-30); --default-button-size: 2rem; @@ -135,7 +135,7 @@ --popup-text-inverted-color: #2c2c2c; --select-highlight-color: #{$oc-blue-4}; --text-primary-color: var(--color-gray-40); - --button-hover: var(--color-gray-80); + --button-hover-bg: var(--color-gray-80); --default-border-color: var(--color-gray-80); --shadow-island: 0px 13px 33px rgba(0, 0, 0, 0.07), 0px 4.13px 9.94853px rgba(0, 0, 0, 0.0456112), diff --git a/src/css/variables.module.scss b/src/css/variables.module.scss index 5a367ffe..39bf4e29 100644 --- a/src/css/variables.module.scss +++ b/src/css/variables.module.scss @@ -39,11 +39,11 @@ .ToolIcon__icon { &:hover { - background: var(--button-hover); + background: var(--button-hover-bg); } &:active { - background: var(--button-hover); + background: var(--button-hover-bg); border: 1px solid var(--color-primary-darkest); } } @@ -54,24 +54,25 @@ justify-content: center; align-items: center; padding: 0.625rem; - width: var(--default-button-size); - height: var(--default-button-size); + width: var(--button-width, var(--default-button-size)); + height: var(--button-height, var(--default-button-size)); box-sizing: border-box; border-width: 1px; border-style: solid; - border-color: var(--default-border-color); + border-color: var(--button-border, var(--default-border-color)); border-radius: var(--border-radius-lg); cursor: pointer; - background-color: transparent; - color: var(--text-primary-color); + background-color: var(--button-bg, var(--island-bg-color)); + color: var(--button-color, var(--text-primary-color)); &:hover { - background-color: var(--button-hover); + background-color: var(--button-hover-bg); + border-color: var(--button-hover-border, var(--default-border-color)); } &:active { - background-color: var(--button-hover); - border-color: var(--color-primary-darkest); + background-color: var(--button-active-bg); + border-color: var(--button-active-border, var(--color-primary-darkest)); } &.active { @@ -83,7 +84,10 @@ } svg { - color: var(--color-primary-darker); + color: var(--button-color, var(--color-primary-darker)); + + width: var(--button-width, var(--lg-icon-size)); + height: var(--button-height, var(--lg-icon-size)); } } } diff --git a/src/packages/excalidraw/example/CustomFooter.tsx b/src/packages/excalidraw/example/CustomFooter.tsx index e6f3ff42..fbc2ea73 100644 --- a/src/packages/excalidraw/example/CustomFooter.tsx +++ b/src/packages/excalidraw/example/CustomFooter.tsx @@ -1,5 +1,7 @@ import { ExcalidrawImperativeAPI } from "../../../types"; import { MIME_TYPES } from "../entry"; +import { Button } from "../../../components/Button"; + const COMMENT_SVG = ( { return ( <> +