Skip to content
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

8349091: Charts: exception initializing in a background thread #1697

Closed
Changes from 24 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
40857f5
initial test
andy-goryachev-oracle Jan 29, 2025
4539e56
more tests
andy-goryachev-oracle Jan 30, 2025
aba8476
all controls
andy-goryachev-oracle Jan 30, 2025
e9259a7
thread count
andy-goryachev-oracle Jan 30, 2025
02f3272
pagination
andy-goryachev-oracle Jan 30, 2025
8786bab
cleanup
andy-goryachev-oracle Jan 30, 2025
d1356dc
review comments
andy-goryachev-oracle Jan 31, 2025
e0b3033
fast loop
andy-goryachev-oracle Jan 31, 2025
3fde513
better name
andy-goryachev-oracle Jan 31, 2025
a05d391
jiggler
andy-goryachev-oracle Feb 3, 2025
e25f6f6
skip tests
andy-goryachev-oracle Feb 3, 2025
597e39d
Merge remote-tracking branch 'origin/master' into 8348423.node.thread…
andy-goryachev-oracle Feb 3, 2025
30700a6
more jitter
andy-goryachev-oracle Feb 3, 2025
756bda5
chart tests only
andy-goryachev-oracle Feb 4, 2025
e9a6519
tests pass
andy-goryachev-oracle Feb 5, 2025
c74744a
cleanup
andy-goryachev-oracle Feb 5, 2025
125ee60
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Feb 6, 2025
4070d8c
whitespace
andy-goryachev-oracle Feb 6, 2025
4b6089e
Merge branch 'master' into 8349091.charts.thread.safety
andy-goryachev-oracle Feb 7, 2025
30deca1
Merge branch 'master' into 8349091.charts.thread.safety
andy-goryachev-oracle Feb 12, 2025
93fc7b9
enabled pie chart test
andy-goryachev-oracle Feb 12, 2025
e60c027
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Feb 20, 2025
757c640
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Feb 25, 2025
75d4b83
review comments
andy-goryachev-oracle Feb 25, 2025
4288d1d
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Feb 25, 2025
4976c58
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Feb 28, 2025
d212ffd
unnecessary
andy-goryachev-oracle Mar 3, 2025
643d4cf
Merge remote-tracking branch 'origin/master' into 8349091.charts.thre…
andy-goryachev-oracle Mar 4, 2025
fc40e9d
use subscription
andy-goryachev-oracle Mar 4, 2025
509c051
review comments
andy-goryachev-oracle Mar 4, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2010, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2010, 2025, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -25,24 +25,33 @@

package javafx.scene.chart;

import java.util.*;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import javafx.animation.FadeTransition;
import javafx.animation.Interpolator;
import javafx.animation.KeyFrame;
import javafx.animation.KeyValue;
import javafx.animation.Timeline;
import javafx.application.Platform;
import javafx.beans.NamedArg;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.DoubleProperty;
import javafx.beans.property.SimpleDoubleProperty;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.css.CssMetaData;
import javafx.css.Styleable;
import javafx.css.StyleableBooleanProperty;
import javafx.css.StyleableProperty;
import javafx.css.converter.BooleanConverter;
import javafx.scene.AccessibleRole;
import javafx.scene.Group;
import javafx.scene.Node;
import javafx.scene.chart.LineChart.SortingPolicy;
import javafx.scene.layout.StackPane;
import javafx.scene.shape.ClosePath;
import javafx.scene.shape.LineTo;
@@ -51,14 +60,7 @@
import javafx.scene.shape.PathElement;
import javafx.scene.shape.StrokeLineJoin;
import javafx.util.Duration;

import com.sun.javafx.charts.Legend.LegendItem;
import javafx.css.converter.BooleanConverter;
import javafx.css.CssMetaData;
import javafx.css.Styleable;
import javafx.css.StyleableBooleanProperty;
import javafx.css.StyleableProperty;
import javafx.scene.chart.LineChart.SortingPolicy;

/**
* AreaChart - Plots the area between the line that connects the data points and
@@ -541,7 +543,7 @@ private Node createSymbol(Series<X,Y> series, int seriesIndex, final Data<X,Y> i
symbol = new StackPane();
symbol.setAccessibleRole(AccessibleRole.TEXT);
symbol.setAccessibleRoleDescription("Point");
symbol.focusTraversableProperty().bind(Platform.accessibilityActiveProperty());
symbol.setFocusTraversable(isAccessibilityActive());
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like you are mixing the setting up of the listeners with setting the focus traversable of this symbol to the current state of accessibilityActive, which will be Platform.accessibilityActiveProperty for the FX app thread) and "false" (for background thread. As a result, this method, which is called once per symbol, will set up a new listener each time it is called.

You might consider refactoring this a bit to only ever set up the listeners when the object is constructed. Either that or ensure that you guard against repeatedly recreating the listener.

Comment on lines -544 to +546
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So, if I understand correctly, we can't directly do this bind:

symbol.focusTraversableProperty().bind(Platform.accessibilityActiveProperty())

Because on a background thread there may not yet be an initialized Platform? Or perhaps, Platform would need to initialize and we don't want to do that yet on a background thread?

Copy link
Contributor Author

@andy-goryachev-oracle andy-goryachev-oracle Mar 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The reason we should not be binding in this PR is because in part it's bad design(tm): we are creating a zillion of listeners (one per plot data point); a better solution would be to create one listener and use (already existing series) to toggle the data points.

The other reason is https://bugs.openjdk.org/browse/JDK-8351067 - the Platform::accessibilityActive property getter is not thread safe, but even if was, registering a listener with it may cause the change to come in before the node becomes a part of the scene graph, leading to unsynchronized multi-threaded access.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Was this solution considered:

ObservableValue<Boolean> partOfSceneGraph = this.sceneProperty()
  .flatMap(Scene::windowProperty)
  .flatMap(Window::showingProperty)
  .orElse(false);
  
symbol.focusTraversableProperty()
    .bind(Platform.accessibilityActiveProperty().when(partOfSceneGraph));

What the above code does is create a binding on the when statement (a listener is added on the when result). However, the when property will not add a listener on Platform.accessibilityActiveProperty() until partOfSceneGraph is true. As soon as it does though, a listener is installed on Platform.accessibilityActiveProperty() and its latest value is set to focusTraversableProperty via the bind.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The above does require though that Platform.accessibilityActiveProperty() is properly synchronized. I think it may be a good idea to fix that first. Perhaps all platform provided properties (if there are more) should ensure they're only initialized once.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unrelated, but I would rather disallow background access to any platform properties.

per the earlier email, this might (read: will) create concurrent access when the node is not yet attached to the scene graph:

background thread                   fx app thread
1. start creating the node
2. add listener to platform prop
3. keep initializing the node        <-- platform property change notification
4. some more code

step 3 is when concurrent access occurs.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, but that's because there is no synchronization. Here is a version that does do synchronization; I see no more exceptions (and it runs just as fast really):

public class PropTest {
  public static void main(String[] args) {
    Thread mainThread = Thread.currentThread();
    final var prop = new SimpleBooleanProperty(false) {
      @Override
      public synchronized void addListener(ChangeListener<? super Boolean> listener) {
        super.addListener(listener);
      }

      @Override
      public synchronized void set(boolean newValue) {
        super.set(newValue);
      }
    };

    // add listeners in a background thread
    var thr = new Thread(() -> {
        for (int i = 0; i < 1000000; i++) {
            ChangeListener<Boolean> listener = (obs, o, n) -> {
                if (!mainThread.equals(Thread.currentThread())) {
                    System.err.println("***** ERROR: listener called on wrong thread: " + Thread.currentThread());
                }
            };
            prop.addListener(listener);
        }
    });
    thr.start();

    // Fire events in main thread
    for (int jj = 0; jj < 1000000; jj++) {
        prop.set(!prop.get());
    }
  }
}

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Of course for a generic solution, we could provide a wrapper for properties, or custom synchronized properties.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, using a custom property, or a wrapper, would solve this particular synchronization problem, although that wouldn't solve all of the problems.

We would then be left with the problem that if the accessibility change listener did fire -- which necessarily happens on the FX app thread -- while the object was being set up on a background thread, the listener might try to modify properties that are being touched on the background thread. This is just another case of a more general problem we already have with animation, and which Andy fixed by not starting animation in a few places unless and until we are on the FX app thread.

So I think the solution Andy has chosen of deferring adding the listener is better than trying to fix the accessibility property, followed by fixing the listeners that use it, to be thread-safe.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you, @kevinrushforth .

Getting back to the rule of allowing Node construction from a background thread but not "use", should we consider enforcing the thread access rules for global and static objects (like Platform properties)?

In other words, constructing objects that might modify internal properties is ok, but trying to access shared objects is not?

(maybe it's time to move the discussion out of this PR into the mailing list perhaps?)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Enforcing the threading restriction when using properties is a larger conversation. If we want to have it, I agree it should be on the mailing list.

item.setNode(symbol);
}
// set symbol styles
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2010, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2010, 2025, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -25,38 +25,38 @@

package javafx.scene.chart;

import java.util.*;

import javafx.scene.AccessibleRole;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import javafx.animation.Animation;
import javafx.animation.FadeTransition;
import javafx.animation.Interpolator;
import javafx.animation.KeyFrame;
import javafx.animation.KeyValue;
import javafx.animation.ParallelTransition;
import javafx.animation.Timeline;
import javafx.application.Platform;
import javafx.beans.NamedArg;
import javafx.beans.property.DoubleProperty;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.css.CssMetaData;
import javafx.css.PseudoClass;
import javafx.css.Styleable;
import javafx.css.StyleableDoubleProperty;
import javafx.css.StyleableProperty;
import javafx.css.converter.SizeConverter;
import javafx.geometry.Orientation;
import javafx.scene.AccessibleRole;
import javafx.scene.Node;
import javafx.scene.layout.StackPane;
import javafx.util.Duration;

import com.sun.javafx.charts.Legend.LegendItem;

import javafx.css.StyleableDoubleProperty;
import javafx.css.CssMetaData;
import javafx.css.PseudoClass;

import javafx.css.converter.SizeConverter;
import javafx.collections.ListChangeListener;

import javafx.css.Styleable;
import javafx.css.StyleableProperty;

/**
* A chart that plots bars indicating data values for a category. The bars can be vertical or horizontal depending on
* which axis is a category axis.
@@ -559,7 +559,7 @@ private Node createBar(Series<X,Y> series, int seriesIndex, final Data<X,Y> item
bar = new StackPane();
bar.setAccessibleRole(AccessibleRole.TEXT);
bar.setAccessibleRoleDescription("Bar");
bar.focusTraversableProperty().bind(Platform.accessibilityActiveProperty());
bar.setFocusTraversable(isAccessibilityActive());
item.setNode(bar);
}
bar.getStyleClass().setAll("chart-bar", "series" + seriesIndex, "data" + itemIndex, series.defaultColorStyleClass);
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2010, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2010, 2025, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -28,10 +28,8 @@
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import javafx.animation.FadeTransition;
import javafx.animation.ParallelTransition;
import javafx.application.Platform;
import javafx.beans.NamedArg;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
@@ -41,7 +39,6 @@
import javafx.scene.layout.StackPane;
import javafx.scene.shape.Ellipse;
import javafx.util.Duration;

import com.sun.javafx.charts.Legend.LegendItem;

/**
@@ -270,7 +267,7 @@ public Object queryAccessibleAttribute(AccessibleAttribute attribute, Object...
};
bubble.setAccessibleRole(AccessibleRole.TEXT);
bubble.setAccessibleRoleDescription("Bubble");
bubble.focusTraversableProperty().bind(Platform.accessibilityActiveProperty());
bubble.setFocusTraversable(isAccessibilityActive());
item.setNode(bubble);
}
// set bubble styles
99 changes: 82 additions & 17 deletions modules/javafx.controls/src/main/java/javafx/scene/chart/Chart.java
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2010, 2022, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2010, 2025, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -28,9 +28,6 @@
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import com.sun.javafx.scene.control.skin.Utils;

import javafx.animation.Animation;
import javafx.animation.KeyFrame;
import javafx.application.Platform;
@@ -40,28 +37,29 @@
import javafx.beans.property.SimpleBooleanProperty;
import javafx.beans.property.StringProperty;
import javafx.beans.property.StringPropertyBase;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.beans.value.WritableValue;
import javafx.collections.ObservableList;
import javafx.css.CssMetaData;
import javafx.css.Styleable;
import javafx.css.StyleableBooleanProperty;
import javafx.css.StyleableObjectProperty;
import javafx.css.StyleableProperty;
import javafx.css.converter.BooleanConverter;
import javafx.css.converter.EnumConverter;
import javafx.geometry.Pos;
import javafx.geometry.Side;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.layout.Pane;
import javafx.scene.layout.Region;

import javafx.stage.Window;
import com.sun.javafx.charts.ChartLayoutAnimator;
import com.sun.javafx.charts.Legend;
import com.sun.javafx.scene.NodeHelper;

import javafx.css.StyleableBooleanProperty;
import javafx.css.StyleableObjectProperty;
import javafx.css.CssMetaData;

import javafx.css.converter.BooleanConverter;
import javafx.css.converter.EnumConverter;

import javafx.css.Styleable;
import javafx.css.StyleableProperty;
import com.sun.javafx.scene.control.skin.Utils;

/**
* Base class for all charts. It has 3 parts the title, legend and chartContent. The chart content is populated by the
@@ -104,6 +102,9 @@ public abstract class Chart extends Region {
/** Animator for animating stuff on the chart */
private final ChartLayoutAnimator animator = new ChartLayoutAnimator(chartContent);

// SimpleBooleanProperty or ObjectBinding
private volatile Object accessibilityActive;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can use ObservableValue<?> instead of Object as the type. Alternatively, use two fields, a SimpleBooleanProperty for use by the FX app thread and an ObjectBinding for use by a background thread. They wouldn't need to be volatile in that case. What you have is OK, but using two properties might simplify the logic a bit.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's a design decision - I won't want to waste an extra pointer.
The cpu cycles are much cheaper nowadays than bytes.
Extra bytes cost much more in cpu cycles (gc) and electricity.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While I still think the code would be cleaner with two properties, and the extra memory is not significant, what you have will work, so I won't object.


// -------------- PUBLIC PROPERTIES --------------------------------------------------------------------------------

/** The chart title */
@@ -274,7 +275,6 @@ protected ObservableList<Node> getChartChildren() {
*/
public Chart() {
titleLabel.setAlignment(Pos.CENTER);
titleLabel.focusTraversableProperty().bind(Platform.accessibilityActiveProperty());
getChildren().addAll(titleLabel, chartContent);
getStyleClass().add("chart");
titleLabel.getStyleClass().add("chart-title");
@@ -511,6 +511,71 @@ public StyleableProperty<Boolean> getStyleableProperty(Chart node) {
return getClassCssMetaData();
}

}
private void handleAccessibilityActive(boolean on) {
titleLabel.setFocusTraversable(on);
updateSymbolFocusable(on);
}

/**
* Invoked in the FX application thread when accessibility active platform property changes.
* The child classes should override this method to set focus traversable flag on every symbol, as well as
* other elements that need to be focusable.
* @param on whether the accessibility is active
*/
// package protected: custom charts must handle accessbility on their own
void updateSymbolFocusable(boolean on) {
}

/**
* When called from JavaFX application thread, returns the value of the property.
* When called from any other thread, returns false and sets up the machinery to
* invoke {@code updateSymbolFocusable()} when needed.
* The chart implementations should use this method to set focus travesable flags on the nodes
* which needs to be focus traversable when accessibility is on.
* @return
*/
// package protected: custom charts must handle accessbility on their own
final boolean isAccessibilityActive() {
if (Platform.isFxApplicationThread()) {
if (accessibilityActive instanceof SimpleBooleanProperty p) {
return p.get();
} else {
boolean aa = Platform.accessibilityActiveProperty().get();
SimpleBooleanProperty active = new SimpleBooleanProperty(aa);
accessibilityActive = active;
active.bind(Platform.accessibilityActiveProperty());
active.addListener((src, prev, on) -> {
handleAccessibilityActive(on);
});
return active.get();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm confused as to what is happening here. A boolean property active is created, which is bound to Platform.accessibilityActiveProperty (this creates a listener on accessibilityActiveProperty). We add then another listener on active.

What confuses me is what this is supposed to achieve, and also why we're initialising the property, but then bind it anyway... active = new SimpleBooleanProperty(aa) and later active.bind means that the variable aa is completely unnecessary as the bind will do this get for you.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

aa is unnecessary, you are right, bind() will set the value.

the property is created as a way to signal any subsequent calls that no more work is needed.

}
} else {
// chart and its data are allowed to be constructed in a background thread
if (accessibilityActive == null) {
// set up a listener which, once the chart gets connected to a window (always in the context of
// the fx application thread), calls isAccessibilityActive() again,
// which in turn installs the listener on the Platform.accessibilityActiveProperty.
ObservableValue<Window> winProp = sceneProperty().flatMap(Scene::windowProperty);
accessibilityActive = winProp; // keep the reference so it won't get gc

// lambda cannot be used in place of a ChangeListener in removeListener()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not use a Subscription then? It seems tailor-made for what you are trying to do.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know how to use Subscription in this case.
This does not work:

                ObservableValue<Window> winProp = sceneProperty().flatMap(Scene::windowProperty);
                accessibilityActive = winProp; // keep the reference so it won't get gc
                Subscription sub = winProp.subscribe((win) -> {
                    if (win != null) {
                        if (accessibilityActive == winProp) {
                            accessibilityActive = null;
                        }
                        if (isAccessibilityActive()) {
                            handleAccessibilityActive(true);
                        }
                        //winProp.removeListener(this);
                        sub.unsubscribe(); <-- COMPILE ERROR
                    }
                });

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@hjohn could you help here please? How could we use Subscription in a situation when it has to be unsubscribed from within the lambda?

Copy link
Collaborator

@hjohn hjohn Mar 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I haven't been tracking these fixes for allowing initialization in background threads, but it seems to me that basically anything should be allowed as long as you're not part of a visible scene graph -- and I think there's also no expectation that all functionality of a control "works" as long as it is not yet part of such a graph (ie. the listeners are only needed once it is part of a scene graph).

If you make listeners conditional on being part of a scene graph, then I think you can handle these with a single code path. Such a condition would be:

ObservableValue<Boolean> partOfSceneGraph = node.sceneProperty()
  .flatMap(Scene::windowProperty)
  .flatMap(Window::showingProperty)
  .orElse(false);
  
  // The node here is "this", the chart, because as soon as it is part of the scene
  // graph, you want the listeners to function.

You can then safely install any listeners directly, regardless if they're on a background thread or not:

someProperty().when(partOfSceneGraph).subscribe( ... );

Or:

someProperty().when(partOfSceneGraph).addListener( ... );

Or with any mappings added, you can even pick where you put the when:

someProperty().flatMap(xyz).when(partOfSceneGraph).addListener( ... );
someProperty().when(partOfSceneGraph).flatMap(xyz).addListener( ... );

On a background thread, partOfSceneGraph will be false, and no listener gets installed (yet). As soon as it becomes true all listeners with that same condition become active. It becomes true automatically when the node has a scene belonging to a window that is visible. Vice versa, if it ever becomes false again, (which it may when the Node is removed or replaced) all listeners using this condition are removed again.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you!

This is not what I asked though: my question was about using Subscription in one-off case, and more specifically, about whether it is even possible to unsubscribe from within the lambda.

What you are proposing is a slightly more involved change: for example, registering and unregistering listeners each time might introduce other issues (especially since we have no way to prioritize listeners/handlers). It also complicates the management of skin (as skins can be added/removed), something I would rather avoid.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As for this current code:

            ObservableValue<Window> winProp = sceneProperty().flatMap(Scene::windowProperty);
            accessibilityActive = winProp; // keep the reference so it won't get gc
            Subscription sub = winProp.subscribe((win) -> {
                if (win != null) {
                    if (accessibilityActive == winProp) {
                        accessibilityActive = null;
                    }
                    if (isAccessibilityActive()) {
                        handleAccessibilityActive(true);
                    }
                    //winProp.removeListener(this);
                    sub.unsubscribe(); <-- COMPILE ERROR
                }
            });

What you want there is not possible in this way as you can't use this in a lambda, nor refer to it via a local. You can achieve it only if you store the subscription in a field, not in a local. Assigning the Lambda to a ChangeListener local, and using a ChangeListener may be simpler (unless you go for the when solution that may eliminate the need for 2 different code paths).

Note that there is no need to store winProp; flatMap does not use weak listeners (when in active use), and so as long as sceneProperty() exists, the derived property via flatMap will also exist, and also its listener. The way the fluent mapping system works is that listeners are only present when something is actually listening:

ObservableValue<Window> winProp = sceneProperty().flatMap(Scene::windowProperty);
// ^^ nobody is listening, so no listeners installed and no hard references; it could GC but
// it is currently in a local, so it won't

ObservableValue<Window> winProp = sceneProperty().flatMap(Scene::windowProperty);
winProp.subscribe(v -> { .. });
// ^^ somebody is listening; all listeners get installed automatically, and now there are
// hard references.  No risk of GC.

You can then write it as a single statement which won't be GC'd as long as the subscription is active:

subscriptionFIeld = sceneProperty()
    .flatMap(Scene::windowProperty);
    .subscribe(v -> { 
        // use subscription field here if you want to unregister the lambda
        // Note: when you do, there will again be no listeners and thus no hard references and
        // thus the intermediate property created by flatMap can now be GC'd :)
    });

ChangeListener<Window> li = new ChangeListener<>() {
@Override
public void changed(ObservableValue<? extends Window> src, Window old, Window win) {
if (win != null) {
if (accessibilityActive == winProp) {
accessibilityActive = null;
}
if (isAccessibilityActive()) {
handleAccessibilityActive(true);
}
winProp.removeListener(this);
}
}
};
winProp.addListener(li);
}
return false;
}
}
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2010, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2010, 2025, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -26,17 +26,16 @@
package javafx.scene.chart;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javafx.animation.Animation;
import javafx.animation.FadeTransition;
import javafx.animation.Interpolator;
import javafx.animation.KeyFrame;
import javafx.animation.KeyValue;
import javafx.animation.Timeline;
import javafx.animation.Animation;
import javafx.application.Platform;
import javafx.beans.NamedArg;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.DoubleProperty;
@@ -46,26 +45,20 @@
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.css.CssMetaData;
import javafx.css.Styleable;
import javafx.css.StyleableBooleanProperty;
import javafx.css.StyleableProperty;
import javafx.css.converter.BooleanConverter;
import javafx.scene.AccessibleRole;
import javafx.scene.Node;
import javafx.scene.layout.StackPane;
import javafx.scene.shape.LineTo;
import javafx.scene.shape.Path;
import javafx.scene.shape.StrokeLineJoin;
import javafx.util.Duration;

import com.sun.javafx.charts.Legend.LegendItem;

import javafx.css.StyleableBooleanProperty;
import javafx.css.CssMetaData;

import javafx.css.converter.BooleanConverter;

import java.util.*;

import javafx.css.Styleable;
import javafx.css.StyleableProperty;

/**
* Line Chart plots a line connecting the data points in a series. The data points
* themselves can be represented by symbols optionally. Line charts are usually used
@@ -530,7 +523,7 @@ private Node createSymbol(Series<X, Y> series, int seriesIndex, final Data<X,Y>
symbol = new StackPane();
symbol.setAccessibleRole(AccessibleRole.TEXT);
symbol.setAccessibleRoleDescription("Point");
symbol.focusTraversableProperty().bind(Platform.accessibilityActiveProperty());
symbol.setFocusTraversable(isAccessibilityActive());
item.setNode(symbol);
}
// set symbol styles
Loading