| > It doesn’t at all allow composition. Of course it does. The most basic programming primitives of composition is functions. Want to compose hooks with these APIs? Call one hook from another. They’re just functions. Edit: or call one hook-using factory/HOC from another. > Honestly, it feels like you haven’t spent much time with hooks if this is your improvement It’s true, I do avoid using them. But I spend a lot of time reading the docs, as well as working with a large variety of different JSX APIs. > It's a great case of "making it look better but removing the valuable properties in the name of purity". I don’t know why that’s what you took from my pseudocode, since one of the examples maintained the exact same interface. Why would it be less valuable if hooks-consuming components are factories/HOCs? > I already explained fairly well why you want hooks to be inline in render, as they need to handle side effects, they need to compose with each other and react to changing values, and therefore they need to constantly run to check for updates. It’s 100% how they are used by everyone. And all of that is possible with the APIs I suggested. > I hope you find time to actually look at what hooks do more closely and find the power behind them. Please don’t condescend. I’m plenty familiar with the material. > You can try and hide that fact behind more syntax, but it doesn’t change the fact that your inner function you defined is depending on an outer closure that is stateful, breaking your law of no functions that act differently when called with the same values. It’s not about hiding statefulness, it’s about: 1. Isolating it: this is every letter in the SOLID principles. 2. Maintaining clearly identifiable interfaces that are clear in their boundaries and expected behavior. In Clojure, a language that's pure FP by default with clearly defined APIs for mutable escape hatches, this is achieved by convention with @ prefixes for state and ! suffixes for dereference. In JS/TS, this is achieved by convention by separating state init from post-init behavior (my factory/HOC example), or by passing/wrapping state (see common redux patterns). Both have corresponding types that make clear what’s going on. Shoving state and return values into the same space makes that impossible. |
And algebraic effects are a FP concept, go look at OCaml which has been pioneering them, so your appeal to FP purity doesn't even hold.