This page is not created by, affiliated with, or supported by Slack Technologies, Inc.
2018-07-16
Channels
- # beginners (48)
- # cider (21)
- # clara (6)
- # cljdoc (3)
- # cljs-dev (11)
- # cljsrn (5)
- # clojure (30)
- # clojure-canada (1)
- # clojure-dusseldorf (2)
- # clojure-italy (10)
- # clojure-losangeles (2)
- # clojure-nl (4)
- # clojure-russia (8)
- # clojure-spain (18)
- # clojure-uk (39)
- # clojurescript (84)
- # core-async (17)
- # cursive (22)
- # data-science (27)
- # datomic (27)
- # docker (3)
- # editors (5)
- # emacs (2)
- # figwheel-main (18)
- # fulcro (54)
- # hoplon (3)
- # hyperfiddle (2)
- # immutant (4)
- # jobs (1)
- # jobs-discuss (1)
- # lein-figwheel (7)
- # leiningen (3)
- # lumo (1)
- # onyx (5)
- # re-frame (64)
- # reagent (5)
- # reitit (7)
- # ring-swagger (6)
- # shadow-cljs (118)
- # specter (23)
- # tools-deps (38)
hi! is there any example of using https://github.com/ptaoussanis/tempura w/ re-frame? i was thinking about using subscriptions to get the correct translations for the current user but i don't want all of my views to become impure by depending on subscriptions, but i'd also like to avoid passing arguments far down the chain. any idea?
Hi. I have tr
function which I then call wherever I want in the app:
(ns
(:require
[taoensso.tempura :as tempura]
[app.messages :refer [messages]]))
(defn- missing-resource-fn [{:keys [resource-ids]}]
(subs (str (first resource-ids)) 1))
(defn tr [& rest]
(apply tempura/tr
{:dict messages
:missing-resource-fn missing-resource-fn}
[:cs]
rest))
so in some component, I just call it like:
[ui/DialogContentText
(tr [:app.list.file.delete_confirmation/body]]
Hi, I struggled with the same question a while ago and ended up passing args down the chain
Not with Tempura though but https://github.com/tonsky/tongue which is very similar
However, I’m pretty new to re-frame and not yet very confident deciding when to use subscriptions vs passing props/args. In the beginning I tended to pass more args down the chain but now I feel that subscriptions are actually really nice in many situations. It depends.
hm.. giving it a little thought, maybe subscriptions aren't so bad? i mean impurity would only be a problem for me if it makes things less predictable / testable, but there is this approach here: https://github.com/Day8/re-frame/blob/master/docs/Testing.md#view-functions---part-2b
of course it would be nicer to do avoid impurity, but i18n will be necessarily sprinkled all over the app and i don't see an easy way out at the moment. if anybody has any more input i'm all ears
@arne-clojurians There were some efforts to create a pure version of re-frame
. E.g. https://github.com/binaryage/pure-frame
Some discussion is here https://github.com/Day8/re-frame/issues/137. There's also interesting info available by the links in that issue.
@p-himik @arne-clojurians and further (incomplete) exploration here https://github.com/Day8/re-frame/blob/master/docs/EPs/002-ReframeInstances.md
I have a table that is populated with data from Firestore. What I am trying to achieve is to have an edit button after each row, and when that is clicked have the fields in that row become editable text fields. What I am struggling with is getting that change to occur. Here is what I have so far:
(defn device-row [device]
(let [edit-mode? (reagent/atom false)
text-val (reagent/atom "")
status (reagent/atom nil)
status-tooltip (reagent/atom "")
]
[h-box
:class "rc-div-table-row"
:width "1060px"
:gap "15px"
:children [ ;;use to bump columns to the right
[box :size "initial" :width "1px" :child [title :label ""]]
[box :size "initial" :width "160px"
:child [p (get device "deviceName" "Default Value")]]
[box :size "initial" :width "100px"
:child [p (get device "owned" "Default Value")]]
[gap :size "17px"] ;; required to accomodate Owned? sorting buttons
[box :size "initial" :width "115px"
:child [p (get device "ownedBy" "Not Owned")]]
[box :size "initial" :width "185px"
:child [p (get device "initialStateLink" "Default Value")]]
[box :size "initial" :width "180px"
:child [p (get device "number" "Default Value")]]
[box :size "initial" :width "170px"
:child [p (get device "assignedDate" "Default Value")]]
[md-icon-button :md-icon-name "zmdi zmdi-edit"
:on-click #(do (reset! edit-mode? true)
(println "Trying to turn on edit mode"))];;;;;;add on-click
(when @edit-mode? [h-box
:children [[box :size "initial" :width "1px" :child [title :label ""]]
[box :size "initial" :width "160px"
:child [p (get device "deviceName" "Default Value")]]
[input-text
:model text-val
:on-change #(reset! text-val %)
:width "100px"
:height "20px"
:placeholder "Hello"
, meet with rachel ]
[gap :size "17px"] ;; required to accomodate Owned? sorting buttons
[box :size "initial" :width "115px"
:child [p (get device "ownedBy" "Not Owned")]]
[box :size "initial" :width "185px"
:child [p (get device "initialStateLink" "Default Value")]]
[box :size "initial" :width "180px"
:child [p (get device "number" "Default Value")]]
[box :size "initial" :width "170px"
:child [p (get device "assignedDate" "Default Value")]]
[md-icon-button :md-icon-name "zmdi zmdi-edit"
:on-click #(reset! edit-mode? false)]
]])]]))
What’s the behavior you’re seeing now, and how is that different from what you want to see?
@aaron993 You have atoms defined inside the Form-1 component, so they're recreated each time the component is rerendered. You either have to create a Form-2 component, or use subscriptions.
Good question: When clicking the edit button, the loggin "Trying to turn on edit mode" works, but nothing new renders
@aaronmmartz that's a Form-1 component. So when it rerenders, the reagent/atoms will be destroyed and recreated losing whatever previous value they had.
In particular https://github.com/reagent-project/reagent/blob/master/doc/CreatingReagentComponents.md
Hmm so I’ve got a case where a user fills out a short form, at the end of it, I want to set focus back to a specific field so additional entry can take place. to be tidy I think I’d use reg-event-fx
and set focus via a side effect?
That should work as long as the field you’re trying to set the focus on does not re-render in the meantime
Otherwise you can set a value in the app-db, and have your target field call focus
on itself during :component-did-update
Yeah I considered that, too, it’s sometimes a little tedious to drive every ounce of behavior by something in the app-db. Is there a time and place for calling something like a focus-on
function that sets the focus directly?
Is that a cardinal sin? I can find a place in my app-db model to drive it by data. I’m sorta just trying to calibrate my mental model here on which it’s OK to be explicit about setting behavior in the DOM vs “data all the things”
I’d try it without the app-db first, but just be aware of the “lost focus because component re-rendered” gotcha. If it re-renders and drops focus as soon as you set it, then go ahead and set the flag in the db (or wherever)
Hi,
I'm using the :active-panel
pattern from the docs, but after the change to panel2 it shows for millis and then the app does a full reload (browser console says "Navigated to http://localhost:10555/" which is the same message I would get after refreshing browser). If I swap!
to toggle the panels via the repl the views toggle perfectly. I even created a new chestnut and re-implemented activepanel and do not get the page reload.
Obviously, I have introduced a subtle difference.
I am just looking for a technique to find the trigger that is forcing the reload.
@gmercer Is this in an :on-click
handler? If so you might try returning false
from the handler explicitly, or calling .preventDefault
on the event that gets passed in to your handler.
which meets the criteria on this page https://github.com/Day8/re-frame/wiki/Beware-Returning-False
@U06CM8C3V
changing
from :on-click #(dispatch [:signup/navigate])
to :on-click (fn [e] (.preventDefault e) (dispatch [:signup/navigate]))
stopped the reload
@U06CM8C3V but why? .. do I need to do this everywhere ?
The default action on anchors is to change the page location, triggering a reload
If you use something besides an a
tag, you won’t have this problem.
button as in a [:button ...]
tag?
I think I might chuck that handler function from the doco above and sprinkle in a preventDefault
[:button.btn.btn-info.btn-xs.offset-md-1 {:on-click (fn [e] (.preventDefault e) (dispatch [:signup/navigate]))} "Create one now!"]
I would probably make a utility function somewhere like
(defn mk-click-handler [event-vec]
(fn [e]
(.preventDefault e)
(dispatch event-vec)))
then do
[:button {:on-click (mk-click-handler [:signup/navigate])}]
so you get that handy one-line :on-click
line, and the .preventDefault
built in
`(defmacro handler-fn
([& body]
(fn [~'event] ~@body nil)))
is the suggestion on the page, so
`(defmacro handler-fn
([& body]
(fn [~'event] (.preventDefault event) ~@body nil)))
would be the upgrade ;)
Hmm, a macro seems like overkill to me, unless you’re going to need to do something really heavy duty in the body
Personally I think an :on-click
handler should do the bare minimum, dispatch an event and maybe call .preventDefault if you need it.
and an ordinary fn
is sufficient for that.
Is your button inside a [:form ...]
tag?
That’s why then. The browser is saying, “Oh, you’re clicking a button in the form, you must want to submit the form.”
That’s my guess anyway.
When you click the button, the browser “submits” the form data, expecting the server to send back a result page
In this case, the server sends back the same page, so you get the reload effect.
If you’re handling sending the form data to the server directly, via AJAX calls, then you don’t need the [:form...]
tag, and getting rid of it will remove that default behavior
:thumbsup:
Hi all, I messaged earlier about about a problem and was directed to use a Form-2 instead of a Form-1. I am trying that, and now that I added a fn[] to my function, the view is not being rendered. Any suggestions? Is this a common problem?
`
(defn device-row [device]
(let [edit-mode? (reagent/atom false)
text-val (reagent/atom "")
status (reagent/atom nil)
status-tooltip (reagent/atom "")
]
(fn [device]
[h-box
:class "rc-div-table-row"
:width "1060px"
:gap "15px"
:children [ ;;use to bump columns to the right
[box :size "initial" :width "1px" :child [title :label ""]]
[box :size "initial" :width "160px"
:child [p (get device "deviceName" "Default Value")]]
[box :size "initial" :width "100px"
:child [p (get device "owned" "Default Value")]]
[gap :size "17px"] ;; required to accomodate Owned? sorting buttons
[box :size "initial" :width "115px"
:child [p (get device "ownedBy" "Not Owned")]]
[box :size "initial" :width "185px"
:child [p (get device "initialStateLink" "Default Value")]]
[box :size "initial" :width "180px"
:child [p (get device "number" "Default Value")]]
[box :size "initial" :width "170px"
:child [p (get device "assignedDate" "Default Value")]]
[md-icon-button :md-icon-name "zmdi zmdi-edit"
:on-click #(do (reset! edit-mode? true)
(println "Trying to turn on edit mode"))]
(when @edit-mode? [h-box
:children [ ;;use to bump columns to the right
[box :size "initial" :width "1px" :child [title :label ""]]
[box :size "initial" :width "160px"
:child [p (get device "deviceName" "Default Value")]]
[box :size "initial" :width "100px"
:child [p (get device "owned" "Default Value")]]
[gap :size "17px"] ;; required to accomodate Owned? sorting buttons
[box :size "initial" :width "115px"
:child [p (get device "ownedBy" "Not Owned")]]
[box :size "initial" :width "185px"
:child [p (get device "initialStateLink" "Default Value")]]
[box :size "initial" :width "180px"
:child [p (get device "number" "Default Value")]]
[box :size "initial" :width "170px"
:child [p (get device "assignedDate" "Default Value")]]
[md-icon-button :md-icon-name "zmdi zmdi-edit"
:on-click #(do (reset! edit-mode? true)
(println "Trying to turn on edit mode"))]
]])]
])))