Proposal: Add Skin.install() method
Kevin Rushforth
kevin.rushforth at oracle.com
Fri Jul 22 19:15:23 UTC 2022
I think the Skin.install proposal is a good one; it looks ready to move
forward. I like the current direction that Andy is proposing because it
maximizes backward compatibility, while solving the primary set of
problems identified with having the constructor be the only place a Skin
can install listeners, handlers, and the like.
-- Kevin
On 7/22/2022 8:58 AM, Andy Goryachev wrote:
>
> > control.installSkin(MySkin::new);
>
> This is an interesting idea. Control.installSkin(Function<Control,Skin>).
>
> One of the requirements we ought to consider is maximizing the
> backward compatibility. If we were to add a new installSkin method it
> would not solve the problem with the old method, and replacing
> setSkin(Skin) with installSkin() would break compatibility with the
> existing code.
>
> There is one more reason to allow for creation of a skin outside of
> setSkin() - configuration. Imagine customer skins require
> configuration, in which case the sequence of events looks like this
>
> instantiation -> configuration -> uninstall old skin -> install new skin.
>
> with installSkin(MySkin::new) such a model will be impossible.
>
> Thank you
>
> -andy
>
> *From: *openjfx-dev <openjfx-dev-retn at openjdk.org> on behalf of John
> Hendrikx <john.hendrikx at gmail.com>
> *Date: *Thursday, 2022/07/21 at 15:49
> *To: *openjfx-dev at openjdk.org <openjfx-dev at openjdk.org>
> *Subject: *Re: Proposal: Add Skin.install() method
>
> Hi Andy,
>
> Was a single step install process considered, something like:
>
> control.installSkin(MySkin::new);
>
> This would also make it much more clear that Skins are single use
> only, where the API currently has to bend over backwards to make that
> clear and enforce it.
>
> Other than that, I think your suggestion would be a definite
> improvement over the current situation. Something never felt quite
> right about how skins where set up and attached to controls, it felt
> fragile and cumbersome -- possibly as the result of relying on a
> writable property as the means to install a skin.
>
> --John
>
> On 20/07/2022 23:39, Andy Goryachev wrote:
>
> Hi,
>
> I'd like to propose an API change in Skin interface (details
> below). Your feedback will be greatly appreciated!
>
> Thank you,
>
> -andy
>
> Summary
>
> -------
>
> Introduce a new Skin.install() method with an empty default
> implementation. Modify Control.setSkin(Skin) implementation to
> invoke install() on the new skin after the old skin has been
> removed with dispose().
>
> Problem
>
> -------
>
> Presently, switching skins is a two-step process: first, a new
> skin is constructed against the target Control instance, and is
> attached (i.s. listeners added, child nodes added) to that
> instance in the constructor. Then, Control.setSkin() is invoked
> with a new skin - and inside, the old skin is detached via its
> dispose() method.
>
> This creates two problems:
>
> 1. if the new skin instance is discarded before setSkin(), it
> remains attached, leaving the control in a weird state with two
> skins attached, causing memory leaks and performance degradation.
>
> 2. if, in addition to adding listeners and child nodes, the skin
> sets a property, such as an event listener, or a handler, it
> overwrites the current value irreversibly. As a result, either
> the old skin would not be able to cleanly remove itself, or the
> new skin would not be able to set the new values, as it does not
> know whether it should overwrite or keep a handler installed
> earlier (possibly by design). Unsurprisingly, this also might
> cause memory leaks.
>
> We can see the damage caused by looking at JDK-8241364
> <https://bugs.openjdk.org/browse/JDK-8241364> ☂/Cleanup skin
> implementations to allow switching/, which refers a number of bugs:
>
> JDK-8245145 Spinner: throws IllegalArgumentException when
> replacing skin
>
> JDK-8245303 InputMap: memory leak due to incomplete cleanup on
> remove mapping
>
> JDK-8268877 TextInputControlSkin: incorrect inputMethod event
> handler after switching skin
>
> JDK-8236840 Memory leak when switching ButtonSkin
>
> JDK-8240506 TextFieldSkin/Behavior: misbehavior on switching skin
>
> JDK-8242621 TabPane: Memory leak when switching skin
>
> JDK-8244657 ChoiceBox/ToolBarSkin: misbehavior on switching skin
>
> JDK-8245282 Button/Combo Behavior: memory leak on dispose
>
> JDK-8246195 ListViewSkin/Behavior: misbehavior on switching skin
>
> JDK-8246202 ChoiceBoxSkin: misbehavior on switching skin, part 2
>
> JDK-8246745 ListCell/Skin: misbehavior on switching skin
>
> JDK-8247576 Labeled/SkinBase: misbehavior on switching skin
>
> JDK-8253634 TreeCell/Skin: misbehavior on switching skin
>
> JDK-8256821 TreeViewSkin/Behavior: misbehavior on switching skin
>
> JDK-8269081 Tree/ListViewSkin: must remove flow on dispose
>
> JDK-8273071 SeparatorSkin: must remove child on dispose
>
> JDK-8274061 Tree-/TableRowSkin: misbehavior on switching skin
>
> JDK-8244419 TextAreaSkin: throws UnsupportedOperation on dispose
>
> JDK-8244531 Tests: add support to identify recurring issues with
> controls et al
>
> Solution
>
> --------
>
> This problem does not exist in e.g. Swing because the steps of
> instantiation, uninstalling the old ComponentUI ("skin"), and
> installing a new skin are cleanly separated. ComponentUI
> constructor does not alter the component itself,
> ComponentUI.uninstallUI(JComponent) cleanly removes the old skin,
> ComponentUI.installUI(JComponent) installs the new skin. We should
> follow the same model in javafx.
>
> Specifically, I'd like to propose the following changes:
>
> 1. Add Skin.install() with a default no-op implementation.
>
> 2. Clarify skin creation-attachment-detachment sequence in Skin
> and Skin.install() javadoc
>
> 3. Modify Control.setSkin(Skin) method (== invalidate listener in
> skin property) to call oldSkin.dispose() followed by newSkin.install()
>
> 4. Many existing skins that do not set properties in the
> corresponding control may remain unchanged. The skins that do,
> such as TextInputControlSkin (JDK-8268877), must be refactored to
> utilize the new install() method. I think the refactoring would
> simply move all the code that accesses its control instance away
> from the constructor to install().
>
> Impact Analysis
>
> -------------
>
> The changes should be fairly trivial. Only a subset of skins needs
> to be refactored, and the refactoring itself is trivial.
>
> The new API is backwards compatible with the existing code, the
> customer-developed skins can remain unchanged (thanks to default
> implementation). In case where customers could benefit from the
> new API, the change is trivial.
>
> The change will require CSR as it modifies a public API.
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/openjfx-dev/attachments/20220722/ce226215/attachment-0001.htm>
More information about the openjfx-dev
mailing list