0
Fork 0
mirror of https://github.com/penpot/penpot.git synced 2025-01-22 14:39:45 -05:00

♻️ Refactor auth related components

This commit is contained in:
Andrey Antukh 2023-07-20 14:54:05 +02:00
parent c598656f60
commit 3f2cd30f52
6 changed files with 267 additions and 222 deletions

View file

@ -92,7 +92,7 @@
(assoc :message (tr "errors.email-invalid")))))) (assoc :message (tr "errors.email-invalid"))))))
(mf/defc login-form (mf/defc login-form
[{:keys [params on-success-callback] :as props}] [{:keys [params on-success] :as props}]
(let [initial (mf/use-memo (mf/deps params) (constantly params)) (let [initial (mf/use-memo (mf/deps params) (constantly params))
totp* (mf/use-state false) totp* (mf/use-state false)
@ -104,64 +104,74 @@
:initial initial) :initial initial)
on-error on-error
(fn [cause] (mf/use-fn
(when (map? cause) (fn [form cause]
(err/print-trace! cause) (when (map? cause)
(err/print-data! cause) (err/print-trace! cause)
(err/print-explain! cause)) (err/print-data! cause)
(err/print-explain! cause))
(cond (cond
(and (= :totp (:code cause))
(= :negotiation (:type cause))) (and (= :totp (:code cause))
(do (= :negotiation (:type cause)))
(reset! totp* true) (reset! totp* true)
(reset! error (tr "errors.missing-totp"))
(swap! form (fn [form] (and (= :invalid-totp (:code cause))
(-> form (= :negotiation (:type cause)))
(update :errors assoc :totp {:message (tr "errors.missing-totp")}) (do
(update :touched assoc :totp true))))) ;; (reset! error (tr "errors.invalid-totp"))
(swap! form (fn [form]
(-> form
(update :errors assoc :totp {:message (tr "errors.invalid-totp")})
(update :touched assoc :totp true)))))
(and (= :restriction (:type cause)) (and (= :restriction (:type cause))
(= :passkey-disabled (:code cause))) (= :passkey-disabled (:code cause)))
(reset! error (tr "errors.wrong-credentials")) (reset! error (tr "errors.wrong-credentials"))
(and (= :restriction (:type cause)) (and (= :restriction (:type cause))
(= :profile-blocked (:code cause))) (= :profile-blocked (:code cause)))
(reset! error (tr "errors.profile-blocked")) (reset! error (tr "errors.profile-blocked"))
(and (= :restriction (:type cause)) (and (= :restriction (:type cause))
(= :admin-only-profile (:code cause))) (= :admin-only-profile (:code cause)))
(reset! error (tr "errors.profile-blocked")) (reset! error (tr "errors.profile-blocked"))
(and (= :validation (:type cause)) (and (= :validation (:type cause))
(= :wrong-credentials (:code cause))) (= :wrong-credentials (:code cause)))
(reset! error (tr "errors.wrong-credentials")) (reset! error (tr "errors.wrong-credentials"))
(and (= :validation (:type cause)) (and (= :validation (:type cause))
(= :account-without-password (:code cause))) (= :account-without-password (:code cause)))
(reset! error (tr "errors.wrong-credentials")) (reset! error (tr "errors.wrong-credentials"))
:else :else
(reset! error (tr "errors.generic")))) (reset! error (tr "errors.generic")))))
on-success-default on-success-default
(fn [data] (mf/use-fn
(when-let [token (:invitation-token data)] (fn [data]
(st/emit! (rt/nav :auth-verify-token {} {:token token})))) (when-let [token (:invitation-token data)]
(st/emit! (rt/nav :auth-verify-token {} {:token token})))))
on-success on-success
(fn [data] (mf/use-fn
(if (nil? on-success-callback) (mf/deps on-success)
(on-success-default data) (fn [data]
(on-success-callback))) (if (fn? on-success)
(on-success data)
(on-success-default data))))
on-submit on-submit
(mf/use-fn (mf/use-fn
(fn [form event] (fn [form event]
(let [event (dom/event->native-event event) (let [event (dom/event->native-event event)
submitter (unchecked-get event "submitter") submitter (unchecked-get event "submitter")
submitter (dom/get-data submitter "role")] submitter (dom/get-data submitter "role")
on-error (partial on-error form)
on-success (partial on-success form)]
(case submitter (case submitter
"login-with-passkey" "login-with-passkey"
@ -175,6 +185,7 @@
{:on-error on-error {:on-error on-error
:on-success on-success})] :on-success on-success})]
(st/emit! (du/login-with-password params))) (st/emit! (du/login-with-password params)))
nil)))) nil))))
on-submit-ldap on-submit-ldap
@ -284,7 +295,7 @@
(tr "auth.login-with-oidc-submit")]])) (tr "auth.login-with-oidc-submit")]]))
(mf/defc login-methods (mf/defc login-methods
[{:keys [params on-success-callback] :as props}] [{:keys [params on-success] :as props}]
[:* [:*
(when show-alt-login-buttons? (when show-alt-login-buttons?
[:* [:*
@ -306,35 +317,40 @@
(when (or (contains? cf/flags :login) (when (or (contains? cf/flags :login)
(contains? cf/flags :login-with-password) (contains? cf/flags :login-with-password)
(contains? cf/flags :login-with-ldap)) (contains? cf/flags :login-with-ldap))
[:& login-form {:params params :on-success-callback on-success-callback}])]) [:& login-form {:params params :on-success on-success}])])
(mf/defc login-page (mf/defc login-page
[{:keys [params] :as props}] {::mf/wrap-props false}
[:div.generic-form.login-form [{:keys [params]}]
[:div.form-container (let [nav-to-recovery (mf/use-fn #(st/emit! (rt/nav :auth-recovery-request)))
[:h1 {:data-test "login-title"} (tr "auth.login-title")] nav-to-register (mf/use-fn (mf/deps params) #(st/emit! (rt/nav :auth-register {} params)))
create-demo (mf/use-fn #(st/emit! (du/create-demo-profile)))]
[:& login-methods {:params params}] [:div.generic-form.login-form
[:div.form-container
[:h1 {:data-test "login-title"} (tr "auth.login-title")]
[:div.links [:& login-methods {:params params}]
(when (or (contains? cf/flags :login)
(contains? cf/flags :login-with-password))
[:div.link-entry
[:& lk/link {:action #(st/emit! (rt/nav :auth-recovery-request))
:data-test "forgot-password"}
(tr "auth.forgot-password")]])
(when (contains? cf/flags :registration) [:div.links
[:div.link-entry (when (or (contains? cf/flags :login)
[:span (tr "auth.register") " "] (contains? cf/flags :login-with-password))
[:& lk/link {:action #(st/emit! (rt/nav :auth-register {} params)) [:div.link-entry
:data-test "register-submit"} [:& lk/link {:on-click nav-to-recovery
(tr "auth.register-submit")]])] :data-test "forgot-password"}
(tr "auth.forgot-password")]])
(when (contains? cf/flags :demo-users) (when (contains? cf/flags :registration)
[:div.links.demo [:div.link-entry
[:div.link-entry [:span (tr "auth.register") " "]
[:span (tr "auth.create-demo-profile") " "] [:& lk/link {:on-click nav-to-register
[:& lk/link {:action #(st/emit! (du/create-demo-profile)) :data-test "register-submit"}
:data-test "demo-account-link"} (tr "auth.register-submit")]])]
(tr "auth.create-demo-account")]]])]])
(when (contains? cf/flags :demo-users)
[:div.links.demo
[:div.link-entry
[:span (tr "auth.create-demo-profile") " "]
[:& lk/link {:on-click create-demo
:data-test "demo-account-link"}
(tr "auth.create-demo-account")]]])]]))

View file

@ -21,7 +21,9 @@
[rumext.v2 :as mf])) [rumext.v2 :as mf]))
(s/def ::email ::us/email) (s/def ::email ::us/email)
(s/def ::recovery-request-form (s/keys :req-un [::email])) (s/def ::recovery-request-form
(s/keys :req-un [::email]))
(defn handle-error-messages (defn handle-error-messages
[errors _data] [errors _data]
(d/update-when errors :email (d/update-when errors :email
@ -31,24 +33,27 @@
(assoc :message (tr "errors.email-invalid")))))) (assoc :message (tr "errors.email-invalid"))))))
(mf/defc recovery-form (mf/defc recovery-form
[{:keys [on-success-callback] :as props}] {::mf/wrap-props false}
[{:keys [on-success]}]
(let [form (fm/use-form :spec ::recovery-request-form (let [form (fm/use-form :spec ::recovery-request-form
:validators [handle-error-messages] :validators [handle-error-messages]
:initial {}) :initial {})
submitted (mf/use-state false) submitted (mf/use-state false)
default-success-finish #(st/emit! (dm/info (tr "auth.notifications.recovery-token-sent"))) default-on-success
(mf/use-fn #(st/emit! (dm/info (tr "auth.notifications.recovery-token-sent"))))
on-success on-success
(mf/use-callback (mf/use-fn
(mf/deps default-on-success on-success)
(fn [cdata _] (fn [cdata _]
(reset! submitted false) (reset! submitted false)
(if (nil? on-success-callback) (if (fn? on-success)
(default-success-finish) (on-success (:email cdata))
(on-success-callback (:email cdata))))) (default-on-success))))
on-error on-error
(mf/use-callback (mf/use-fn
(fn [data {:keys [code] :as error}] (fn [data {:keys [code] :as error}]
(reset! submitted false) (reset! submitted false)
(case code (case code
@ -64,7 +69,7 @@
(rx/throw error)))) (rx/throw error))))
on-submit on-submit
(mf/use-callback (mf/use-fn
(fn [] (fn []
(reset! submitted true) (reset! submitted true)
(let [cdata (:clean-data @form) (let [cdata (:clean-data @form)
@ -74,32 +79,34 @@
(reset! form nil) (reset! form nil)
(st/emit! (du/request-profile-recovery params)))))] (st/emit! (du/request-profile-recovery params)))))]
[:& fm/form {:on-submit on-submit [:& fm/form {:on-submit on-submit :form form}
:form form}
[:div.fields-row [:div.fields-row
[:& fm/input {:name :email [:& fm/input
:label (tr "auth.email") {:name :email
:help-icon i/at :label (tr "auth.email")
:type "text"}]] :help-icon i/at
:type "text"}]]
[:> fm/submit-button* [:> fm/submit-button*
{:label (tr "auth.recovery-request-submit") {:label (tr "auth.recovery-request-submit")
:data-test "recovery-resquest-submit"}]])) :data-test "recovery-resquest-submit"}]]))
;; --- Recovery Request Page ;; --- Recovery Request Page
(mf/defc recovery-request-page (mf/defc recovery-request-page
[{:keys [params on-success-callback go-back-callback] :as props}] {::mf/wrap-props false}
(let [default-go-back #(st/emit! (rt/nav :auth-login)) [{:keys [params on-success on-go-back]}]
go-back (or go-back-callback default-go-back)] (let [default-go-back (mf/use-fn #(st/emit! (rt/nav :auth-login)))
on-go-back (or on-go-back default-go-back)]
[:section.generic-form [:section.generic-form
[:div.form-container [:div.form-container
[:h1 (tr "auth.recovery-request-title")] [:h1 (tr "auth.recovery-request-title")]
[:div.subtitle (tr "auth.recovery-request-subtitle")] [:div.subtitle (tr "auth.recovery-request-subtitle")]
[:& recovery-form {:params params :on-success-callback on-success-callback}] [:& recovery-form
{:params params
:on-success on-success}]
[:div.links [:div.links
[:div.link-entry [:div.link-entry
[:& lk/link {:action go-back [:& lk/link {:on-click on-go-back
:data-test "go-back-link"} :data-test "go-back-link"}
(tr "labels.go-back")]]]]])) (tr "labels.go-back")]]]]]))

View file

@ -16,6 +16,7 @@
[app.main.ui.auth.login :as login] [app.main.ui.auth.login :as login]
[app.main.ui.components.forms :as fm] [app.main.ui.components.forms :as fm]
[app.main.ui.components.link :as lk] [app.main.ui.components.link :as lk]
[app.main.ui.hooks :as hooks]
[app.main.ui.icons :as i] [app.main.ui.icons :as i]
[app.main.ui.messages :as msgs] [app.main.ui.messages :as msgs]
[app.util.i18n :refer [tr]] [app.util.i18n :refer [tr]]
@ -55,63 +56,64 @@
(s/keys :req-un [::password ::email] (s/keys :req-un [::password ::email]
:opt-un [::invitation-token])) :opt-un [::invitation-token]))
(defn- handle-prepare-register-error
[form {:keys [type code] :as cause}]
(condp = [type code]
[:restriction :registration-disabled]
(st/emit! (dm/error (tr "errors.registration-disabled")))
[:restriction :profile-blocked]
(st/emit! (dm/error (tr "errors.profile-blocked")))
[:validation :email-has-permanent-bounces]
(let [email (get @form [:data :email])]
(st/emit! (dm/error (tr "errors.email-has-permanent-bounces" email))))
[:validation :email-already-exists]
(swap! form assoc-in [:errors :email]
{:message "errors.email-already-exists"})
[:validation :email-as-password]
(swap! form assoc-in [:errors :password]
{:message "errors.email-as-password"})
(st/emit! (dm/error (tr "errors.generic")))))
(defn- handle-prepare-register-success
[params]
(st/emit! (rt/nav :auth-register-validate {} params)))
(mf/defc register-form (mf/defc register-form
[{:keys [params on-success-callback] :as props}] [{:keys [params on-success]}]
(let [initial (mf/use-memo (mf/deps params) (constantly params)) (let [initial (hooks/use-equal-memo params)
form (fm/use-form :spec ::register-form form (fm/use-form :spec ::register-form
:validators [validate :validators [validate
(fm/validate-not-empty :password (tr "auth.password-not-empty"))] (fm/validate-not-empty :password (tr "auth.password-not-empty"))]
:initial initial) :initial initial)
submitted? (mf/use-state false) submitted? (mf/use-state false)
on-success (fn [p] on-success-default
(if (nil? on-success-callback) (mf/use-fn #(st/emit! (rt/nav :auth-register-validate {} %)))
(handle-prepare-register-success p)
(on-success-callback p))) on-success
(mf/use-fn
(mf/deps on-success on-success-default)
(fn [_ data]
(if (fn? on-success)
(on-success data)
(on-success-default data))))
on-error
(mf/use-fn
(fn [form {:keys [type code] :as cause}]
(condp = [type code]
[:restriction :registration-disabled]
(st/emit! (dm/error (tr "errors.registration-disabled")))
[:restriction :profile-blocked]
(st/emit! (dm/error (tr "errors.profile-blocked")))
[:validation :email-has-permanent-bounces]
(let [email (get @form [:data :email])]
(st/emit! (dm/error (tr "errors.email-has-permanent-bounces" email))))
[:validation :email-already-exists]
(swap! form assoc-in [:errors :email]
{:message "errors.email-already-exists"})
[:validation :email-as-password]
(swap! form assoc-in [:errors :password]
{:message "errors.email-as-password"})
(st/emit! (dm/error (tr "errors.generic"))))))
on-submit on-submit
(mf/use-callback (mf/use-fn
(mf/deps on-success on-error)
(fn [form _event] (fn [form _event]
(reset! submitted? true) (reset! submitted? true)
(let [cdata (:clean-data @form)] (let [cdata (:clean-data @form)
on-error (partial on-error form)
on-success (partial on-success form)]
(->> (rp/cmd! :prepare-register-profile cdata) (->> (rp/cmd! :prepare-register-profile cdata)
(rx/map #(merge % params)) (rx/map #(merge % params))
(rx/finalize #(reset! submitted? false)) (rx/finalize #(reset! submitted? false))
(rx/subs (rx/subs on-success on-error)))))]
on-success
(partial handle-prepare-register-error form))))))]
[:& fm/form {:on-submit on-submit :form form}
[:& fm/form {:on-submit on-submit
:form form}
[:div.fields-row [:div.fields-row
[:& fm/input {:type "email" [:& fm/input {:type "email"
:name :email :name :email
@ -131,7 +133,7 @@
(mf/defc register-methods (mf/defc register-methods
[{:keys [params on-success-callback] :as props}] [{:keys [params on-success]}]
[:* [:*
(when login/show-alt-login-buttons? (when login/show-alt-login-buttons?
[:* [:*
@ -149,62 +151,39 @@
[:span.text (tr "labels.or")] [:span.text (tr "labels.or")]
[:span.line]])]) [:span.line]])])
[:& register-form {:params params :on-success-callback on-success-callback}]]) [:& register-form {:params params :on-success on-success}]])
(mf/defc register-page (mf/defc register-page
[{:keys [params] :as props}] [{:keys [params]}]
[:div.form-container (let [nav-to-login (mf/use-fn (mf/deps params) #(st/emit! (rt/nav :auth-login {} params)))
create-demo (mf/use-fn #(st/emit! (du/create-demo-profile)))]
[:h1 {:data-test "registration-title"} (tr "auth.register-title")] [:div.form-container
[:div.subtitle (tr "auth.register-subtitle")]
(when (contains? cf/flags :demo-warning) [:h1 {:data-test "registration-title"} (tr "auth.register-title")]
[:& demo-warning]) [:div.subtitle (tr "auth.register-subtitle")]
[:& register-methods {:params params}] (when (contains? cf/flags :demo-warning)
[:& demo-warning])
[:div.links [:& register-methods {:params params}]
[:div.link-entry
[:span (tr "auth.already-have-account") " "]
[:& lk/link {:action #(st/emit! (rt/nav :auth-login {} params)) [:div.links
:data-test "login-here-link"}
(tr "auth.login-here")]]
(when (contains? cf/flags :demo-users)
[:div.link-entry [:div.link-entry
[:span (tr "auth.create-demo-profile") " "] [:span (tr "auth.already-have-account") " "]
[:& lk/link {:action #(st/emit! (du/create-demo-profile))}
(tr "auth.create-demo-account")]])]]) [:& lk/link {:on-click nav-to-login
:data-test "login-here-link"}
(tr "auth.login-here")]]
(when (contains? cf/flags :demo-users)
[:div.link-entry
[:span (tr "auth.create-demo-profile") " "]
[:& lk/link {:on-click create-demo}
(tr "auth.create-demo-account")]])]]))
;; --- PAGE: register validation ;; --- PAGE: register validation
(defn- handle-register-error
[form error]
(case (:code error)
:email-already-exists
(swap! form assoc-in [:errors :email]
{:message "errors.email-already-exists"})
(do
(println (:explain error))
(st/emit! (dm/error (tr "errors.generic"))))))
(defn- handle-register-success
[data]
(cond
(some? (:invitation-token data))
(let [token (:invitation-token data)]
(st/emit! (rt/nav :auth-verify-token {} {:token token})))
;; The :is-active flag is true, when insecure-register is enabled
;; or the user used external auth provider.
(:is-active data)
(st/emit! (du/login-from-register))
:else
(st/emit! (rt/nav :auth-register-success {} {:email (:email data)}))))
(s/def ::accept-terms-and-privacy (s/and ::us/boolean true?)) (s/def ::accept-terms-and-privacy (s/and ::us/boolean true?))
(s/def ::accept-newsletter-subscription ::us/boolean) (s/def ::accept-newsletter-subscription ::us/boolean)
@ -218,30 +197,63 @@
::accept-newsletter-subscription]))) ::accept-newsletter-subscription])))
(mf/defc register-validate-form (mf/defc register-validate-form
[{:keys [params on-success-callback] :as props}] [{:keys [params on-success]}]
(let [form (fm/use-form :spec ::register-validate-form (let [params (hooks/use-equal-memo params)
form (fm/use-form :spec ::register-validate-form
:validators [(fm/validate-not-empty :fullname (tr "auth.name.not-all-space")) :validators [(fm/validate-not-empty :fullname (tr "auth.name.not-all-space"))
(fm/validate-length :fullname fm/max-length-allowed (tr "auth.name.too-long"))] (fm/validate-length :fullname fm/max-length-allowed (tr "auth.name.too-long"))]
:initial params) :initial params)
submitted? (mf/use-state false) submitted? (mf/use-state false)
on-success (fn [p] on-error
(if (nil? on-success-callback) (mf/use-fn
(handle-register-success p) (fn [form error]
(on-success-callback (:email p)))) (case (:code error)
:email-already-exists
(swap! form assoc-in [:errors :email]
{:message "errors.email-already-exists"})
(do
(println (:explain error))
(st/emit! (dm/error (tr "errors.generic")))))))
on-success-default
(mf/use-fn
(fn [data]
(cond
(some? (:invitation-token data))
(let [token (:invitation-token data)]
(st/emit! (rt/nav :auth-verify-token {} {:token token})))
;; The :is-active flag is true, when insecure-register is enabled
;; or the user used external auth provider.
(:is-active data)
(st/emit! (du/login-from-register))
:else
(st/emit! (rt/nav :auth-register-success {} {:email (:email data)})))))
on-success
(mf/use-fn
(mf/deps on-success on-success-default)
(fn [_ data]
(if (fn? on-success)
(on-success data)
(on-success-default data))))
on-submit on-submit
(mf/use-callback (mf/use-fn
(fn [form _event] (fn [form _event]
(reset! submitted? true) (reset! submitted? true)
(let [params (:clean-data @form)] (let [params (:clean-data @form)
on-success (partial on-success form)
on-error (partial on-error form)]
(->> (rp/cmd! :register-profile params) (->> (rp/cmd! :register-profile params)
(rx/finalize #(reset! submitted? false)) (rx/finalize #(reset! submitted? false))
(rx/subs on-success (rx/subs on-success on-error)))))]
(partial handle-register-error form))))))]
[:& fm/form {:on-submit on-submit [:& fm/form {:on-submit on-submit :form form}
:form form}
[:div.fields-row [:div.fields-row
[:& fm/input {:name :fullname [:& fm/input {:name :fullname
:label (tr "auth.fullname") :label (tr "auth.fullname")
@ -265,20 +277,20 @@
(mf/defc register-validate-page (mf/defc register-validate-page
[{:keys [params] :as props}] [{:keys [params]}]
[:div.form-container (let [nav-to-register (mf/use-fn #(st/emit! (rt/nav :auth-register {} {})))]
[:h1 {:data-test "register-title"} (tr "auth.register-title")] [:div.form-container
[:div.subtitle (tr "auth.register-subtitle")] [:h1 {:data-test "register-title"} (tr "auth.register-title")]
[:div.subtitle (tr "auth.register-subtitle")]
[:& register-validate-form {:params params}] [:& register-validate-form {:params params}]
[:div.links [:div.links
[:div.link-entry [:div.link-entry
[:& lk/link {:action #(st/emit! (rt/nav :auth-register {} {}))} [:& lk/link {:on-click nav-to-register} (tr "labels.go-back")]]]]))
(tr "labels.go-back")]]]])
(mf/defc register-success-page (mf/defc register-success-page
[{:keys [params] :as props}] [{:keys [params]}]
[:div.form-container [:div.form-container
[:div.notification-icon i/icon-verify] [:div.notification-icon i/icon-verify]
[:div.notification-text (tr "auth.verification-email-sent")] [:div.notification-text (tr "auth.verification-email-sent")]

View file

@ -9,13 +9,19 @@
[app.util.keyboard :as kbd] [app.util.keyboard :as kbd]
[rumext.v2 :as mf])) [rumext.v2 :as mf]))
(mf/defc link [{:keys [action klass data-test keyboard-action children]}] (mf/defc link
(let [keyboard-action (or keyboard-action action)] {::mf/wrap-props false}
[:a {:on-click action [{:keys [on-click class data-test on-key-enter children]}]
:class klass (let [on-key-enter (or on-key-enter on-click)
:on-key-down (fn [event] on-key-down
(when (kbd/enter? event) (mf/use-fn
(keyboard-action event))) (mf/deps on-key-enter)
(fn [event]
(when (and (kbd/enter? event) (fn? on-key-enter))
(on-key-enter event))))]
[:a {:on-click on-click
:on-key-down on-key-down
:class class
:tab-index "0" :tab-index "0"
:data-test data-test} :data-test data-test}
[:* children]])) children]))

View file

@ -629,19 +629,19 @@
[:ul.sidebar-nav.no-overflow [:ul.sidebar-nav.no-overflow
[:li.recent-projects [:li.recent-projects
{:class-name (when projects? "current")} {:class-name (when projects? "current")}
[:& link {:action go-projects [:& link {:on-click go-projects
:keyboard-action go-projects-with-key} :on-key-enter go-projects-with-key}
[:span.element-title (tr "labels.projects")]]] [:span.element-title (tr "labels.projects")]]]
[:li {:class-name (when drafts? "current")} [:li {:class-name (when drafts? "current")}
[:& link {:action go-drafts [:& link {:on-click go-drafts
:keyboard-action go-drafts-with-key} :on-key-enter go-drafts-with-key}
[:span.element-title (tr "labels.drafts")]]] [:span.element-title (tr "labels.drafts")]]]
[:li {:class-name (when libs? "current")} [:li {:class-name (when libs? "current")}
[:& link {:action go-libs [:& link {:on-click go-libs
:keyboard-action go-libs-with-key} :on-key-enter go-libs-with-key}
[:span.element-title (tr "labels.shared-libraries")]]]]] [:span.element-title (tr "labels.shared-libraries")]]]]]
[:hr] [:hr]
@ -650,8 +650,8 @@
[:ul.sidebar-nav.no-overflow [:ul.sidebar-nav.no-overflow
[:li {:class-name (when fonts? "current")} [:li {:class-name (when fonts? "current")}
[:& link {:action go-fonts [:& link {:on-click go-fonts
:keyboard-action go-fonts-with-key :on-key-enter go-fonts-with-key
:data-test "fonts"} :data-test "fonts"}
[:span.element-title (tr "labels.fonts")]]]]] [:span.element-title (tr "labels.fonts")]]]]]

View file

@ -38,17 +38,21 @@
(fn [event] (fn [event]
(dom/prevent-default event) (dom/prevent-default event)
(st/emit! (modal/hide))) (st/emit! (modal/hide)))
success-email-sent success-email-sent
(fn [email] (fn [{:keys [email]}]
(reset! user-email email) (reset! user-email email)
(set-current-section :email-sent)) (set-current-section :email-sent))
success-login success-login
(fn [] (fn [_]
(.reload js/window.location true)) (.reload js/window.location true))
success-register success-register
(fn [data] (fn [data]
(reset! register-token (:token data)) (reset! register-token (:token data))
(set-current-section :register-validate))] (set-current-section :register-validate))]
(mf/with-effect [] (mf/with-effect []
(swap! storage assoc :redirect-url uri)) (swap! storage assoc :redirect-url uri))
@ -66,7 +70,7 @@
:login :login
[:div.generic-form.login-form [:div.generic-form.login-form
[:div.form-container [:div.form-container
[:& login-methods {:on-success-callback success-login}] [:& login-methods {:on-success success-login}]
[:div.links [:div.links
[:div.link-entry [:div.link-entry
[:a {:on-click #(set-current-section :recovery-request)} [:a {:on-click #(set-current-section :recovery-request)}
@ -78,7 +82,7 @@
:register :register
[:div.form-container [:div.form-container
[:& register-methods {:on-success-callback success-register}] [:& register-methods {:on-success success-register}]
[:div.links [:div.links
[:div.link-entry [:div.link-entry
[:span (tr "auth.already-have-account") " "] [:span (tr "auth.already-have-account") " "]
@ -88,15 +92,15 @@
:register-validate :register-validate
[:div.form-container [:div.form-container
[:& register-validate-form {:params {:token @register-token} [:& register-validate-form {:params {:token @register-token}
:on-success-callback success-email-sent}] :on-success success-email-sent}]
[:div.links [:div.links
[:div.link-entry [:div.link-entry
[:a {:on-click #(set-current-section :register)} [:a {:on-click #(set-current-section :register)}
(tr "labels.go-back")]]]] (tr "labels.go-back")]]]]
:recovery-request :recovery-request
[:& recovery-request-page {:go-back-callback #(set-current-section :login) [:& recovery-request-page {:on-go-back #(set-current-section :login)
:on-success-callback success-email-sent}] :on-success success-email-sent}]
:email-sent :email-sent
[:div.form-container [:div.form-container
[:& register-success-page {:params {:email @user-email}}]])] [:& register-success-page {:params {:email @user-email}}]])]