- bug with sources.source called several times. Use share or shareReplay(1) wisely - tricky to debug
- in the case of drivers specially think long about using share or shareReplay..., but there should be one, as there is no guarantee a driver source won't be called several times...
- trace every single f.. sink
- it is relatively easy to see data flow, but when the problem is that the data does not flow it gets hairy, as this is often due to subscription not happening, and there is no way to trace subscription (unsubscription is possible to trace somehow due to
finally
). - NOTE : onSubscribe possible. cf. https://stackoverflow.com/questions/41883339/observable-onsubscribe-equivalent-in-rxjs
- it is relatively easy to see data flow, but when the problem is that the data does not flow it gets hairy, as this is often due to subscription not happening, and there is no way to trace subscription (unsubscription is possible to trace somehow due to
- in the example aplication, interesting how important to think that Switch is disconnection a component sinks, so here the router never gets to emit a route
- DO NOT FORGET THE SINKS in SINKNAMES (3x)
- adding a sink in
run
but not in settings.sinkNames will lead to hard to trace bugs!! - Look if I can issue a warning if a component emits a sink not part of sinknames
- adding a sink in
- WHEN USING DIRECT DOM READ, dont do it twice at two different times, values will be different,
and some time error will be launched
- case where first time dom element exists and value read ok, then removed, then read again : boom! read ONCE for all the time it is needed! no multicast with direct read!
- if a imports from b, and b imports from a, then it could happen that some imports are undefined at the time of evaluation. AVOID THAT SITUATION!!
// NOTE : Here we present a case of optimistic update for local state. This is fundamented in // the fact that local state update cannot fail, that is the key difference vs. remotely persisted // state. // So: user clicks on a tab from the tab group, that updates the local state for the tab group, // the ui updates, and then that local state is translated into a state update action for // the corresponding app level piece of state (i.e. task filter) // Note again that we have two semantically distinct concepts, which belong to two distinct // layers : // - which button of the button group is clicked // - that is UI state corresponding to the UI layer // - that state is not visible out of the specific portion of the UI it is declared in // - what is the current filter for the tasks // - that is non-persisted local app state corresponding to the domain layer // - that is visible by the whole app
cf. bug tags in the code. How to
-
bug in snabbdom
- when using keys, weird, dont have the repro anymore
- when having a #selector after a .selector
-
desync. snabbdom vs. actual DOM
- in case of input for instance (a more elegant solution would be to have an input component??) ALWAYS USE A KEY FOR INPUTS
-
firebase
- cannot set a property to undefined
makeDomainActionDriver: an error occured Error: Firebase.set failed: First argument contains undefined in property 'projects.-KtxLeGQmuU5ViOm8XQO.tasks.4.done'
- cannot set a property to null : it means remove
- cannot set a property to undefined
- are the constant part of the rendering function
- holes are for the variable part
- initialize all source of truth before starting the app
- for instance filter tasks starts with ALL, so put 'all' there BEFORE starting
- initialize all dependent sources of state when building state/behaviour
- for instance task button group state is connected to in-memory task-button-state/filter.
- the connection is made through stream, EXCEPT for
- the initial value where the connection has to be done by hand
- OR linked to the current value of the source of truth
- the connection is made through stream, EXCEPT for
- for instance task button group state is connected to in-memory task-button-state/filter.
- order of drivers not predictable
- problem when one driver must execute after another one
- with using strings for identifiers of sources or else, it is easy to make a mistake that is not captured by js (but maybe ts would)
- don't replace links with . if it goes to classes, use _
- also the equality used to recognize the route is obviously weak (indexOf) would be better to recognize the full route as a prefix oh well
- bugs can update wrongly the database, which can create other bugs because one forgets to put back the database
- be careful with Ramda that always will evaluate only once its operand. i.e. always(x++) will only evalue x++ once, so event$.map(always(x++)) will not work, not evaluated for each event
- InSlot has to be right after where the slot is asked, any other component combinator is wrapping over previous vnodes with a div without slot
- best way to equiv. templates is to isolate components and then use Container and slots, and pass the slots as children
- this is equiv. to container = f(state_container, children) where children = sum:f(state_child, events)
- or container = f(children), and children = f(state, events)
- and dom from children will be assembled to according to slot logic, and sinks reduced according to default merge
-
- state injected in the form of InjectSources
-
- for standard components, adapters in the form of InjectSettings, or InjectSourcesAndSettings if the standard component has also requirement (signature) about sources too
-
- slot injected via InSlot
- Basically child = standardComponent(Settings), i.e. is a specialization of the standard component
-
- when refactoring state stream names, be careful to refactor also the strings... for instance ForEach(from : 'stream name in string')
- incremental development which helps debugging
- we have a tree. At any moment we can replace a whole branch with dummy component, to isolate the part which is failing
- we don' have to do everything at the same time. Get the structure of the tree, and put dummy component anywhere we have pending
- using less levels of streams, as we make use of settings. This decreases the potential for error and unintended concurrency
- separation of variable and constants
- DOM is a function of state (the part that can change) and some fixed settings, which can be given at configuration time, or at run time.
- in other component frameworks, components are parameterized through props, which can either change dynamically (linked to the change of the state of the parent who is passing them) OR be constant. In this framework, we separate both
- i.e. we isolate the parameterization concern from the state dependency concern
- readability!!
- this isolates very nicely the structure of the app, allowing to understand it more quickly but also to nail down faster source of issues
- maintainability!!
- because concerns are isolated, it is easy to narrow down the modification targets, and discard/replace/change the implementation of such
- identify state$, event$, intent$ if any, and actions$
- make it so final sinks is one liner
- state$ should be the closest to where it is applied (cohesion)
- BUT when state has to be shared between components pass it at the closest common ancestor
with InjectSources
- in that case it should be in a separate file
- BUT when state has to be shared between components pass it at the closest common ancestor
with InjectSources
- ?? write ad-hoc combinators
- always have last the component definition as a function of other components
- helpers above or in a separate files if there are many
- or if there are used by several components, in which they go in separate file at common ancestor level
- helpers above or in a separate files if there are many
- construct the UI piece by piece
- side panel
- main panel
- three sections
- etc...
- i.e. do the structure first and then fill in the details
- what does the component do
- which state do I need
- write components so that they are small and take care of a minimum of concerns
- app shell
- initial state
- drivers, etc.
- configuration
- app sink names, etc.
- recursive breakdown
- write the target breakdown
- final formula
- mix of components, ad-hoc and general component combinators
- specifications of components
- final formula
- write the components
- identify component type from specs
- component?
- component combinator?
- component tree?
- children components?
- identify inputs from source
- identify outputs to sinks
- IF COMPONENT
- specify the reactive function as actions = f(events)
- identify the events
- identify the actions
- write the formula f for each sink
- identify state
- keep it local if possible
- keep it as close as possible to where it is used
- specify the reactive function as actions = f(events)
- IF COMPONENT COMBINATOR
- specify inputs
- contracts
- settings, sources, others
- children components (type, slot, etc.)
- component tree vs. children components
- contracts
- specify output
- specification how the children are reduced into sinks
- turn it into
m
and other combinators- choose one of the three
m
patterns - for each pattern, fill in the blank
- sinks reducing formula
- choose one of the three
- specify inputs
- test vs. specifications
- refactor
- inject state above the component when shared by sibling, i.e. up to common ancestor
- identify component type from specs
- write the target breakdown