-
-
Notifications
You must be signed in to change notification settings - Fork 730
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Admin, Orders list: when capturing an order, fix an issue that makes the tooltip invisible (+ errors in the console) #11577
Admin, Orders list: when capturing an order, fix an issue that makes the tooltip invisible (+ errors in the console) #11577
Conversation
5e81d61
to
986be46
Compare
app/reflexes/admin/orders_reflex.rb
Outdated
morph dom_id(@order), render(partial: "spree/admin/orders/table_row", | ||
locals: { order: @order.reload, success: true }) | ||
cable_ready.inner_html( | ||
selector: dom_id(@order), | ||
html: render(partial: "spree/admin/orders/table_row", | ||
locals: { order: @order.reload, success: true }) | ||
).broadcast | ||
|
||
morph :nothing |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's an interesting observation. It sounds like a bug in StimulusReflex to me. I don't like this new pattern but if it works then it's good.
I tried to find any information about why this is happening. And I found an interesting section in the docs:
inner_html
completely wipes out any Stimulus controllers present in the replaced DOM hierarchy and doesn't respect the data-reflex-permanent attribute. How can we adapt this so that both morph operations are performed by the morphdom library?
https://docs.stimulusreflex.com/guide/morph-modes.html#real-world-example-pagy-refactoring
So there's a difference between inner_html and morphdom and morphdom seems to keep as much as possible, therefore not disconnecting and reconnecting controllers. I think it tries to update the existing tooltip elements without replacing them.
Maybe a problem of the tooltip controller is that it renders the tooltip on connect and is not aware of data attribute changes after that.
Another thing I found is:
You cannot change the attributes of your morph target.
https://docs.stimulusreflex.com/appendices/troubleshooting.html#you-cannot-change-the-attributes-of-your-morph-target
So either solution means that the tr
attribute for order state never updates. This is probably a pre-existing bug. But looking at morphdom it seems to be able to update attributes on the target. So, I'm not sure where that limitation comes from or if the warning is only there in case StimulusReflex decides to replace inner_html only (which it seems to do in some cases).
Maybe we should test if the state change is a problem and then we need to use outer_html. Otherwise we are good.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this describe quite neatly the issue and the recommended solution here :
stimulusreflex/stimulus_reflex#314 (comment)
I am wondering if we should rewrite the tooltip stimulus controller to use stimulus-reflex:after
callback as described above, it seems to me to be the better approach. It would prevent the issue in any other page that uses the tooltip controller.As it stands, the onus is on the reflex to do the correct render, and I feel like it's not its job to have to worry about any stimulus controller it's not interacting directly with.
That said the stimulus docs do describe the pattern used here : https://docs.stimulusreflex.com/guide/cableready.html#using-cableready-inside-a-reflex-action
What do you guys think ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Even after adding this.element.addEventListener("stimulus-reflex:after", this.insertToolTipMarkup);
into the connect()
method, I get a
Uncaught TypeError: this.elementTarget is undefined
insertToolTipMarkup tooltip_controller.js:78
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for the links, Gaetan, very interesting. I didn't know that you could also queue cable_ready events without broadcasting them immediately and that they get then broadcasted after the StimulusReflex morph.
I guess the idealistic solution for the tooltip is to render the tooltip server-side and include it in the morphing partial. Hm, that's actually already done in other places. Do we just need to get rid of insertToolTipMarkup
and replace it with a partial?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe original intend was to use a partial : b76d904 but then Matt did some refactoring here : d64a9af which led to the current behaviour.
I had a go at using stimulus-reflex
callbacks, I can trigger a re render but for some reason it's not finding the data attributes linked to the element with the tooltip controller 😕
From the linked I shared above :
In the context of SR (Stimulus Reflex), the expectation is that you would specify the UI you want to exist in the browser on the server. There's no need for a render stage because the markup is already perfect before it is sent down the wire. There is no build stage, no rehydration. We're not taking over the task of redrawing the UI from the browser.
I'll look at moving back to a partial.
a9c15b4
to
501d182
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice, this is great.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good! 👍 I just have a query about the need for sanitize
.
%button{class: button_class, "data-reflex": button_reflex, "data-id": reflex_data_id, "data-tooltip-target": "element" } | ||
.tooltip-container | ||
.tooltip{"data-tooltip-target": "tooltip"} | ||
= sanitize tooltip_text |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it not be sanitised automatically?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe rails escape html automatically but does not sanitize unless you do it manually.
To be sure that every stimulus controllers attached to the DOM that is replaced is will connected. What I saw is that when using `morph`, the stimulus controller (specially `tooltip_controller`) were not attached to the DOM (and not disconnected as well) that triggered some errors in the console. Adds test case for payment capture thanks to @filipefurtad0
The issue is with with the stimilus tooltip controller, it add some element to the DOM which create issue when it's modified by StimulusReflex. See here for a more detailed explanation: stimulusreflex/stimulus_reflex#314 (comment)
It's not great to have Stimulus controller rendering markup on `connect` Stimulus is intended to add behavior to existing markup. Plus add some documentation
501d182
to
d2952d4
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Argh, looking back I'm sorry I stalled this while you were on leave.
Thanks for responding. Looking at this again, I can see tooltip_text
comes from the locale, so it shouldn't require any further sanitising.
But I can also see that it was copy/pasted from existing code, so all good 👍
I don't want to stall it anymore, so will move to test ready when specs pass!
Hi @rioug, Before staging
Peek.2023-10-19.21-11.mp4After staging
Peek.2023-10-18.11-20.mp4SummaryI will merge this one because it's fixing two out of three issues. Thanks again! 🎉 |
What? Why?
We should usecable_ready.inner_html
instead ofmorph
to be sure that every stimulus controllers attached to the DOM that is replaced is will connected. Actually, I don't know why, this is empirical. What I saw is that when usingmorph
, the stimulus controller (speciallytooltip_controller
) were not attached to the DOM (and not disconnected as well) that triggered some errors in the console.This explains nicely what is happening here: stimulusreflex/stimulus_reflex#314 (comment)
The tooltip controller was adding some html element to the page on connect, which doesn't play well with StimulusReflex. So we refactored the controller and use a partial to populate tooltip content.
What should we test?
/admin/orders
Release notes
Changelog Category (reviewers may add a label for the release notes):
The title of the pull request will be included in the release notes.