Fork me on GitHub
#fulcro
<
2017-12-19
>
bbktsk00:12:22

I’m looking at targeted data loads in http://fulcro.fulcrologic.com/guide.html#!/fulcro_devguide.H05_Basic_Loading and I do not understand the part “If you’ve followed our earlier recommendations, then your applications UI is normalized as well,“. What earlier recommendations this refers to? I’ve seen docs about normalizing data, but not avout UI. Have I missed that? And does this apply when app uses UI routing, esp. with multiple routers? I somehow do not see how a component can use targeted load without precisely knowing where in the UI tree it it is located…

tony.kay00:12:47

@bbktsk So, you read about normalization, so you should know that every possible thing in your database should be in a table, especially anything from a server or that changes.

tony.kay01:12:49

I.e. all of your components have an ident, which means they are at an assoc-in path like [:table/by-id 3], and any field is just one level deeper [:person/by-id 3 :person/name]

tony.kay01:12:39

So, targeting never needs to go deeper than that. The auto-normalization takes care of the graph of incoming data (and puts it in tables), so the top-level edge can either go into root, or go into a field.

tony.kay01:12:53

(or both, or some combination of fields).

tony.kay01:12:20

Channel: Fulcro 2.0.0-RC3 on clojars. Not much changed in code, but the Getting Started Guide was updated.

tony.kay01:12:33

as were the github files associated with it.

tony.kay01:12:01

@bbktsk UI routing is a separate topic from load targeting. They are unrelated. Since everything is normalized the structure of your “pure UI” (e.g. routing, layout, etc) isn’t even involved. The good part: You’re almost there! This is the crux and beauty of the system. Keep looking at it. It’s quite simple…just different enough that it takes most people a few tries to grok it.

tony.kay01:12:36

If I want to load a list of people into person list 4, I might do this: (df/load :friends Person {:target [:person-list/by-id 4 :person-list/people]) This assumes you’ve got the UI components normalizing into the tables implied. See the Getting Started guide for this exact example in detail. Also, try the YouTube video series. Some of the alternate materials might make it more clear. I happen to think the video guide is the easiest to get started with.

LL02:12:37

Need help, about semantic-ui-wrapper, I want to use dimmer, the code in http://react.semantic-ui.com:

<Dimmer.Dimmable as={Segment} blurring dimmed={active}>
          <Dimmer active={active} onClickOutside={this.handleHide} />

          <p>
            <Image src='/assets/images/wireframe/short-paragraph.png' />
          </p>
          <p>
            <Image src='/assets/images/wireframe/short-paragraph.png' />
          </p>
        </Dimmer.Dimmable>
my code:
(f/ui-dimmer-dimmable #js {:as (f/ui-segment) :blurring true :dimmed dimmed}
      (f/ui-dimmer #js {:active dimmed
                        :onClickOutside #(prim/transact! this `[(m/dimmer-dimmed {})])}
        (dom/p nil "aaaaaaaaaaaaa")
        (dom/p nil "bbbbbbbbbbbbbb")))
but got errors in the console in Chrome:
Warning: Failed prop type: Invalid prop `as` supplied to `a`.
    in a (created by qsweb.A0-UI-Components/Edit-Dimmer)
    in qsweb.A0-UI-Components/Edit-Dimmer
the :as (f/ui-segment) case the errors.

tony.kay02:12:06

@tpliliang I’ll accept PRs on that, but the wrappers just push it through to the underlying library.

tony.kay02:12:54

perhaps it should be :as f/ui-segment?

tony.kay02:12:22

or is “as” only accept a string for the kind of regular DOm element?

tony.kay02:12:24

read their docs

tony.kay02:12:53

their docs seem to imply that it should be a class? Perhaps Segment is the right thing? Then again, perhaps a factory function, which would be f/ui-segment without the parens, as I said above.

LL03:12:21

Thanks, and I tried, but neither "Segment" nor f/ui-segment works:sob:

liamd04:12:48

are the videos up to date with 2.0?

liamd04:12:19

ah it says it right there in the description

LL05:12:36

@tony.kay :as (goog.object/getValueByKeys js/semanticUIReact "Segment"), this works.

LL05:12:23

:as (fulcrologic.semantic-ui.factories/get-sui "Segment") works too.

levitanong06:12:32

Hi all, is there an idiomatic way to handle network requests that should be debounced? For example, the google autocomplete API returns suggestions for places as you type. This should probably be in a remote, but it probably shouldn’t trigger with every keystroke.

LL11:12:56

@tony.kay Need help, I want to test websockets client in devcards. So I use:

(defcard-fulcro card1
  #_(df/load app :all-ids All-Ids)
  All-Ids
  {}
  {:inspect-data true
   :fulcro       {:networking       cs-net
                  :started-callback (fn [app]
                                      (wn/install-push-handlers cs-net app)
                                      (df/load app :all-ids All-Ids))}}
  )
but the client db cannot be normalized, but if I use ui component's initial-state instead, the normalizing work. I tried js/console.log in ui component and found if use websockets client, there are 4 times console prints, but only print 1 time when using initial-state. I guess there may be a few different stages for ui render and db normalize, At the db normalize stage, the app cannot get the query result from remote server for using :started-callback. How can I do? and How to print the normalized db in the console in the Chorme?

tony.kay16:12:09

@tpliliang Hm. sounds like we need symbols for those?

tony.kay16:12:22

On your second thing: what do you mean client DB cannot be normalized? Normalization is an action that happens at specific times, and is always tied to UI queries and idents.

tony.kay16:12:04

If you want to see the normalized db…you’ve already got it in the card (inspect true). You can also use Fulcro Inspect. From the REPL you can access it from the app saved in an atom. The defcard-fulcro docs tell you how to find the card’s app. The GettingStarted guide shows examples of dumping the DB from a REPL.

tony.kay16:12:09

Regarding normalization: Cards supply an atom, so if you put data in the card’s data map, it must start out hand normalized. If you leave it empty, the card uses initial state and auto-normalizes.

tony.kay16:12:04

The rules internally allow for hand-built databases or auto-normalized. The rules are: 1. If you supply no (or empty) state, then UI initial state is pulled (as a tree) and normalized into your db 2. If you supply a non-empty MAP, it will use the UI query to normalize it into an internal atom holding a map 3. If you supply an ATOM (which cards do, even though you write what you want in it as a map), then it takes it as the db state you want (no auto-normalization)

tony.kay16:12:42

That’s why all of my examples show that first map argument as empty. You can use it to make a database, but you almost always want UI initial state.

tony.kay19:12:56

ALL: New EAP build of Cursive just dropped!!!! This version contains direct support for Fulcro’s special bits (defsc, defmutation, etc.)!

tony.kay19:12:31

No more yellow highlighting around things like action sections of defmutation! It fully understands those macros and does proper indexing and refactoring on them now.

wilkerlucio21:12:06

does it? I did updated today and I didn't see the defmutation been recognized, had you tested that?

wilkerlucio21:12:49

I just wanna make sure, because if it is something is broken on my end

tony.kay19:12:56

I added this to #announcements with a thanks to Colin…react there to let him know we appreciate it!

tony.kay19:12:06

Fulcro 2.0.0-RC4-SNAPSHOT pushed to clojars. Has support for refs in fallbacks (as ::prim/ref in fallback parameters). See http://localhost:8081/demos.html#!/cards.server_error_handling/error-handling when running full-stack demos locally…or the source here: https://github.com/fulcrologic/fulcro/blob/d94ebf21ab5fe3e0a8e4f04edc27bfbf54d4297b/src/demos/cards/server_error_handling.cljc#L35

piotrek21:12:33

@tony.kay I am going through the forms-2.0 sample card with phone book. What I have noticed is that the edits done in the form modifies the entity data directly in the :phone/by-id which is not optimal if the form for editing the entity and a table displaying the entity are visible at the same time (which might be quite often). Is it an intended design goal or is it just a simplistic demo that doesn’t work on a copy of the entity data?

tony.kay22:12:34

@piotrek It is the intended design to work on the active data, but keep a checkpoint for reverting. This is because nested forms won’t work right otherwise.

tony.kay22:12:11

and it doesn’t really matter…you have two copies of data. old and new. preference aside, it isn’t appreciably different

tony.kay22:12:01

Oh, you’re saying you don’t want the user to see the changes “in place” in the sibling UI

piotrek22:12:14

It does however forces you to design the UI in a specific way (e.g. don’t show the list of entities and the entity edit form so they are both visible)

piotrek22:12:59

I just spliced the (if (contains? phone ::phone-number) ...) sexp so I have both the phone numbers list and the edit form both visible

piotrek22:12:20

(that’s the UI layout I need in my app)

tony.kay22:12:48

Choices: - Don’t use forms.cljc. Roll your own. It isn’t that hard. It’s just data. But, you will have to invent your own caching scheme - Change your UI design - Adopt the philosophy that things are optimistic, and that the change should be visible…Undo is right there, “save to server” is deferred

tony.kay22:12:27

The nesting of entities in forms is what dictated the choice. I want you to be able to render the form data. When you have sub-forms you need to be able to write generic mutations that can manipualte state (e.g. add phone number), and you want to write the render code identically to what you’re used to. Try to implement that when the “edited data” is all hidden, and you can’t

tony.kay22:12:46

I would personally design the UI to not be showing the “table” version of it while editing it.

piotrek22:12:05

I will probably try this first - thank you

piotrek22:12:28

BTW, do you have plans to include forms 2.0 in the mainstream builds?

piotrek22:12:04

(or at least rebase forms-2.0 branch on the latest changes so it’s possible to get an up to date local build with forms 2.0?)

tony.kay22:12:55

eventually. Did you play with it at all? It would be simple to pull the code into your own build and try it out.

piotrek22:12:55

I am just playing it directly in the fulcro repo right now but I was thinking about trying it in my simple app where I use the latest 2.0 RC version

tony.kay22:12:15

oh, I was talking about jsut copy-pasting the code into your app

piotrek22:12:30

ok - I will try this then - is it just fulcro.ui.form-state?

piotrek22:12:11

Yeah - it looks it’s just this ns

tony.kay22:12:16

yeah, I think it is just one file

tony.kay22:12:27

If you get it all tuned up, maybe that will get it into production faster in Fulcro 🙂

tony.kay22:12:50

It needs to be exercised in real scenarios, so I know what holes it has

tony.kay22:12:00

that’s where I have not had time

piotrek22:12:14

I wouldn’t count on it - I am a very new to all this so I am doing only some basic stuff for now 🙂

tony.kay22:12:27

ok…worth encouraging 😜

piotrek22:12:35

Nice try though 😉

OliverM22:12:01

FWIW I've been trying to come to grips with Fulcro specifically for it's application to a traditional CRUD web project

OliverM22:12:42

I'm translating a clojure REST API & web 1.0 application to an entirely new implementation using Datomic & an SPA using Fulcro

OliverM22:12:03

and the Fulcro forms support is what intrigued me the most

OliverM22:12:55

As an aside, I've almost 100% agreed with everything Fulcro does except (and this is going to seem super finicky) calling the fulcro base namespace 'prim' - it's too much like the English word 'prim' and I would be much happier with an acronym like 'fcp' or similar in the examples (which is what I've been using for my own code in any case)

OliverM22:12:26

prim/sc sounds like a spinster aunt being super critical of my code 😂

tony.kay22:12:39

Well, I love that you’re as picky about names as I am 😜

tony.kay22:12:25

how has your work with forms gone?

OliverM22:12:05

I've wound up doing my own customised stuff as I can make more assumptions for my use case than what's in the fulcro mainline now

OliverM22:12:35

TBH the conventions fulcro brings in terms of data normalisation are far, far more powerful than the forms stuff IMHO

OliverM22:12:10

So, loving fulcro for that reason, though the forms stuff as is (while it was the initial hook) isn't why I'm sticking with the learning curve etc

tony.kay22:12:44

Actually that isn’t an unexpected answer. It’s why I’m wanting to simplify the forms support to just the core state management helpers

OliverM22:12:23

yep. I've a different approach to CSS too (I'm a big fan of tachyons and atomic CSS) so I've been combining that with my own form needs to good effect

OliverM22:12:07

and routing, I think there's a good story to tell there too, though it's hard to dissociate what I'm doing with what a general case might look like

OliverM22:12:53

but arriving at the underlying principles of form data gathering, submission & display would be a great outcome

OliverM22:12:38

oh hey, if you were available for code review that would be a service I could totally pay you to do btw

tony.kay22:12:30

@oliver.mooney I do Fulcro consulting through http://www.fulcrologic.com. Be glad to help.