mirror of
https://github.com/penpot/penpot.git
synced 2025-04-04 11:01:20 -05:00
🐛 Pass color format to css formatter (#5890)
* 🐛 Pass color format to css formatter * 📎 Use get method to retrieve format * 📎 Add UI test to check background color is correctly copied to clipboard when changing the format
This commit is contained in:
parent
44acd79081
commit
f3040fc10d
9 changed files with 84 additions and 19 deletions
|
@ -34,6 +34,8 @@ export default defineConfig({
|
|||
trace: "on-first-retry",
|
||||
|
||||
locale: "en-US",
|
||||
|
||||
permissions: ["clipboard-write", "clipboard-read"],
|
||||
},
|
||||
|
||||
/* Configure projects for major browsers */
|
||||
|
|
|
@ -32,3 +32,41 @@ test("Bug 9042 - Measurement unit dropdowns for columns are cut off in grid layo
|
|||
await rowsContainer.getByText("FR").nth(2).click();
|
||||
await expect(rowsContainer.getByText("%")).toBeInViewport();
|
||||
});
|
||||
|
||||
test("[Taiga #9116] Copy CSS background color in the selected format in the INSPECT tab", async ({
|
||||
page,
|
||||
context,
|
||||
}) => {
|
||||
const workspacePage = new WorkspacePage(page);
|
||||
await workspacePage.setupEmptyFile(page);
|
||||
await workspacePage.goToWorkspace();
|
||||
|
||||
await workspacePage.rectShapeButton.click();
|
||||
await workspacePage.clickWithDragViewportAt(128, 128, 200, 100);
|
||||
await workspacePage.clickLeafLayer("Rectangle");
|
||||
|
||||
const inspectButton = workspacePage.page.getByRole("tab", {
|
||||
name: "Inspect",
|
||||
});
|
||||
await inspectButton.click();
|
||||
|
||||
const colorDropdown = workspacePage.page
|
||||
.getByRole("combobox")
|
||||
.getByText("HEX");
|
||||
await colorDropdown.click();
|
||||
|
||||
const rgbaFormatButton = workspacePage.page.getByRole("option", {
|
||||
name: "RGBA",
|
||||
});
|
||||
await rgbaFormatButton.click();
|
||||
|
||||
const copyColorButton = workspacePage.page.getByRole("button", {
|
||||
name: "Copy color",
|
||||
});
|
||||
await copyColorButton.click();
|
||||
|
||||
const rgbaColorText = await page.evaluate(() =>
|
||||
navigator.clipboard.readText(),
|
||||
);
|
||||
expect(rgbaColorText).toContain("background: rgba(");
|
||||
});
|
||||
|
|
|
@ -11,13 +11,14 @@
|
|||
[app.main.data.event :as-alias ev]
|
||||
[app.main.ui.icons :as i]
|
||||
[app.util.dom :as dom]
|
||||
[app.util.i18n :refer [tr]]
|
||||
[app.util.timers :as tm]
|
||||
[app.util.webapi :as wapi]
|
||||
[rumext.v2 :as mf]))
|
||||
|
||||
(mf/defc copy-button
|
||||
{::mf/props :obj}
|
||||
[{:keys [data on-copied children class]}]
|
||||
[{:keys [data on-copied children class aria-label]}]
|
||||
(let [active* (mf/use-state false)
|
||||
active? (deref active*)
|
||||
|
||||
|
@ -38,6 +39,7 @@
|
|||
(if (fn? data) (data) data)))))]
|
||||
|
||||
[:button {:class class
|
||||
:aria-label (or aria-label (tr "labels.copy"))
|
||||
:data-active (dm/str active?)
|
||||
:on-click on-click}
|
||||
children
|
||||
|
|
|
@ -102,6 +102,8 @@
|
|||
current-icon (:icon selected-option)
|
||||
current-icon-ref (i/key->icon current-icon)]
|
||||
[:div {:on-click open-dropdown
|
||||
:role "combobox"
|
||||
:aria-activedescendent current-value
|
||||
:class (dm/str (stl/css-case :custom-select true
|
||||
:disabled disabled
|
||||
:icon (some? current-icon-ref))
|
||||
|
@ -116,11 +118,13 @@
|
|||
(for [[index item] (d/enumerate options)]
|
||||
(if (= :separator item)
|
||||
[:li {:class (dom/classnames (stl/css :separator) true)
|
||||
:role "option"
|
||||
:key (dm/str current-id "-" index)}]
|
||||
(let [[value label icon] (as-key-value item)
|
||||
icon-ref (i/key->icon icon)]
|
||||
[:li
|
||||
{:key (dm/str current-id "-" index)
|
||||
:role "option"
|
||||
:class (stl/css-case
|
||||
:checked-element true
|
||||
:disabled (:disabled item)
|
||||
|
|
|
@ -117,6 +117,7 @@
|
|||
[:div {:class (stl/css :format-info)} "rgba"])]
|
||||
|
||||
[:& copy-button {:data copy-data
|
||||
:aria-label (tr "labels.copy-color")
|
||||
:class (stl/css-case :color-row-copy-btn true
|
||||
:one-line (not color-library-name)
|
||||
:two-line (some? color-library-name))}
|
||||
|
|
|
@ -45,7 +45,7 @@
|
|||
{:color color
|
||||
:format format
|
||||
:on-change-format on-change
|
||||
:copy-data (css/get-shape-properties-css objects {:fills [shape]} properties)}]]))
|
||||
:copy-data (css/get-shape-properties-css objects {:fills [shape]} properties {:format format})}]]))
|
||||
|
||||
(mf/defc fill-panel
|
||||
{::mf/wrap-props false}
|
||||
|
|
|
@ -53,28 +53,30 @@
|
|||
:else value))
|
||||
|
||||
(defn format-color
|
||||
[value _options]
|
||||
(cond
|
||||
(:image value)
|
||||
(let [image-url (cfg/resolve-file-media (:image value))
|
||||
opacity-color (when (not= (:opacity value) 1)
|
||||
(uc/gradient->css {:type :linear
|
||||
:stops [{:color "#FFFFFF" :opacity (:opacity value)}
|
||||
{:color "#FFFFFF" :opacity (:opacity value)}]}))]
|
||||
(if opacity-color
|
||||
;; CSS doesn't allow setting directly opacity to background image, we should add a dummy gradient to get it
|
||||
(dm/fmt "%, url(%) no-repeat center center / cover" opacity-color image-url)
|
||||
(dm/fmt "url(%) no-repeat center center / cover" image-url)))
|
||||
[value options]
|
||||
(let [format (get options :format :hex)]
|
||||
(cond
|
||||
(:image value)
|
||||
(let [image-url (cfg/resolve-file-media (:image value))
|
||||
opacity-color (when (not= (:opacity value) 1)
|
||||
(uc/gradient->css {:type :linear
|
||||
:stops [{:color "#FFFFFF" :opacity (:opacity value)}
|
||||
{:color "#FFFFFF" :opacity (:opacity value)}]}))]
|
||||
(if opacity-color
|
||||
;; CSS doesn't allow setting directly opacity to background image, we should add a dummy gradient to get it
|
||||
(dm/fmt "%, url(%) no-repeat center center / cover" opacity-color image-url)
|
||||
(dm/fmt "url(%) no-repeat center center / cover" image-url)))
|
||||
|
||||
(not= (:opacity value) 1)
|
||||
(uc/color->background value)
|
||||
(not= (:opacity value) 1)
|
||||
(uc/color->format->background value format)
|
||||
|
||||
:else
|
||||
(str/upper (:color value))))
|
||||
:else
|
||||
(uc/color->format->background value format))))
|
||||
|
||||
(defmethod format-value :color
|
||||
[_ value options]
|
||||
(format-color value options))
|
||||
(let [format (get options :format :hex)]
|
||||
(format-color value (assoc options :format format))))
|
||||
|
||||
(defmethod format-value :color-array
|
||||
[_ value options]
|
||||
|
|
|
@ -1807,6 +1807,14 @@ msgstr "You can continue with a Penpot account"
|
|||
msgid "labels.copy-invitation-link"
|
||||
msgstr "Copy link"
|
||||
|
||||
#: src/app/main/ui/components/copy_button.cljs:42
|
||||
msgid "labels.copy"
|
||||
msgstr "Copy"
|
||||
|
||||
#: src/app/main/ui/inspect/attributes/common.cljs:120
|
||||
msgid "labels.copy-color"
|
||||
msgstr "Copy color"
|
||||
|
||||
#: src/app/main/ui/static.cljs:62
|
||||
msgid "labels.copyright"
|
||||
msgstr "Kaleidos @2024"
|
||||
|
|
|
@ -1817,6 +1817,14 @@ msgstr "Puedes continuar con una cuenta de Penpot"
|
|||
msgid "labels.copy-invitation-link"
|
||||
msgstr "Copiar enlace"
|
||||
|
||||
#: src/app/main/ui/components/copy_button.cljs:42
|
||||
msgid "labels.copy"
|
||||
msgstr "Copiar"
|
||||
|
||||
#: src/app/main/ui/inspect/attributes/common.cljs:120
|
||||
msgid "labels.copy-color"
|
||||
msgstr "Copiar color"
|
||||
|
||||
#: src/app/main/ui/static.cljs:62
|
||||
msgid "labels.copyright"
|
||||
msgstr "Kaleidos @2024"
|
||||
|
|
Loading…
Add table
Reference in a new issue