1
0
Fork 0
mirror of https://github.com/heyarne/airsonic-ui.git synced 2026-05-07 02:33:39 +02:00

Add user notifications

This commit is contained in:
Arne Schlüter 2018-06-11 15:47:39 +02:00
commit 68a96bc398
9 changed files with 140 additions and 18 deletions

View file

@ -14,6 +14,8 @@
;; for CIDER ;; for CIDER
[cider/cider-nrepl "0.18.0-SNAPSHOT"]] [cider/cider-nrepl "0.18.0-SNAPSHOT"]]
:nrepl {:port 9000}
:builds :builds
{:app {:target :browser {:app {:target :browser
:output-dir "public/app/js" :output-dir "public/app/js"

View file

@ -3,4 +3,5 @@
(def default-db (def default-db
{;; because navigate! executes asynchronously we force to display the login screen first {;; because navigate! executes asynchronously we force to display the login screen first
:current-route [routes/default-route]}) :current-route [routes/default-route]
:notifications (sorted-map)})

View file

@ -87,14 +87,16 @@
(let [creds (:credentials db)] (let [creds (:credentials db)]
(api/url (:server creds) endpoint (merge params (select-keys creds [:u :p]))))) (api/url (:server creds) endpoint (merge params (select-keys creds [:u :p])))))
(re-frame/reg-event-fx (defn api-request [{:keys [db]} [_ endpoint k params]]
:api-request
(fn [{:keys [db]} [_ endpoint k params]]
{:http-xhrio {:method :get {:http-xhrio {:method :get
:uri (api-url db endpoint params) :uri (api-url db endpoint params)
:response-format (ajax/json-response-format {:keywords? true}) :response-format (ajax/json-response-format {:keywords? true})
:on-success [::api-success k] :on-success [::api-success k]
:on-failure [::api-failure]}})) :on-failure [::api-failure]}})
(re-frame/reg-event-fx
:api-request
api-request)
(re-frame/reg-event-db (re-frame/reg-event-db
::api-success ::api-success
@ -173,3 +175,29 @@
{:routes/navigate [routes/default-route] {:routes/navigate [routes/default-route]
:routes/unset-credentials nil :routes/unset-credentials nil
:db db/default-db})) :db db/default-db}))
;; user messages
(defn show-notification
"Displays an informative message to the user"
[db [_ level message]]
(let [id (.now js/performance)]
(if (nil? message)
(let [message level
level :info]
(assoc-in db [:notifications id] {:level level
:message message}))
(assoc-in db [:notifications id] {:level level
:message message}))))
(re-frame/reg-event-db
:notification/show
show-notification)
(defn hide-notification
[db [_ notification-id]]
(update db :notifications dissoc notification-id))
(re-frame/reg-event-db
:notification/hide
hide-notification)

View file

@ -6,40 +6,38 @@
;; FIXME: this is used for cover images and it's quite ugly tbh ;; FIXME: this is used for cover images and it's quite ugly tbh
(re-frame/reg-sub (re-frame/reg-sub
::login ::login
(fn [db] (fn [db _]
(select-keys (:credentials db) [:u :p]))) (select-keys (:credentials db) [:u :p])))
(re-frame/reg-sub (re-frame/reg-sub
::user ::user
(fn [{:keys [credentials]}] (fn [{:keys [credentials]} [_]]
{:name (:u credentials)})) {:name (:u credentials)}))
(re-frame/reg-sub (re-frame/reg-sub
::server ::server
(fn [db] (fn [db _]
(get-in db [:credentials :server]))) (get-in db [:credentials :server])))
;; current hashbang ;; current hashbang
(re-frame/reg-sub (re-frame/reg-sub
::current-route ::current-route
(fn [db] (fn [db _]
(:current-route db))) (:current-route db)))
;; ---
;; TODO: Make this nice and clean ;; TODO: Make this nice and clean
(re-frame/reg-sub (re-frame/reg-sub
::current-content ::current-content
(fn [db] (fn [db _]
(db :response))) (:response db)))
(re-frame/reg-sub (re-frame/reg-sub
; returns info on the current song as is (basically the metadata you can read from the file system) ; returns info on the current song as is (basically the metadata you can read from the file system)
::currently-playing ::currently-playing
(fn [db] (fn [db _]
(db :currently-playing))) (:currently-playing db)))
(re-frame/reg-sub (re-frame/reg-sub
::is-playing? ::is-playing?
@ -49,3 +47,10 @@
(let [status (:status currently-playing)] (let [status (:status currently-playing)]
(and (not (:paused? status)) (and (not (:paused? status))
(not (:ended? status)))))) (not (:ended? status))))))
;; user notifications
(re-frame/reg-sub
::notifications
(fn [db _]
(:notifications db)))

View file

@ -22,3 +22,18 @@
(defn cover-url [server credentials item size] (defn cover-url [server credentials item size]
(url server "getCoverArt" (merge {:id (:coverArt item) :size size} credentials))) (url server "getCoverArt" (merge {:id (:coverArt item) :size size} credentials)))
(defn is-error? [response]
(= "failed" (get-in response [:subsonic-response :status])))
(defn unwrap-response
"Retrieves the actual response body"
[response]
(if (is-error? response)
(let [error (:error response)]
(throw (ex-info (:message response) error)))
(-> (get-in response [:subsonic-response])
(dissoc :status :version)
vals
first)))

View file

@ -46,12 +46,26 @@
{:on-click #(dispatch [::events/initialize-db]) :href "#"} {:on-click #(dispatch [::events/initialize-db]) :href "#"}
(str "Logout (" (:name user) ")")]]]]) (str "Logout (" (:name user) ")")]]]])
;; user notifications
(defn notification-list [notifications]
[:div.notifications
(for [[id notification] notifications]
(let [class (case (:level notification)
:error "danger"
"info")]
^{:key id} [:div {:class-name (str "notification is-small is-" class)}
[:button.delete {:on-click #(dispatch [:notification/hide id])}]
(:message notification)]))])
;; putting everything together ;; putting everything together
(defn app [route params query] (defn app [route params query]
(let [user @(subscribe [::subs/user]) (let [user @(subscribe [::subs/user])
notifications @(subscribe [::subs/notifications])
content @(subscribe [::subs/current-content])] content @(subscribe [::subs/current-content])]
[:div [:div
[notification-list notifications]
[:main.columns [:main.columns
[:div.column.is-2.sidebar [:div.column.is-2.sidebar
[sidebar user]] [sidebar user]]

View file

@ -73,3 +73,11 @@
.table .table
.grow .grow
width: 100% width: 100%
// floating notifications
.notifications
@extend .container
padding-top: 3.2rem
position: fixed
left: 0
right: 0

View file

@ -43,3 +43,26 @@
(testing "When there's no previous login data" (testing "When there's no previous login data"
(testing "remembering has no effect" (testing "remembering has no effect"
(is (nil? (events/try-remember-user {} [:_])))))) (is (nil? (events/try-remember-user {} [:_]))))))
(defn- first-notification [db]
(-> (:notifications db) vals first))
(deftest user-notifications
(testing "Should be able to display a message with an assigned level"
(is (= :error (:level (first-notification (events/show-notification {} [:_ :error "foo"])))))
(is (= :info (:level (first-notification (events/show-notification {} [:_ :info "some other message"]))))))
(testing "Should default to level :info"
(is (= :info (:level (first-notification (events/show-notification {} [:_ "and another one"]))))))
(testing "Should create a unique id for each message"
(let [state (->
{}
(events/show-notification [:_ :info "Something something"])
(events/show-notification [:_ :error "Something important"]))
ids (keys (:notifications state))]
(is (= (count ids) (count (set ids))))))
(testing "Should remove a message, given it's id"
(let [state (events/show-notification {} [:_ "This is a notification"])
id (-> (:notifications state)
keys
first)]
(is (empty? (:notifications (events/hide-notification state [:_ id])))))))

View file

@ -9,7 +9,17 @@
(api/url server endpoint {})) (api/url server endpoint {}))
(def fixtures (def fixtures
{:default-url (url "http://localhost:8080" "ping")}) {:default-url (url "http://localhost:8080" "ping")
:responses {:error {:subsonic-response
{:error {:code 40
:message "Wrong username or password"}
:status "failed"
:version "1.15.0"}}}
:ok {:subsonic-response
{:scanStatus {:count 10326
:scanning false}
:status "ok"
:version "1.15.0"}}})
(deftest general-url-construction (deftest general-url-construction
(testing "Handles missing slashes" (testing "Handles missing slashes"
@ -30,3 +40,19 @@
(is (true? (str/includes? (api/cover-url "http://server.tld" {} album -1) (str "id=" (:coverArt album)))))) (is (true? (str/includes? (api/cover-url "http://server.tld" {} album -1) (str "id=" (:coverArt album))))))
(testing "Should scale an image to a given size" (testing "Should scale an image to a given size"
(is (true? (str/includes? (api/cover-url "http://server.tld" {} album 48) "size=48")))))) (is (true? (str/includes? (api/cover-url "http://server.tld" {} album 48) "size=48"))))))
(deftest response-handling
(testing "Should unwrap responses"
(let [response (get-in fixtures [:responses :ok])]
(is (= (get-in response [:subsonic-response :scanStatus])
(api/unwrap-response response)))))
(testing "Should detect errors"
(is (true? (api/is-error? (get-in fixtures [:responses :error]))))
(is (false? (api/is-error? (get-in fixtures [:responses :ok])))))
(testing "Should throw an informative error when trying to unwrap an erroneous response"
(let [error-response (get-in fixtures [:responses :error])]
(is (thrown? ExceptionInfo (api/unwrap-response error-response)))
(try
(api/unwrap-response error-response)
(catch ExceptionInfo e
(= (:error error-response) (ex-data e)))))))