;; This Source Code Form is subject to the terms of the Mozilla Public ;; License, v. 2.0. If a copy of the MPL was not distributed with this ;; file, You can obtain one at http://mozilla.org/MPL/2.0/. ;; ;; Copyright (c) KALEIDOS INC (ns app.main.ui (:require [app.common.data :as d] [app.config :as cf] [app.main.data.common :as dcm] [app.main.data.team :as dtm] [app.main.refs :as refs] [app.main.repo :as rp] [app.main.router :as rt] [app.main.store :as st] [app.main.ui.context :as ctx] [app.main.ui.debug.icons-preview :refer [icons-preview]] [app.main.ui.ds.product.loader :refer [loader*]] [app.main.ui.error-boundary :refer [error-boundary*]] [app.main.ui.exports.files] [app.main.ui.frame-preview :as frame-preview] [app.main.ui.notifications :as notifications] [app.main.ui.onboarding.newsletter :refer [onboarding-newsletter]] [app.main.ui.onboarding.questions :refer [questions-modal]] [app.main.ui.onboarding.team-choice :refer [onboarding-team-modal]] [app.main.ui.releases :refer [release-notes-modal]] [app.main.ui.static :as static] [app.util.dom :as dom] [app.util.i18n :refer [tr]] [beicon.v2.core :as rx] [potok.v2.core :as ptk] [rumext.v2 :as mf])) (def auth-page (mf/lazy-component app.main.ui.auth/auth)) (def verify-token-page (mf/lazy-component app.main.ui.auth.verify-token/verify-token)) (def viewer-page (mf/lazy-component app.main.ui.viewer/viewer*)) (def dashboard-page (mf/lazy-component app.main.ui.dashboard/dashboard*)) (def settings-page (mf/lazy-component app.main.ui.settings/settings)) (def workspace-page (mf/lazy-component app.main.ui.workspace/workspace*)) (mf/defc workspace-legacy-redirect* {::mf/props :obj ::mf/private true} [{:keys [project-id file-id page-id layout]}] (mf/with-effect [] (->> (rp/cmd! :get-project {:id project-id}) (rx/subs! (fn [{:keys [team-id]}] (st/emit! (dcm/go-to-workspace :team-id team-id :file-id file-id :page-id page-id :layout layout))) ptk/handle-error))) [:> loader* {:title (tr "labels.loading") :overlay true}]) (mf/defc dashboard-legacy-redirect* {::mf/props :obj ::mf/private true} [{:keys [section team-id project-id search-term plugin-url]}] (let [section (case section :dashboard-legacy-search :dashboard-search :dashboard-legacy-projects :dashboard-recent :dashboard-legacy-files :dashboard-files :dashboard-legacy-libraries :dashboard-libraries :dashboard-legacy-fonts :dashboard-fonts :dashboard-legacy-font-providers :dashboard-font-providers :dashboard-legacy-team-members :dashboard-members :dashboard-legacy-team-invitations :dashboard-invitations :dashboard-legacy-team-webhooks :dashboard-webhooks :dashboard-legacy-team-settings :dashboard-settings)] (mf/with-effect [] (let [params {:team-id team-id :project-id project-id :search-term search-term :plugin plugin-url}] (st/emit! (rt/nav section (d/without-nils params))))) [:> loader* {:title (tr "labels.loading") :overlay true}])) (mf/defc viewer-legacy-redirect* {::mf/props :obj ::mf/private true} [{:keys [page-id file-id section index share-id interactions-mode frame-id share]}] (mf/with-effect [] (let [params {:page-id page-id :file-id file-id :section section :index index :share-id share-id :interactions-mode interactions-mode :frame-id frame-id :share share}] (st/emit! (rt/nav :viewer (d/without-nils params))))) [:> loader* {:title (tr "labels.loading") :overlay true}]) (mf/defc team-container* {::mf/props :obj ::mf/private true} [{:keys [team-id children]}] (mf/with-effect [team-id] (st/emit! (dtm/initialize-team team-id)) (fn [] (st/emit! (dtm/finalize-team team-id)))) (let [team (mf/deref refs/team)] (when (= team-id (:id team)) [:& (mf/provider ctx/current-team-id) {:value team-id} [:& (mf/provider ctx/permissions) {:value (:permissions team)} ;; The `:key` is mandatory here because we want to reinitialize ;; all dom tree instead of simple rerender. [:* {:key (str team-id)} children]]]))) (mf/defc page* {::mf/props :obj ::mf/private true} [{:keys [route profile]}] (let [{:keys [data params]} route props (get profile :props) section (get data :name) team (mf/deref refs/team) show-question-modal? (and (contains? cf/flags :onboarding) (not (:onboarding-viewed props)) (not (contains? props :onboarding-questions))) show-newsletter-modal? (and (contains? cf/flags :onboarding) (not (:onboarding-viewed props)) (not (contains? props :newsletter-updates)) (contains? props :onboarding-questions)) show-team-modal? (and (contains? cf/flags :onboarding) (not (:onboarding-viewed props)) (not (contains? props :onboarding-team-id)) (contains? props :newsletter-updates) (:is-default team)) show-release-modal? (and (contains? cf/flags :onboarding) (not (contains? cf/flags :hide-release-modal)) (:onboarding-viewed props) (not= (:release-notes-viewed props) (:main cf/version)) (not= "0.0" (:main cf/version)))] [:& (mf/provider ctx/current-route) {:value route} (case section (:auth-login :auth-register :auth-register-validate :auth-register-success :auth-recovery-request :auth-recovery) [:? [:& auth-page {:route route}]] :auth-verify-token [:? [:& verify-token-page {:route route}]] (:settings-profile :settings-password :settings-options :settings-feedback :settings-access-tokens :settings-notifications) [:? [:& settings-page {:route route}]] :debug-icons-preview (when *assert* [:& icons-preview]) (:dashboard-search :dashboard-recent :dashboard-files :dashboard-libraries :dashboard-fonts :dashboard-font-providers :dashboard-members :dashboard-invitations :dashboard-webhooks :dashboard-settings) (let [params (get params :query) team-id (some-> params :team-id uuid) project-id (some-> params :project-id uuid) search-term (some-> params :search-term) plugin-url (some-> params :plugin)] [:? #_[:& app.main.ui.releases/release-notes-modal {:version "2.4"}] #_[:& app.main.ui.onboarding/onboarding-templates-modal] #_[:& app.main.ui.onboarding/onboarding-modal] #_[:& app.main.ui.onboarding.team-choice/onboarding-team-modal] (cond show-question-modal? [:& questions-modal] show-newsletter-modal? [:& onboarding-newsletter] show-team-modal? [:& onboarding-team-modal {:go-to-team? true}] show-release-modal? [:& release-notes-modal {:version (:main cf/version)}]) [:> team-container* {:team-id team-id} [:> dashboard-page {:profile profile :section section :team-id team-id :search-term search-term :plugin-url plugin-url :project-id project-id}]]]) :workspace (let [params (get params :query) team-id (some-> params :team-id uuid) file-id (some-> params :file-id uuid) page-id (some-> params :page-id uuid) layout (some-> params :layout keyword)] [:? {} (when (cf/external-feature-flag "onboarding-03" "test") (cond show-question-modal? [:& questions-modal] show-newsletter-modal? [:& onboarding-newsletter] show-team-modal? [:& onboarding-team-modal {:go-to-team? false}] show-release-modal? [:& release-notes-modal {:version (:main cf/version)}])) [:> team-container* {:team-id team-id} [:> workspace-page {:team-id team-id :file-id file-id :page-id page-id :layout-name layout :key file-id}]]]) :viewer (let [params (get params :query) index (some-> (:index params) parse-long) share-id (some-> (:share-id params) parse-uuid) section (or (some-> (:section params) keyword) :interactions) file-id (some-> (:file-id params) parse-uuid) page-id (some-> (:page-id params) parse-uuid) imode (or (some-> (:interactions-mode params) keyword) :show-on-click) frame-id (some-> (:frame-id params) parse-uuid) share (:share params)] [:? {} [:> viewer-page {:page-id page-id :file-id file-id :frame-id frame-id :section section :index index :share-id share-id :interactions-mode imode :share share}]]) :workspace-legacy (let [project-id (some-> params :path :project-id uuid) file-id (some-> params :path :file-id uuid) page-id (some-> params :query :page-id uuid) layout (some-> params :query :layout keyword)] [:> workspace-legacy-redirect* {:project-id project-id :file-id file-id :page-id page-id :layout layout}]) (:dashboard-legacy-search :dashboard-legacy-projects :dashboard-legacy-files :dashboard-legacy-libraries :dashboard-legacy-fonts :dashboard-legacy-font-providers :dashboard-legacy-team-members :dashboard-legacy-team-invitations :dashboard-legacy-team-webhooks :dashboard-legacy-team-settings) (let [team-id (some-> params :path :team-id uuid) project-id (some-> params :path :project-id uuid) search-term (some-> params :query :search-term) plugin-url (some-> params :query :plugin)] [:> dashboard-legacy-redirect* {:team-id team-id :section section :project-id project-id :search-term search-term :plugin-url plugin-url}]) :viewer-legacy (let [{:keys [query-params path-params]} route {:keys [index share-id section page-id interactions-mode frame-id share] :or {section :interactions interactions-mode :show-on-click}} query-params {:keys [file-id]} path-params] [:> viewer-legacy-redirect* {:page-id page-id :file-id file-id :section section :index index :share-id share-id :interactions-mode (keyword interactions-mode) :frame-id frame-id :share share}]) :frame-preview [:& frame-preview/frame-preview] nil)])) (mf/defc app [] (let [route (mf/deref refs/route) edata (mf/deref refs/exception) profile (mf/deref refs/profile) theme (or (:theme profile) "default")] (mf/with-effect [theme] (dom/set-html-theme-color theme)) [:& (mf/provider ctx/current-route) {:value route} [:& (mf/provider ctx/current-profile) {:value profile} (if edata [:> static/exception-page* {:data edata :route route}] [:> error-boundary* {:fallback static/internal-error*} [:& notifications/current-notification] (when route [:> page* {:route route :profile profile}])])]]))