mirror of
https://github.com/penpot/penpot.git
synced 2025-01-22 14:39:45 -05:00
Merge pull request #312 from tokens-studio/import-sd-2
✨ Import: Verify data with StyleDictionary
This commit is contained in:
commit
bef648a63f
10 changed files with 362 additions and 146 deletions
|
@ -36,7 +36,6 @@
|
||||||
(def token-type->dtcg-token-type
|
(def token-type->dtcg-token-type
|
||||||
{:boolean "boolean"
|
{:boolean "boolean"
|
||||||
:border-radius "borderRadius"
|
:border-radius "borderRadius"
|
||||||
:box-shadow "boxShadow"
|
|
||||||
:color "color"
|
:color "color"
|
||||||
:dimensions "dimension"
|
:dimensions "dimension"
|
||||||
:numeric "numeric"
|
:numeric "numeric"
|
||||||
|
|
|
@ -197,6 +197,21 @@
|
||||||
(assoc-in acc path (update-token-fn token))))
|
(assoc-in acc path (update-token-fn token))))
|
||||||
{} tokens))
|
{} tokens))
|
||||||
|
|
||||||
|
(defn backtrace-tokens-tree
|
||||||
|
"Convert tokens into a nested tree with their `:name` as the path.
|
||||||
|
Generates a uuid per token to backtrace a token from an external source (StyleDictionary).
|
||||||
|
The backtrace can't be the name as the name might not exist when the user is creating a token."
|
||||||
|
[tokens]
|
||||||
|
(reduce
|
||||||
|
(fn [acc [_ token]]
|
||||||
|
(let [temp-id (random-uuid)
|
||||||
|
token (assoc token :temp/id temp-id)
|
||||||
|
path (split-token-path (:name token))]
|
||||||
|
(-> acc
|
||||||
|
(assoc-in (concat [:tokens-tree] path) token)
|
||||||
|
(assoc-in [:ids temp-id] token))))
|
||||||
|
{:tokens-tree {} :ids {}} tokens))
|
||||||
|
|
||||||
(defprotocol ITokenSet
|
(defprotocol ITokenSet
|
||||||
(add-token [_ token] "add a token at the end of the list")
|
(add-token [_ token] "add a token at the end of the list")
|
||||||
(update-token [_ token-name f] "update a token in the list")
|
(update-token [_ token-name f] "update a token in the list")
|
||||||
|
@ -508,6 +523,7 @@ When `before-set-name` is nil, move set to bottom")
|
||||||
(update-set-name [_ old-set-name new-set-name] "updates set name in themes")
|
(update-set-name [_ old-set-name new-set-name] "updates set name in themes")
|
||||||
(encode-dtcg [_] "Encodes library to a dtcg compatible json string")
|
(encode-dtcg [_] "Encodes library to a dtcg compatible json string")
|
||||||
(decode-dtcg-json [_ parsed-json] "Decodes parsed json containing tokens and converts to library")
|
(decode-dtcg-json [_ parsed-json] "Decodes parsed json containing tokens and converts to library")
|
||||||
|
(get-all-tokens [_] "all tokens in the lib")
|
||||||
(validate [_]))
|
(validate [_]))
|
||||||
|
|
||||||
(deftype TokensLib [sets set-groups themes active-themes]
|
(deftype TokensLib [sets set-groups themes active-themes]
|
||||||
|
@ -800,6 +816,12 @@ When `before-set-name` is nil, move set to bottom")
|
||||||
themes
|
themes
|
||||||
active-themes)))
|
active-themes)))
|
||||||
|
|
||||||
|
(get-all-tokens [this]
|
||||||
|
(reduce
|
||||||
|
(fn [tokens' set]
|
||||||
|
(into tokens' (map (fn [x] [(:name x) x]) (get-tokens set))))
|
||||||
|
{} (get-sets this)))
|
||||||
|
|
||||||
(validate [_]
|
(validate [_]
|
||||||
(and (valid-token-sets? sets) ;; TODO: validate set-groups
|
(and (valid-token-sets? sets) ;; TODO: validate set-groups
|
||||||
(valid-token-themes? themes)
|
(valid-token-themes? themes)
|
||||||
|
|
|
@ -177,6 +177,12 @@
|
||||||
|
|
||||||
:devtools {:http-port 3460}
|
:devtools {:http-port 3460}
|
||||||
|
|
||||||
|
:js-options
|
||||||
|
{:entry-keys ["module" "browser" "main"]
|
||||||
|
:resolve {"penpot/vendor/text-editor-v2"
|
||||||
|
{:target :file
|
||||||
|
:file "vendor/text_editor_v2.js"}}}
|
||||||
|
|
||||||
:compiler-options
|
:compiler-options
|
||||||
{:output-feature-set :es2020
|
{:output-feature-set :es2020
|
||||||
:output-wrapper false
|
:output-wrapper false
|
||||||
|
|
|
@ -3,7 +3,23 @@
|
||||||
[cuerdas.core :as str]))
|
[cuerdas.core :as str]))
|
||||||
|
|
||||||
(def error-codes
|
(def error-codes
|
||||||
{:error.token/direct-self-reference
|
{:error.import/json-parse-error
|
||||||
|
{:error/code :error.import/json-parse-error
|
||||||
|
:error/message "Import Error: Could not parse json"}
|
||||||
|
|
||||||
|
:error.import/invalid-json-data
|
||||||
|
{:error/code :error.import/invalid-json-data
|
||||||
|
:error/message "Import Error: Invalid token data in json."}
|
||||||
|
|
||||||
|
:error.import/style-dictionary-reference-errors
|
||||||
|
{:error/code :error.import/style-dictionary-reference-errors
|
||||||
|
:error/fn #(str "Import Error:\n\n" (str/join "\n\n" %))}
|
||||||
|
|
||||||
|
:error.import/style-dictionary-unknown-error
|
||||||
|
{:error/code :error.import/style-dictionary-reference-errors
|
||||||
|
:error/message "Import Error:"}
|
||||||
|
|
||||||
|
:error.token/direct-self-reference
|
||||||
{:error/code :error.token/direct-self-reference
|
{:error/code :error.token/direct-self-reference
|
||||||
:error/message "Token has self reference"}
|
:error/message "Token has self reference"}
|
||||||
|
|
||||||
|
@ -30,6 +46,11 @@
|
||||||
(-> (get-error-code error-key)
|
(-> (get-error-code error-key)
|
||||||
(assoc :error/value error-value)))
|
(assoc :error/value error-value)))
|
||||||
|
|
||||||
|
(defn error-ex-info [error-key error-value exception]
|
||||||
|
(let [err (-> (error-with-value error-key error-value)
|
||||||
|
(assoc :error/exception exception))]
|
||||||
|
(ex-info (:error/code err) err)))
|
||||||
|
|
||||||
(defn has-error-code? [error-key errors]
|
(defn has-error-code? [error-key errors]
|
||||||
(some #(= (:error/code %) error-key) errors))
|
(some #(= (:error/code %) error-key) errors))
|
||||||
|
|
||||||
|
|
|
@ -115,7 +115,7 @@ Token names should only contain letters and digits separated by . characters.")}
|
||||||
(-> (update tokens token-name merge {:value value
|
(-> (update tokens token-name merge {:value value
|
||||||
:name token-name
|
:name token-name
|
||||||
:type (:type token)})
|
:type (:type token)})
|
||||||
(sd/resolve-tokens+ {:names-map? true})
|
(sd/resolve-tokens+)
|
||||||
(p/then
|
(p/then
|
||||||
(fn [resolved-tokens]
|
(fn [resolved-tokens]
|
||||||
(let [{:keys [errors resolved-value] :as resolved-token} (get resolved-tokens token-name)]
|
(let [{:keys [errors resolved-value] :as resolved-token} (get resolved-tokens token-name)]
|
||||||
|
@ -205,9 +205,8 @@ Token names should only contain letters and digits separated by . characters.")}
|
||||||
color? (wtt/color-token? token)
|
color? (wtt/color-token? token)
|
||||||
selected-set-tokens (mf/deref refs/workspace-selected-token-set-tokens)
|
selected-set-tokens (mf/deref refs/workspace-selected-token-set-tokens)
|
||||||
active-theme-tokens (mf/deref refs/workspace-active-theme-sets-tokens)
|
active-theme-tokens (mf/deref refs/workspace-active-theme-sets-tokens)
|
||||||
resolved-tokens (sd/use-resolved-tokens active-theme-tokens
|
resolved-tokens (sd/use-resolved-tokens active-theme-tokens {:cache-atom form-token-cache-atom
|
||||||
{:names-map? true
|
:interactive? true})
|
||||||
:cache-atom form-token-cache-atom})
|
|
||||||
token-path (mf/use-memo
|
token-path (mf/use-memo
|
||||||
(mf/deps (:name token))
|
(mf/deps (:name token))
|
||||||
#(wtt/token-name->path (:name token)))
|
#(wtt/token-name->path (:name token)))
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
[app.main.ui.workspace.sidebar.assets.common :as cmm]
|
[app.main.ui.workspace.sidebar.assets.common :as cmm]
|
||||||
[app.main.ui.workspace.tokens.changes :as wtch]
|
[app.main.ui.workspace.tokens.changes :as wtch]
|
||||||
[app.main.ui.workspace.tokens.context-menu :refer [token-context-menu]]
|
[app.main.ui.workspace.tokens.context-menu :refer [token-context-menu]]
|
||||||
|
[app.main.ui.workspace.tokens.errors :as wte]
|
||||||
[app.main.ui.workspace.tokens.sets :refer [sets-list]]
|
[app.main.ui.workspace.tokens.sets :refer [sets-list]]
|
||||||
[app.main.ui.workspace.tokens.sets-context :as sets-context]
|
[app.main.ui.workspace.tokens.sets-context :as sets-context]
|
||||||
[app.main.ui.workspace.tokens.sets-context-menu :refer [sets-context-menu]]
|
[app.main.ui.workspace.tokens.sets-context-menu :refer [sets-context-menu]]
|
||||||
|
@ -39,6 +40,7 @@
|
||||||
[beicon.v2.core :as rx]
|
[beicon.v2.core :as rx]
|
||||||
[cuerdas.core :as str]
|
[cuerdas.core :as str]
|
||||||
[okulary.core :as l]
|
[okulary.core :as l]
|
||||||
|
[promesa.core :as p]
|
||||||
[rumext.v2 :as mf]
|
[rumext.v2 :as mf]
|
||||||
[shadow.resource]))
|
[shadow.resource]))
|
||||||
|
|
||||||
|
@ -281,32 +283,15 @@
|
||||||
(fn [event]
|
(fn [event]
|
||||||
(let [file (-> event .-target .-files (aget 0))]
|
(let [file (-> event .-target .-files (aget 0))]
|
||||||
(->> (wapi/read-file-as-text file)
|
(->> (wapi/read-file-as-text file)
|
||||||
(rx/map (fn [data]
|
(sd/process-json-stream)
|
||||||
(try
|
|
||||||
(t/decode-str data)
|
|
||||||
(catch js/Error e
|
|
||||||
(throw (ex-info "Json parse error"
|
|
||||||
{:user-error "Import Error: Could not parse json"
|
|
||||||
:type :json-parse-error
|
|
||||||
:data data
|
|
||||||
:exception e}))))))
|
|
||||||
(rx/map (fn [json-data]
|
|
||||||
(try
|
|
||||||
(ctob/decode-dtcg-json (ctob/ensure-tokens-lib nil) json-data)
|
|
||||||
(catch js/Error e
|
|
||||||
(throw (ex-info "invalid token data"
|
|
||||||
{:user-error "Import Error: Invalid token data in json."
|
|
||||||
:type :invalid-token-data
|
|
||||||
:data json-data
|
|
||||||
:exception e}))))))
|
|
||||||
(rx/subs! (fn [lib]
|
(rx/subs! (fn [lib]
|
||||||
(st/emit! (dt/import-tokens-lib lib)))
|
(st/emit! (dt/import-tokens-lib lib)))
|
||||||
(fn [err]
|
(fn [err]
|
||||||
(let [{:keys [user-error]} (ex-data err)]
|
(js/console.error err)
|
||||||
(st/emit! (ntf/show {:content user-error
|
(st/emit! (ntf/show {:content (wte/humanize-errors [(ex-data err)])
|
||||||
:notification-type :toast
|
:type :toast
|
||||||
:type :warning
|
:level :warning
|
||||||
:timeout 3000}))))))
|
:timeout 9000})))))
|
||||||
(set! (.-value (mf/ref-val input-ref)) "")))
|
(set! (.-value (mf/ref-val input-ref)) "")))
|
||||||
on-export (fn []
|
on-export (fn []
|
||||||
(let [tokens-blob (some-> (deref refs/tokens-lib)
|
(let [tokens-blob (some-> (deref refs/tokens-lib)
|
||||||
|
|
|
@ -3,20 +3,25 @@
|
||||||
["@tokens-studio/sd-transforms" :as sd-transforms]
|
["@tokens-studio/sd-transforms" :as sd-transforms]
|
||||||
["style-dictionary$default" :as sd]
|
["style-dictionary$default" :as sd]
|
||||||
[app.common.logging :as l]
|
[app.common.logging :as l]
|
||||||
|
[app.common.transit :as t]
|
||||||
[app.common.types.tokens-lib :as ctob]
|
[app.common.types.tokens-lib :as ctob]
|
||||||
[app.main.refs :as refs]
|
[app.main.refs :as refs]
|
||||||
[app.main.ui.workspace.tokens.errors :as wte]
|
[app.main.ui.workspace.tokens.errors :as wte]
|
||||||
[app.main.ui.workspace.tokens.tinycolor :as tinycolor]
|
[app.main.ui.workspace.tokens.tinycolor :as tinycolor]
|
||||||
[app.main.ui.workspace.tokens.token :as wtt]
|
[app.main.ui.workspace.tokens.token :as wtt]
|
||||||
|
[beicon.v2.core :as rx]
|
||||||
[cuerdas.core :as str]
|
[cuerdas.core :as str]
|
||||||
[promesa.core :as p]
|
[promesa.core :as p]
|
||||||
[rumext.v2 :as mf]))
|
[rumext.v2 :as mf]
|
||||||
|
[app.common.data :as d]))
|
||||||
|
|
||||||
(l/set-level! "app.main.ui.workspace.tokens.style-dictionary" :warn)
|
(l/set-level! "app.main.ui.workspace.tokens.style-dictionary" :warn)
|
||||||
|
|
||||||
(def StyleDictionary
|
;; === Style Dictionary
|
||||||
"Initiates the global StyleDictionary instance with transforms
|
|
||||||
from tokens-studio used to parse and resolved token values."
|
(def setup-style-dictionary
|
||||||
|
"Initiates the StyleDictionary instance.
|
||||||
|
Setup transforms from tokens-studio used to parse and resolved token values."
|
||||||
(do
|
(do
|
||||||
(sd-transforms/registerTransforms sd)
|
(sd-transforms/registerTransforms sd)
|
||||||
(.registerFormat sd #js {:name "custom/json"
|
(.registerFormat sd #js {:name "custom/json"
|
||||||
|
@ -24,44 +29,182 @@
|
||||||
(.-tokens (.-dictionary res)))})
|
(.-tokens (.-dictionary res)))})
|
||||||
sd))
|
sd))
|
||||||
|
|
||||||
;; Functions -------------------------------------------------------------------
|
(def default-config
|
||||||
|
{:platforms {:json
|
||||||
|
{:transformGroup "tokens-studio"
|
||||||
|
;; Required: The StyleDictionary API is focused on files even when working in the browser
|
||||||
|
:files [{:format "custom/json" :destination "penpot"}]}}
|
||||||
|
:preprocessors ["tokens-studio"]
|
||||||
|
;; Silences style dictionary logs and errors
|
||||||
|
;; We handle token errors in the UI
|
||||||
|
:log {:verbosity "silent"
|
||||||
|
:warnings "silent"
|
||||||
|
:errors {:brokenReferences "console"}}})
|
||||||
|
|
||||||
(defn tokens->style-dictionary+
|
(defn parse-sd-token-color-value
|
||||||
"Resolves references and math expressions using StyleDictionary.
|
"Parses `value` of a color `sd-token` into a map like `{:value 1 :unit \"px\"}`.
|
||||||
Returns a promise with the resolved dictionary."
|
If the value is not parseable and/or has missing references returns a map with `:errors`."
|
||||||
[tokens]
|
[value]
|
||||||
(let [data (cond-> {:tokens tokens
|
(if-let [tc (tinycolor/valid-color value)]
|
||||||
:platforms {:json {:transformGroup "tokens-studio"
|
{:value value :unit (tinycolor/color-format tc)}
|
||||||
:files [{:format "custom/json"
|
{:errors [(wte/error-with-value :error.token/invalid-color value)]}))
|
||||||
:destination "fake-filename"}]}}
|
|
||||||
:log {:verbosity "silent"
|
|
||||||
:warnings "silent"
|
|
||||||
:errors {:brokenReferences "console"}}
|
|
||||||
:preprocessors ["tokens-studio"]}
|
|
||||||
(l/enabled? "app.main.ui.workspace.tokens.style-dictionary" :debug)
|
|
||||||
(update :log merge {:verbosity "verbose"
|
|
||||||
:warnings "warn"}))
|
|
||||||
js-data (clj->js data)]
|
|
||||||
(l/debug :hint "Input Data" :js/data js-data)
|
|
||||||
(sd. js-data)))
|
|
||||||
|
|
||||||
(defn resolve-sd-tokens+
|
(defn parse-sd-token-dimensions-value
|
||||||
"Resolves references and math expressions using StyleDictionary.
|
"Parses `value` of a dimensions `sd-token` into a map like `{:value 1 :unit \"px\"}`.
|
||||||
Returns a promise with the resolved dictionary."
|
If the `value` is not parseable and/or has missing references returns a map with `:errors`."
|
||||||
[tokens]
|
[value]
|
||||||
(let [performance-start (js/performance.now)
|
(or
|
||||||
sd (tokens->style-dictionary+ tokens)]
|
(wtt/parse-token-value value)
|
||||||
(l/debug :hint "StyleDictionary" :js/style-dictionary sd)
|
(if-let [references (seq (ctob/find-token-value-references value))]
|
||||||
(-> sd
|
{:errors [(wte/error-with-value :error.style-dictionary/missing-reference references)]
|
||||||
|
:references references}
|
||||||
|
{:errors [(wte/error-with-value :error.style-dictionary/invalid-token-value value)]})))
|
||||||
|
|
||||||
|
(defn process-sd-tokens
|
||||||
|
"Converts a StyleDictionary dictionary with resolved tokens (aka `sd-tokens`) back to clojure.
|
||||||
|
The `get-origin-token` argument should be a function that takes an `sd-token` and returns the original penpot token, so we can merge the resolved attributes back in.
|
||||||
|
|
||||||
|
The `sd-token` will have references in `value` replaced with the computed value as a string.
|
||||||
|
Here's an example for a `sd-token`:
|
||||||
|
```js
|
||||||
|
{
|
||||||
|
name: 'token.with.reference',
|
||||||
|
value: '12px',
|
||||||
|
type: 'border-radius',
|
||||||
|
path: ['token', 'with', 'reference'],
|
||||||
|
|
||||||
|
// The penpot origin token converted to a js object
|
||||||
|
original: {
|
||||||
|
name: 'token.with.reference',
|
||||||
|
value: '{referenced.token}',
|
||||||
|
type: 'border-radius'
|
||||||
|
},
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
We also convert `sd-token` value string into a unit that can be used as penpot shape attributes.
|
||||||
|
- Dimensions like '12px' will be converted into numbers
|
||||||
|
- Colors will be validated & converted to hex
|
||||||
|
|
||||||
|
Lastly we check for errors in each token
|
||||||
|
`sd-token` will keep the missing references in the `value` (E.g \"{missing} + {existing}\" -> \"{missing} + 12px\")
|
||||||
|
So we parse out the missing references and add them to `:errors` in the final token."
|
||||||
|
[sd-tokens get-origin-token]
|
||||||
|
(reduce
|
||||||
|
(fn [acc ^js sd-token]
|
||||||
|
(let [origin-token (get-origin-token sd-token)
|
||||||
|
value (.-value sd-token)
|
||||||
|
parsed-token-value (case (:type origin-token)
|
||||||
|
:color (parse-sd-token-color-value value)
|
||||||
|
(parse-sd-token-dimensions-value value))
|
||||||
|
output-token (if (:errors parsed-token-value)
|
||||||
|
(merge origin-token parsed-token-value)
|
||||||
|
(assoc origin-token
|
||||||
|
:resolved-value (:value parsed-token-value)
|
||||||
|
:unit (:unit parsed-token-value)))]
|
||||||
|
(assoc acc (:name output-token) output-token)))
|
||||||
|
{} sd-tokens))
|
||||||
|
|
||||||
|
(defprotocol IStyleDictionary
|
||||||
|
(add-tokens [_ tokens])
|
||||||
|
(enable-debug [_])
|
||||||
|
(get-config [_])
|
||||||
|
(build-dictionary [_]))
|
||||||
|
|
||||||
|
(deftype StyleDictionary [config]
|
||||||
|
IStyleDictionary
|
||||||
|
(add-tokens [_ tokens]
|
||||||
|
(StyleDictionary. (assoc config :tokens tokens)))
|
||||||
|
|
||||||
|
(enable-debug [_]
|
||||||
|
(StyleDictionary. (update config :log merge {:verbosity "verbose"})))
|
||||||
|
|
||||||
|
(get-config [_]
|
||||||
|
config)
|
||||||
|
|
||||||
|
(build-dictionary [_]
|
||||||
|
(-> (sd. (clj->js config))
|
||||||
(.buildAllPlatforms "json")
|
(.buildAllPlatforms "json")
|
||||||
(.catch #(l/error :hint "Styledictionary build error" :js/error %))
|
(p/then #(.-allTokens ^js %)))))
|
||||||
(.then (fn [^js resp]
|
|
||||||
(let [performance-end (js/performance.now)
|
(defn resolve-tokens-tree+
|
||||||
duration-ms (- performance-end performance-start)
|
([tokens-tree get-token]
|
||||||
resolved-tokens (.-allTokens resp)]
|
(resolve-tokens-tree+ tokens-tree get-token (StyleDictionary. default-config)))
|
||||||
(l/debug :hint (str "Time elapsed" duration-ms "ms") :duration duration-ms)
|
([tokens-tree get-token style-dictionary]
|
||||||
(l/debug :hint "Resolved tokens" :js/tokens resolved-tokens)
|
(-> style-dictionary
|
||||||
resolved-tokens))))))
|
(add-tokens tokens-tree)
|
||||||
|
(build-dictionary)
|
||||||
|
(p/then #(process-sd-tokens % get-token)))))
|
||||||
|
|
||||||
|
(defn sd-token-name [^js sd-token]
|
||||||
|
(.. sd-token -original -name))
|
||||||
|
|
||||||
|
(defn sd-token-uuid [^js sd-token]
|
||||||
|
(uuid (.-uuid (.-id ^js sd-token))))
|
||||||
|
|
||||||
|
(defn resolve-tokens+ [tokens]
|
||||||
|
(resolve-tokens-tree+ (ctob/tokens-tree tokens) #(get tokens (sd-token-name %))))
|
||||||
|
|
||||||
|
(defn resolve-tokens-interactive+
|
||||||
|
"Interactive check of resolving tokens.
|
||||||
|
Uses a ids map to backtrace the original token from the resolved StyleDictionary token.
|
||||||
|
|
||||||
|
We have to pass in all tokens from all sets in the entire library to style dictionary
|
||||||
|
so we know if references are missing / to resolve them and possibly show interactive previews (in the tokens form) to the user.
|
||||||
|
|
||||||
|
Since we're using the :name path as the identifier we might be throwing away or overriding tokens in the tree that we pass to StyleDictionary.
|
||||||
|
|
||||||
|
So to get back the original token from the resolved sd-token (see my updates for what an sd-token is) we include a temporary :id for the token that we pass to StyleDictionary,
|
||||||
|
this way after the resolving computation we can restore any token, even clashing ones with the same :name path by just looking up that :id in the ids map."
|
||||||
|
[tokens]
|
||||||
|
(let [{:keys [tokens-tree ids]} (ctob/backtrace-tokens-tree tokens)]
|
||||||
|
(resolve-tokens-tree+ tokens-tree #(get ids (sd-token-uuid %)))))
|
||||||
|
|
||||||
|
(defn resolve-tokens-with-errors+ [tokens]
|
||||||
|
(resolve-tokens-tree+
|
||||||
|
(ctob/tokens-tree tokens)
|
||||||
|
#(get tokens (sd-token-name %))
|
||||||
|
(StyleDictionary. (assoc default-config :log {:verbosity "verbose"}))))
|
||||||
|
|
||||||
|
;; === Import
|
||||||
|
|
||||||
|
(defn reference-errors
|
||||||
|
"Extracts reference errors from StyleDictionary."
|
||||||
|
[err]
|
||||||
|
(let [[header-1 header-2 & errors] (str/split err "\n")]
|
||||||
|
(when (and
|
||||||
|
(= header-1 "Error: ")
|
||||||
|
(= header-2 "Reference Errors:"))
|
||||||
|
errors)))
|
||||||
|
|
||||||
|
(defn process-json-stream [data-stream]
|
||||||
|
(->> data-stream
|
||||||
|
(rx/map (fn [data]
|
||||||
|
(try
|
||||||
|
(-> (str/replace data "/" "-") ;; TODO Remove when token groups work
|
||||||
|
(t/decode-str))
|
||||||
|
(catch js/Error e
|
||||||
|
(throw (wte/error-ex-info :error.import/json-parse-error data e))))))
|
||||||
|
(rx/map (fn [json-data]
|
||||||
|
(try
|
||||||
|
(ctob/decode-dtcg-json (ctob/ensure-tokens-lib nil) json-data)
|
||||||
|
(catch js/Error e
|
||||||
|
(throw (wte/error-ex-info :error.import/invalid-json-data json-data e))))))
|
||||||
|
(rx/mapcat (fn [tokens-lib]
|
||||||
|
(try
|
||||||
|
(-> (ctob/get-all-tokens tokens-lib)
|
||||||
|
(resolve-tokens-with-errors+)
|
||||||
|
(p/then (fn [_] tokens-lib))
|
||||||
|
(p/catch (fn [sd-error]
|
||||||
|
(let [reference-errors (reference-errors sd-error)
|
||||||
|
err (if reference-errors
|
||||||
|
(wte/error-ex-info :error.import/style-dictionary-reference-errors reference-errors sd-error)
|
||||||
|
(wte/error-ex-info :error.import/style-dictionary-unknown-error sd-error sd-error))]
|
||||||
|
(throw err)))))
|
||||||
|
(catch js/Error e
|
||||||
|
(p/rejected (wte/error-ex-info :error.import/style-dictionary-unknown-error "" e))))))))
|
||||||
|
|
||||||
|
;; === Errors
|
||||||
|
|
||||||
(defn humanize-errors [{:keys [errors value] :as _token}]
|
(defn humanize-errors [{:keys [errors value] :as _token}]
|
||||||
(->> (map (fn [err]
|
(->> (map (fn [err]
|
||||||
|
@ -71,51 +214,18 @@
|
||||||
errors)
|
errors)
|
||||||
(str/join "\n")))
|
(str/join "\n")))
|
||||||
|
|
||||||
(defn resolve-tokens+
|
;; === Hooks
|
||||||
[tokens & {:keys [names-map?] :as config}]
|
|
||||||
(let [{:keys [tree ids-map]} (wtt/token-names-tree-id-map tokens)]
|
|
||||||
(p/let [sd-tokens (resolve-sd-tokens+ tree)]
|
|
||||||
(let [resolved-tokens (reduce
|
|
||||||
(fn [acc ^js cur]
|
|
||||||
(let [{:keys [type] :as origin-token} (if names-map?
|
|
||||||
(get tokens (.. cur -original -name))
|
|
||||||
(get ids-map (uuid (.-uuid (.-id cur)))))
|
|
||||||
value (.-value cur)
|
|
||||||
token-or-err (case type
|
|
||||||
:color (if-let [tc (tinycolor/valid-color value)]
|
|
||||||
{:value value :unit (tinycolor/color-format tc)}
|
|
||||||
{:errors [(wte/error-with-value :error.token/invalid-color value)]})
|
|
||||||
(or (wtt/parse-token-value value)
|
|
||||||
(if-let [references (-> (ctob/find-token-value-references value)
|
|
||||||
(seq))]
|
|
||||||
{:errors [(wte/error-with-value :error.style-dictionary/missing-reference references)]
|
|
||||||
:references references}
|
|
||||||
{:errors [(wte/error-with-value :error.style-dictionary/invalid-token-value value)]})))
|
|
||||||
output-token (if (:errors token-or-err)
|
|
||||||
(merge origin-token token-or-err)
|
|
||||||
(assoc origin-token
|
|
||||||
:resolved-value (:value token-or-err)
|
|
||||||
:unit (:unit token-or-err)))]
|
|
||||||
(assoc acc (wtt/token-identifier output-token) output-token)))
|
|
||||||
{} sd-tokens)]
|
|
||||||
(l/debug :hint "Resolved tokens" :js/tokens resolved-tokens)
|
|
||||||
resolved-tokens))))
|
|
||||||
|
|
||||||
;; Hooks -----------------------------------------------------------------------
|
|
||||||
|
|
||||||
(defonce !tokens-cache (atom nil))
|
(defonce !tokens-cache (atom nil))
|
||||||
|
|
||||||
(defonce !theme-tokens-cache (atom nil))
|
(defonce !theme-tokens-cache (atom nil))
|
||||||
|
|
||||||
(defn get-cached-tokens [tokens]
|
|
||||||
(get @!tokens-cache tokens tokens))
|
|
||||||
|
|
||||||
(defn use-resolved-tokens
|
(defn use-resolved-tokens
|
||||||
"The StyleDictionary process function is async, so we can't use resolved values directly.
|
"The StyleDictionary process function is async, so we can't use resolved values directly.
|
||||||
|
|
||||||
This hook will return the unresolved tokens as state until they are processed,
|
This hook will return the unresolved tokens as state until they are processed,
|
||||||
then the state will be updated with the resolved tokens."
|
then the state will be updated with the resolved tokens."
|
||||||
[tokens & {:keys [cache-atom names-map?]
|
[tokens & {:keys [cache-atom interactive?]
|
||||||
:or {cache-atom !tokens-cache}
|
:or {cache-atom !tokens-cache}
|
||||||
:as config}]
|
:as config}]
|
||||||
(let [tokens-state (mf/use-state (get @cache-atom tokens))]
|
(let [tokens-state (mf/use-state (get @cache-atom tokens))]
|
||||||
|
@ -124,7 +234,7 @@
|
||||||
(fn []
|
(fn []
|
||||||
(let [cached (get @cache-atom tokens)]
|
(let [cached (get @cache-atom tokens)]
|
||||||
(cond
|
(cond
|
||||||
(nil? tokens) (if names-map? {} [])
|
(nil? tokens) nil
|
||||||
;; The tokens are already processing somewhere
|
;; The tokens are already processing somewhere
|
||||||
(p/promise? cached) (-> cached
|
(p/promise? cached) (-> cached
|
||||||
(p/then #(reset! tokens-state %))
|
(p/then #(reset! tokens-state %))
|
||||||
|
@ -132,19 +242,19 @@
|
||||||
;; Get the cached entry
|
;; Get the cached entry
|
||||||
(some? cached) (reset! tokens-state cached)
|
(some? cached) (reset! tokens-state cached)
|
||||||
;; No cached entry, start processing
|
;; No cached entry, start processing
|
||||||
:else (let [promise+ (resolve-tokens+ tokens config)]
|
:else (let [promise+ (if interactive?
|
||||||
|
(resolve-tokens-interactive+ tokens)
|
||||||
|
(resolve-tokens+ tokens))]
|
||||||
(swap! cache-atom assoc tokens promise+)
|
(swap! cache-atom assoc tokens promise+)
|
||||||
(p/then promise+ (fn [resolved-tokens]
|
(p/then promise+ (fn [resolved-tokens]
|
||||||
(swap! cache-atom assoc tokens resolved-tokens)
|
(swap! cache-atom assoc tokens resolved-tokens)
|
||||||
(reset! tokens-state resolved-tokens))))))))
|
(reset! tokens-state resolved-tokens))))))))
|
||||||
@tokens-state))
|
@tokens-state))
|
||||||
|
|
||||||
(defn use-resolved-workspace-tokens [& {:as config}]
|
(defn use-resolved-workspace-tokens []
|
||||||
(-> (mf/deref refs/workspace-selected-token-set-tokens)
|
(-> (mf/deref refs/workspace-selected-token-set-tokens)
|
||||||
(use-resolved-tokens config)))
|
(use-resolved-tokens)))
|
||||||
|
|
||||||
(defn use-active-theme-sets-tokens [& {:as config}]
|
(defn use-active-theme-sets-tokens []
|
||||||
(-> (mf/deref refs/workspace-active-theme-sets-tokens)
|
(-> (mf/deref refs/workspace-active-theme-sets-tokens)
|
||||||
(use-resolved-tokens (merge {:cache-atom !theme-tokens-cache
|
(use-resolved-tokens {:cache-atom !theme-tokens-cache})))
|
||||||
:names-map? true}
|
|
||||||
config))))
|
|
||||||
|
|
|
@ -125,7 +125,7 @@
|
||||||
(rx/from
|
(rx/from
|
||||||
(->
|
(->
|
||||||
(wtts/get-active-theme-sets-tokens-names-map state)
|
(wtts/get-active-theme-sets-tokens-names-map state)
|
||||||
(wtsd/resolve-tokens+ {:names-map? true})))
|
(wtsd/resolve-tokens+)))
|
||||||
(rx/mapcat
|
(rx/mapcat
|
||||||
(fn [sd-tokens]
|
(fn [sd-tokens]
|
||||||
(let [undo-id (js/Symbol)]
|
(let [undo-id (js/Symbol)]
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
(watch [_ state _]
|
(watch [_ state _]
|
||||||
(->> (rx/from (-> (get-in state [:workspace-data :tokens-lib])
|
(->> (rx/from (-> (get-in state [:workspace-data :tokens-lib])
|
||||||
(ctob/get-active-themes-set-tokens)
|
(ctob/get-active-themes-set-tokens)
|
||||||
(sd/resolve-tokens+ {:names-map? true})))
|
(sd/resolve-tokens+)))
|
||||||
(rx/mapcat #(rx/of (end)))))))
|
(rx/mapcat #(rx/of (end)))))))
|
||||||
|
|
||||||
(defn stop-on
|
(defn stop-on
|
||||||
|
|
|
@ -1,41 +1,115 @@
|
||||||
(ns token-tests.style-dictionary-test
|
(ns token-tests.style-dictionary-test
|
||||||
(:require
|
(:require
|
||||||
[app.common.data :as d]
|
[app.common.transit :as tr]
|
||||||
|
[app.common.types.tokens-lib :as ctob]
|
||||||
[app.main.ui.workspace.tokens.style-dictionary :as sd]
|
[app.main.ui.workspace.tokens.style-dictionary :as sd]
|
||||||
[app.main.ui.workspace.tokens.token :as wtt]
|
[beicon.v2.core :as rx]
|
||||||
[cljs.test :as t :include-macros true]
|
[cljs.test :as t :include-macros true]
|
||||||
[promesa.core :as p]))
|
[promesa.core :as p]))
|
||||||
|
|
||||||
(def border-radius-token
|
|
||||||
{:value "12px"
|
|
||||||
:name "borderRadius.sm"
|
|
||||||
:type :border-radius})
|
|
||||||
|
|
||||||
(def reference-border-radius-token
|
|
||||||
{:value "{borderRadius.sm} * 2"
|
|
||||||
:name "borderRadius.md-with-dashes"
|
|
||||||
:type :border-radius})
|
|
||||||
|
|
||||||
(def tokens (d/ordered-map
|
|
||||||
(:name border-radius-token) border-radius-token
|
|
||||||
(:name reference-border-radius-token) reference-border-radius-token))
|
|
||||||
(t/deftest resolve-tokens-test
|
(t/deftest resolve-tokens-test
|
||||||
(t/async
|
(t/async
|
||||||
done
|
done
|
||||||
(t/testing "resolves tokens using style-dictionary from a ids map"
|
(t/testing "resolves tokens using style-dictionary from a ids map"
|
||||||
(-> (sd/resolve-tokens+ tokens)
|
(let [tokens (-> (ctob/make-tokens-lib)
|
||||||
(p/finally
|
(ctob/add-set (ctob/make-token-set :name "core"))
|
||||||
(fn [resolved-tokens]
|
(ctob/add-token-in-set "core" (ctob/make-token {:name "borderRadius.sm"
|
||||||
(let [expected-tokens {"borderRadius.sm"
|
:value "12px"
|
||||||
(assoc border-radius-token
|
:type :border-radius}))
|
||||||
:resolved-value 12
|
(ctob/add-token-in-set "core" (ctob/make-token {:value "{borderRadius.sm} * 2"
|
||||||
:resolved-unit "px")
|
:name "borderRadius.md-with-dashes"
|
||||||
"borderRadius.md-with-dashes"
|
:type :border-radius}))
|
||||||
(assoc reference-border-radius-token
|
(ctob/get-all-tokens))]
|
||||||
:resolved-value 24
|
(-> (sd/resolve-tokens+ tokens)
|
||||||
:resolved-unit "px")}]
|
(p/finally
|
||||||
(t/is (= 12 (get-in resolved-tokens ["borderRadius.sm" :resolved-value])))
|
(fn [resolved-tokens]
|
||||||
(t/is (= "px" (get-in resolved-tokens ["borderRadius.sm" :unit])))
|
(t/is (= 12 (get-in resolved-tokens ["borderRadius.sm" :resolved-value])))
|
||||||
(t/is (= 24 (get-in resolved-tokens ["borderRadius.md-with-dashes" :resolved-value])))
|
(t/is (= "px" (get-in resolved-tokens ["borderRadius.sm" :unit])))
|
||||||
(t/is (= "px" (get-in resolved-tokens ["borderRadius.md-with-dashes" :unit])))
|
(t/is (= 24 (get-in resolved-tokens ["borderRadius.md-with-dashes" :resolved-value])))
|
||||||
(done))))))))
|
(t/is (= "px" (get-in resolved-tokens ["borderRadius.md-with-dashes" :unit])))
|
||||||
|
(done))))))))
|
||||||
|
|
||||||
|
(t/deftest process-json-stream-test
|
||||||
|
(t/async
|
||||||
|
done
|
||||||
|
(t/testing "processes empty json string"
|
||||||
|
(let [json (-> {"core" {"color" {"$value" "red"
|
||||||
|
"$type" "color"}}}
|
||||||
|
(tr/encode-str {:type :json-verbose}))]
|
||||||
|
(->> (rx/of json)
|
||||||
|
(sd/process-json-stream)
|
||||||
|
(rx/subs! (fn [tokens-lib]
|
||||||
|
(t/is (instance? ctob/TokensLib tokens-lib))
|
||||||
|
(t/is (= "red" (-> (ctob/get-set tokens-lib "core")
|
||||||
|
(ctob/get-token "color")
|
||||||
|
(:value))))
|
||||||
|
(done))))))))
|
||||||
|
|
||||||
|
(t/deftest reference-errros-test
|
||||||
|
(t/testing "Extracts reference errors from StyleDictionary errors"
|
||||||
|
;; Using unicode for the white-space after "Error: " as some editors might remove it and its more visible
|
||||||
|
(t/is (=
|
||||||
|
["Some token references (2) could not be found."
|
||||||
|
""
|
||||||
|
"foo.value tries to reference missing, which is not defined."
|
||||||
|
"color.value tries to reference missing, which is not defined."]
|
||||||
|
(sd/reference-errors "Error:\u0020
|
||||||
|
Reference Errors:
|
||||||
|
Some token references (2) could not be found.
|
||||||
|
|
||||||
|
foo.value tries to reference missing, which is not defined.
|
||||||
|
color.value tries to reference missing, which is not defined.")))
|
||||||
|
(t/is (nil? (sd/reference-errors nil)))
|
||||||
|
(t/is (nil? (sd/reference-errors "none")))))
|
||||||
|
|
||||||
|
(t/deftest process-empty-json-stream-test
|
||||||
|
(t/async
|
||||||
|
done
|
||||||
|
(t/testing "processes empty json string"
|
||||||
|
(->> (rx/of "{}")
|
||||||
|
(sd/process-json-stream)
|
||||||
|
(rx/subs! (fn [tokens-lib]
|
||||||
|
(t/is (instance? ctob/TokensLib tokens-lib))
|
||||||
|
(done)))))))
|
||||||
|
|
||||||
|
(t/deftest process-invalid-json-stream-test
|
||||||
|
(t/async
|
||||||
|
done
|
||||||
|
(t/testing "fails on invalid json"
|
||||||
|
(->> (rx/of "{,}")
|
||||||
|
(sd/process-json-stream)
|
||||||
|
(rx/subs!
|
||||||
|
(fn []
|
||||||
|
(throw (js/Error. "Should be an error")))
|
||||||
|
(fn [err]
|
||||||
|
(t/is (= :error.import/json-parse-error (:error/code (ex-data err))))
|
||||||
|
(done)))))))
|
||||||
|
|
||||||
|
(t/deftest process-non-token-json-stream-test
|
||||||
|
(t/async
|
||||||
|
done
|
||||||
|
(t/testing "fails on non-token json"
|
||||||
|
(->> (rx/of "{\"foo\": \"bar\"}")
|
||||||
|
(sd/process-json-stream)
|
||||||
|
(rx/subs!
|
||||||
|
(fn []
|
||||||
|
(throw (js/Error. "Should be an error")))
|
||||||
|
(fn [err]
|
||||||
|
(t/is (= :error.import/invalid-json-data (:error/code (ex-data err))))
|
||||||
|
(done)))))))
|
||||||
|
|
||||||
|
(t/deftest process-missing-references-json-test
|
||||||
|
(t/async
|
||||||
|
done
|
||||||
|
(t/testing "fails on missing references in tokens"
|
||||||
|
(let [json (-> {"core" {"color" {"$value" "{missing}"
|
||||||
|
"$type" "color"}}}
|
||||||
|
(tr/encode-str {:type :json-verbose}))]
|
||||||
|
(->> (rx/of json)
|
||||||
|
(sd/process-json-stream)
|
||||||
|
(rx/subs!
|
||||||
|
(fn []
|
||||||
|
(throw (js/Error. "Should be an error")))
|
||||||
|
(fn [err]
|
||||||
|
(t/is (= :error.import/style-dictionary-reference-errors (:error/code (ex-data err))))
|
||||||
|
(done))))))))
|
||||||
|
|
Loading…
Add table
Reference in a new issue