Add duplicate button for mobile view (#1146)

* Add a icon for dulplication

* Add PanelComponent for duplication

* Add duplicate button for mobile

* Add styles for layout action buttons

* Add a translation for 'Actions'

* Show left action buttons only for desktop

* Add duplicate button at the bottom of mobile

It is provided depending on whether or not it is `multiElement` to maintain space between buttons.
This commit is contained in:
Sanghyeon Lee 2020-04-02 00:13:53 +09:00 committed by GitHub
parent e9f80d7c31
commit 86d0da5204
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 49 additions and 7 deletions

View File

@ -1,7 +1,13 @@
import React from "react";
import { KEYS } from "../keys"; import { KEYS } from "../keys";
import { register } from "./register"; import { register } from "./register";
import { ExcalidrawElement } from "../element/types"; import { ExcalidrawElement } from "../element/types";
import { duplicateElement } from "../element"; import { duplicateElement } from "../element";
import { isSomeElementSelected } from "../scene";
import { ToolButton } from "../components/ToolButton";
import { clone } from "../components/icons";
import { t } from "../i18n";
import { getShortcutKey } from "../utils";
export const actionDuplicateSelection = register({ export const actionDuplicateSelection = register({
name: "duplicateSelection", name: "duplicateSelection",
@ -28,4 +34,16 @@ export const actionDuplicateSelection = register({
}, },
contextItemLabel: "labels.duplicateSelection", contextItemLabel: "labels.duplicateSelection",
keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.key === "d", keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.key === "d",
PanelComponent: ({ elements, appState, updateData }) => (
<ToolButton
type="button"
icon={clone}
title={`${t("labels.duplicateSelection")} ${getShortcutKey(
"CtrlOrCmd+D",
)}`}
aria-label={t("labels.duplicateSelection")}
onClick={() => updateData(null)}
visible={isSomeElementSelected(elements, appState)}
/>
),
}); });

View File

@ -8,6 +8,7 @@ import { ToolButton } from "./ToolButton";
import { capitalizeString, getShortcutKey } from "../utils"; import { capitalizeString, getShortcutKey } from "../utils";
import { CURSOR_TYPE } from "../constants"; import { CURSOR_TYPE } from "../constants";
import Stack from "./Stack"; import Stack from "./Stack";
import useIsMobile from "../is-mobile";
export function SelectedShapeActions({ export function SelectedShapeActions({
targetElements, targetElements,
@ -18,6 +19,8 @@ export function SelectedShapeActions({
renderAction: ActionManager["renderAction"]; renderAction: ActionManager["renderAction"];
elementType: ExcalidrawElement["type"]; elementType: ExcalidrawElement["type"];
}) { }) {
const isMobile = useIsMobile();
return ( return (
<div className="panelColumn"> <div className="panelColumn">
{renderAction("changeStrokeColor")} {renderAction("changeStrokeColor")}
@ -59,12 +62,15 @@ export function SelectedShapeActions({
{renderAction("bringForward")} {renderAction("bringForward")}
</div> </div>
</fieldset> </fieldset>
<fieldset> {!isMobile && (
<legend>Layers Actions</legend> <fieldset>
<div className="buttonList"> <legend>{t("labels.actions")}</legend>
{renderAction("deleteSelectedElements")} <div className="buttonList">
</div> {renderAction("duplicateSelection")}
</fieldset> {renderAction("deleteSelectedElements")}
</div>
</fieldset>
)}
</div> </div>
); );
} }

View File

@ -122,7 +122,9 @@ export function MobileMenu({
{actionManager.renderAction("toggleEditMenu")} {actionManager.renderAction("toggleEditMenu")}
{actionManager.renderAction("undo")} {actionManager.renderAction("undo")}
{actionManager.renderAction("redo")} {actionManager.renderAction("redo")}
{actionManager.renderAction("finalize")} {actionManager.renderAction(
appState.multiElement ? "finalize" : "duplicateSelection",
)}
{actionManager.renderAction("deleteSelectedElements")} {actionManager.renderAction("deleteSelectedElements")}
</div> </div>
{appState.scrolledOutside && ( {appState.scrolledOutside && (

View File

@ -210,3 +210,9 @@ export const back = createIcon(
512, 512,
{ marginLeft: "-0.2rem" }, { marginLeft: "-0.2rem" },
); );
export const clone = createIcon(
"M464 0c26.51 0 48 21.49 48 48v288c0 26.51-21.49 48-48 48H176c-26.51 0-48-21.49-48-48V48c0-26.51 21.49-48 48-48h288M176 416c-44.112 0-80-35.888-80-80V128H48c-26.51 0-48 21.49-48 48v288c0 26.51 21.49 48 48 48h288c26.51 0 48-21.49 48-48v-48H176z",
512,
512,
);

View File

@ -42,6 +42,7 @@
"canvasBackground": "Canvas background", "canvasBackground": "Canvas background",
"drawingCanvas": "Drawing Canvas", "drawingCanvas": "Drawing Canvas",
"layers": "Layers", "layers": "Layers",
"actions": "Actions",
"language": "Language", "language": "Language",
"createRoom": "Share a live-collaboration session", "createRoom": "Share a live-collaboration session",
"duplicateSelection": "Duplicate selected elements" "duplicateSelection": "Duplicate selected elements"

View File

@ -92,6 +92,15 @@ canvas {
position: absolute; position: absolute;
pointer-events: none; pointer-events: none;
} }
.ToolIcon {
margin: 0 5px;
}
.ToolIcon__icon {
width: 28px;
height: 28px;
}
} }
fieldset { fieldset {