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

Defer image decoding when scrolling fast. #48536

Closed
wants to merge 13 commits into from

Conversation

dnfield
Copy link
Contributor

@dnfield dnfield commented Jan 9, 2020

Description

A List or GridView of images can end up consuming vast amounts of memory, particularly when the user "flings" the list causing the scrollable to go ballistic.

The basic problem is that we push a ton of decoding work onto the IO thread of the engine, and then end up not even needing those images because the children aren't in the tree anymore. This causes thrash on the image cache, wasted CPU load for decoding those images, and in some cases OOM death.

I've resolved this by adding a new ImageProvider implementation that allows the caller to defer or cancel loading the image. The ImageState now uses this to wrap its incoming image provider, and defers loading until scrolling has slowed down enough that we would be seen on screen (defined as scrolling velocity < the physical number of pixels on the longest side of the screen). If we're no longer in the tree at that point, we cancel the load. This provider is cache-aware, and will just return the image in the cache if at any point during its cycle the image appears in the cache.

/cc @hugosantos @ignatz @alml @cvarvara
/cc @liyuqian @chinmaygarde

Related Issues

#32143
#44510
b/144232910

#48305
Fixes #48775

Tests

I added the following tests:

  • Tests that the image cache can tell us whether it has a pending or completed image for a given key
  • Tests that the DeferringImageProvider can cancel, defer, or resolve an image
  • Tests that the DeferringImageProvider will check the cache (in case someone pre-cached the image, or in case another widget managed to cache it for us since we last checked).
  • Tests that a large grid view of images driven by a high velocity ballistic simulation do not try to load images during the high velocity part of the simulation.

Checklist

Before you create this PR confirm that it meets all requirements listed below by checking the relevant checkboxes ([x]). This will ensure a smooth and quick review process.

  • I read the Contributor Guide and followed the process outlined there for submitting PRs.
  • I signed the CLA.
  • I read and followed the Flutter Style Guide, including Features we expect every widget to implement.
  • I read the Tree Hygiene wiki page, which explains my responsibilities.
  • I updated/added relevant documentation (doc comments with ///).
  • All existing and new tests are passing.
  • The analyzer (flutter analyze --flutter-repo) does not report any problems on my PR.
  • I am willing to follow-up on review comments in a timely manner.

Breaking Change

Right now this is - I'm open to feedback suggesting ways to make it not be.

Will add link to design doc shortly - will add link to migration guide if this remains an API break before landing.

Did any tests fail when you ran them? Please read Handling breaking changes.

  • No, no existing tests failed, so this is not a breaking change.
  • Yes, this is a breaking change. If not, delete the remainder of this section.
    • I wrote a design doc: https://flutter.dev/go/deferred-image-decoding
    • I got input from the developer relations team, specifically from: Replace with the names of who gave advice
    • I wrote a migration guide: Replace with a link to your migration guide

Verified

This commit was signed with the committer’s verified signature.
erlend-aasland Erlend E. Aasland

Verified

This commit was signed with the committer’s verified signature.
erlend-aasland Erlend E. Aasland
@dnfield dnfield added c: API break Backwards-incompatible API changes f: scrolling Viewports, list views, slivers, etc. a: images Loading, displaying, rendering images perf: memory Performance issues related to memory labels Jan 9, 2020
@fluttergithubbot fluttergithubbot added the framework flutter/packages/flutter repository. See also f: labels. label Jan 9, 2020
@@ -955,7 +956,6 @@ class _ImageState extends State<Image> with WidgetsBindingObserver {

@override
void dispose() {
assert(_imageStream != null);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

We might not have ever gotten an _imageStream before being disposed of if we were scrolling fast.

Copy link
Member

Choose a reason for hiding this comment

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

How can this happen? Isn't provider.resolve in line 1036 always synchronously returning a stream? I also don't see code that changed when we call resolve on the provider...

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'll put this back. This was from a different experiment

@@ -231,6 +237,11 @@ class ImageCache {
return result;
}

/// Returns whether this [key] has been previously added by [putIfAbsent].
bool containsKey(Object key) {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is breaking (we have internal customers that implement ImageCache). There are ugly ways to make it not be breaking (e.g. make putIfAbsent tolerate having a loader that returns null, which is documented as being bad but not tested against, and we have internal implementers that tolerate it).

I think it's worth the break but am willing to hear other ideas.

Copy link
Contributor

Choose a reason for hiding this comment

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

Don't hesitate to make breaking changes if they are valuable. Just make sure to go through the process (which involves getting feedback about whether it's valuable).

/// method.
/// method. If they need to manage the actual resolution of the image, they
/// should override [resolveForStream] instead.
@nonVirtual
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is a bigger break that I need to make the way I'm doing DeferringImageProvider work. I don't to call resolve from there and end up decoding work before I'm ready.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Interestingly, this would not break any internal customers or existing customer tests - it only breaks our own test implementations.

int get hashCode => hashValues(imageProvider, getNextAction);

@override
String toString() => '$runtimeType(imageProvider: $imageProvider)';
Copy link
Member

Choose a reason for hiding this comment

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

please don't use runtimeType.toString()

Copy link
Member

Choose a reason for hiding this comment

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

it is incredibly slow

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is our standard toString template right?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

(we do this everywhere)

Copy link
Member

Choose a reason for hiding this comment

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

That doesn't make it fast

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Fair - how about we handle this in a separate PR and upate the style guide?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

(or file a bug against Dart SDK to make it be fast?)

Copy link
Contributor

Choose a reason for hiding this comment

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

it's fine to use runtimeType.toString, just don't use it in release builds.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is done.

@dnfield
Copy link
Contributor Author

dnfield commented Jan 9, 2020

Related: #32156

return stream;
}

/// Subclasses that need to manage resolution more finely should override
Copy link
Member

Choose a reason for hiding this comment

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

The documentation should probably emphasize what this method actually does. Then as a note add information about overriding. It's hard to overrride if you don't know the contract of the method.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated

return;
}
switch (getNextAction()) {
case DeferringImageProviderAction.defer:
Copy link
Member

Choose a reason for hiding this comment

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

Do we really need defer? Seems like the cancelling work you did could be sufficient for your problem.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yes. Without this our only options are to either immediately cancel or immediately resolve. We want to be able to defer until a condition is met (scrolling has slowed)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

We also need an image provider here so we can see if we even should bother waiting or not - if the image is already in the cache we shouldn't wait.

// to see if we've slowed down or disappeared.
// It's safe to use these calls as point-in-time - do not subscribe to
// updates of MediaQuery or Scrollable.
final double maxPhysicalDimension = WidgetsBinding.instance.window.physicalSize.longestSide;
Copy link
Member

Choose a reason for hiding this comment

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

Nit: being explicit on units here can help verify the math.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Are you asking for a different variable name or something else?

Copy link
Member

Choose a reason for hiding this comment

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

Yep, putting the units in the variable name is the easiest change.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Ahh sure. That makes sense.

Copy link
Member

Choose a reason for hiding this comment

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

Just to be a bit more obvious. I've seen velocity calculations get messed up when people mix logical pixels and hardware pixels.

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 am intentionally doing that but I'll add some explanation.

Basically, we want something a factor bigger than logical pixels, and I'm assuming this is safe because higher dpi should mean more memory available on device.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Pushed up some changes to the docs and this property name.

@@ -675,6 +675,9 @@ abstract class ScrollPosition extends ViewportOffset with ScrollMetrics {
ScrollActivity get activity => _activity;
ScrollActivity _activity;

/// The current [ScrollActivity.velocity] of the current activity, if any.
double get currentScrollVelocity => activity?.velocity;
Copy link
Member

Choose a reason for hiding this comment

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

Document units in docstring and/or property name.

Copy link
Member

Choose a reason for hiding this comment

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

This is still outstanding =)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Hm. I thought I did this one, I think I did the similar one instead. Will update.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

GitHub is messed up here. The text acutally reads:

  ///
  /// This value is in logical pixels per second. It may be positive, negative,
  /// 0, or null.

In addition to what's showing right now. You can see it at: f732fbe

Copy link
Member

Choose a reason for hiding this comment

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

Cool thanks, what's a velocity of null mean?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Hm. I might be wrong about this. I was thinking there was some code path somewhere that could set it to null or set the activity to null. I'm having trouble finding it. @goderbauer may know if this is accurate.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Oh, derp.

This can return null if you're not in a scrollable context :)

Copy link
Member

Choose a reason for hiding this comment

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

It's not clear to me why this should return null.

This can return null if you're not in a scrollable context :)

How would you have a ScrollPosition if you're not in a scrollable context?

Also: What does it mean from an API point of view when your velocity is null? Shouldn't this always return negative, 0 or positive numbers?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

You are right, I think I copied this over from scrollable.dart, which is where it can be null. Removed the bit about null here.

@dnfield dnfield requested a review from gaaclarke January 14, 2020 23:54
// It's safe to use these calls as point-in-time - do not subscribe to
// updates of MediaQuery or Scrollable.
final double maxPhysicalPixels = WidgetsBinding.instance.window.physicalSize.longestSide;
if ((Scrollable.scrollingVelocityOfContext(context) ?? 0).abs() > maxPhysicalPixels) {
Copy link
Member

Choose a reason for hiding this comment

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

You are mixing units here, velocity versus physical pixels. I'd compare velocity to velocity.

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 intentional though. The idea is the DPI is a good heuristic to use for this case.

Copy link
Member

@gaaclarke gaaclarke Jan 15, 2020

Choose a reason for hiding this comment

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

That's fine, i'd document it with another variable like.
double fullScreenPerSecond = maxPhysicalPixels;

Copy link
Member

Choose a reason for hiding this comment

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

Is there some scenario where we could end up with enough velocity to meet this condition indefinitely? For example, some sort of absurdly pixel scaling combined with a looping animation?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

In such a situation we'd have to be scrolling at high velocity indefinitely. Or, we'd have to be in some kind of crazy scroll activity that's lying to us about how fast it's scrolling indefinitely.

I suppose the most pathological case I can think of is an animation that loops at high velocity such that it scrolls up and down very very fast. If you wanted the image to ever show up in that case, you'd have to precache it.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

And it couldn't be just any animation, it would have to be an animation that drove a scroll controller such that the activity used was a ballistic activity scrolling at high velocity.

Copy link
Member

@gaaclarke gaaclarke left a comment

Choose a reason for hiding this comment

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

+1 . I'd like to get another lgtm. The documentation and testing is good and I don't see any obvious bugs. I still don't 100% grok the necessity of the defer and I'm not too keen on the usage of the DPI as a proxy to fudge the heuristics.

scheduleMicrotask(() => deferredResolve(key));
});
return;
case DeferringImageProviderAction.cancel:
Copy link
Member

Choose a reason for hiding this comment

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

cancel seems like it could stop an image from being loaded, but not once it starts loading? It should be possible to cancel NetworkRequests too, though I'm not sure if HttpClient exposes an API for it

Copy link
Contributor Author

Choose a reason for hiding this comment

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

We can look at cancelling once a load starts, but this PR isn't trying to solve that one.

Part of the problem there is that it will be very hard once we actually start the machinery on the IO thread in the engine to cancel it before allocations have happened.

Network request handling could use some love but I think that's orthogonal to this issue. This is trying to stop us from even making the network request to begin with if we won't need it.

Copy link
Member

Choose a reason for hiding this comment

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

I think it would be unfortunate if we had to come up with an entirely separate solution for network requests. I might propose a slightly more general API:

The deferring image provider accepts a controller/token instead of a callback. This controller maintains a state flag similar to what the callback is providing now.

The controller state can be modified via setters/methods like cancel() or resume().

The cancellation could be propagated up through the network image provider, by exposing some functionality on ImageStream perhaps? Ideally this could eventually be wired up the HttpClient and others.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

My sense is that places we've introduced controllers (e.g. TextEditing) have really only caused more confusion and API challenges than they resolve. I think it will also become very confusing if cancel behaves differently in different contexts - did I cancel before making the request? Did I cancel the active request? Did I cancel it early enough to have effect? Did I call cancel after the whole thing was already done?

I think for NetworkRequests made by the framework in general, we should create another InheritedWidget that would handle things like this (caching requests, cancelling requests, etc). WDYT?

Copy link
Member

Choose a reason for hiding this comment

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

I don't really agree but I'm not involved enough in the problem, so I'll defer to the rest of the framework team

@@ -270,10 +271,29 @@ abstract class ImageProvider<T> {
/// This is the public entry-point of the [ImageProvider] class hierarchy.
///
/// Subclasses should implement [obtainKey] and [load], which are used by this
/// method.
/// method. If they need to manage the actual resolution of the image, they
/// should override [resolveForStream] instead.
Copy link
Member

Choose a reason for hiding this comment

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

nit: instead of what?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Added a couple words (of this method)

/// never fire and the cached image is reused.
@immutable
class DeferringImageProvider<T> extends ImageProvider<T> {

Copy link
Member

Choose a reason for hiding this comment

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

nit: remove extra blank line

Copy link
Contributor Author

Choose a reason for hiding this comment

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

done

/// This callback is never called if the [ImageCache] already has the image
/// present, such as from [precacheImage] or another provider successfully
/// loading it.
final DeferringImageProviderAction Function() getNextAction;
Copy link
Member

Choose a reason for hiding this comment

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

Use a typedef

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

@@ -868,6 +887,104 @@ class ExactAssetImage extends AssetBundleImageProvider {
String toString() => '${objectRuntimeType(this, 'ExactAssetImage')}(name: "$keyName", scale: $scale, bundle: $bundle)';
}

/// The action a [DeferringImageProvider] should take currently.
Copy link
Member

Choose a reason for hiding this comment

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

Maybe expand on this a little and mention that the getNextAction callback on DeferringImageProvider is supposed to return these.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

/// its [DeferringImageProvider.getNextAction] callback anymore.
cancel,

/// Indicates that the caller would like to defer the image and be called
Copy link
Member

Choose a reason for hiding this comment

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

Maybe clearer:

Suggested change
/// Indicates that the caller would like to defer the image and be called
/// Indicates that the caller would like to defer resolving the image and be called

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

@@ -955,7 +956,6 @@ class _ImageState extends State<Image> with WidgetsBindingObserver {

@override
void dispose() {
assert(_imageStream != null);
Copy link
Member

Choose a reason for hiding this comment

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

How can this happen? Isn't provider.resolve in line 1036 always synchronously returning a stream? I also don't see code that changed when we call resolve on the provider...

@@ -1064,14 +1092,14 @@ class _ImageState extends State<Image> with WidgetsBindingObserver {
}

void _listenToStream() {
if (_isListeningToStream)
if (_isListeningToStream || _imageStream == null)
Copy link
Member

Choose a reason for hiding this comment

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

Same here (and below): how can this be null?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Removed - this was from a previous experiment that I forgot to clean up.

// It's safe to use these calls as point-in-time - do not subscribe to
// updates of MediaQuery or Scrollable.
final double maxPhysicalPixels = WidgetsBinding.instance.window.physicalSize.longestSide;
if ((Scrollable.scrollingVelocityOfContext(context) ?? 0).abs() > maxPhysicalPixels) {
Copy link
Member

Choose a reason for hiding this comment

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

What if I very quickly and continuously scroll up and down: Does this block loading images forever?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

You would have to precache the image in such a case.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

And for most cases, you'd eventually lose velocity long enough at some point between frames (during a direction change) that it should still end up working out.

@@ -675,6 +675,9 @@ abstract class ScrollPosition extends ViewportOffset with ScrollMetrics {
ScrollActivity get activity => _activity;
ScrollActivity _activity;

/// The current [ScrollActivity.velocity] of the current activity, if any.
double get currentScrollVelocity => activity?.velocity;
Copy link
Member

Choose a reason for hiding this comment

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

It's not clear to me why this should return null.

This can return null if you're not in a scrollable context :)

How would you have a ScrollPosition if you're not in a scrollable context?

Also: What does it mean from an API point of view when your velocity is null? Shouldn't this always return negative, 0 or positive numbers?

/// is only safe to use as a single point in time reference.
///
/// This value is in logical pixels per second. It may be positive, negative,
/// 0, or null.
Copy link
Member

Choose a reason for hiding this comment

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

Document what null means?

(here it's correct what you said above: You're not in a scrollable context)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

// It's safe to use these calls as point-in-time - do not subscribe to
// updates of MediaQuery or Scrollable.
final double maxPhysicalPixels = WidgetsBinding.instance.window.physicalSize.longestSide;
if ((Scrollable.scrollingVelocityOfContext(context) ?? 0).abs() > maxPhysicalPixels) {
Copy link
Member

Choose a reason for hiding this comment

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

It is a little surprising to tie our Image implementation to our implementation of Scrollable. If somebody were to write a new Scrolling lib (because they don't like our sliver-based on or whatever) they would not be able to make uses of this feature for images shown in their scrolling container... This my be more hypothetical, though.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yes, I agree this is a problem. I'm not sure what we can do about it at the moment. I think we should punt on it for now and deal with it if it comes up - we could rework this so that you could find other ways to notify widgets about how fast you are scrolling.

return stream;
}

/// This method does the acutal work of [resolve], both decoding the image
Copy link
Contributor

Choose a reason for hiding this comment

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

typo actual

///
/// While it returns [DeferringImageProviderAction.defer], it schedules a new
/// call during the [SchedulerPhase.midFrameMicrotasks] of the next frame.
final DeferringImageProviderActionCallback handleDeferringImageProviderAction;
Copy link
Contributor

Choose a reason for hiding this comment

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

per our style guide this should be called onSomething.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

@override
void resolveForStream(ImageConfiguration configuration, ImageStream stream) {
void deferredResolve(Object key) {
if (PaintingBinding.instance.imageCache.containsKey(key)) {
Copy link
Contributor

Choose a reason for hiding this comment

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

we should document that this image provider only works with other image providers that use the cache correctly

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Done

// Did this element get disposed of (scrolled too far out of view before
// scrolling slowed down enough to catch up)?
if (!mounted) {
return DeferringImageProviderAction.cancel;
Copy link
Contributor

Choose a reason for hiding this comment

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

it really would be better if we could do that in dispose...

return;
}
}
obtainKey(configuration).then(deferredResolve);
Copy link
Contributor

Choose a reason for hiding this comment

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

this line drops errors

}

/// This method does the acutal work of [resolve], both decoding the image
/// and making sure it ends up in the [ImageCache].
Copy link
Contributor

Choose a reason for hiding this comment

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

"Decode the image and add it to the [ImageCache] if necessary".

@Hixie
Copy link
Contributor

Hixie commented Jan 22, 2020

@goderbauer @dnfield and I discussed this a bit. Some thoughts:

  • the heuristic probably belongs on the physics, that way it can be overridden in crazy cases involving weird transforms, etc.
  • we could call this something more general, like "shouldDeferLoads", to make it clearer that this can be reused in other context, e.g. whether to load news reports when scrolling through a news feed.

@dnfield
Copy link
Contributor Author

dnfield commented Jan 22, 2020

Talked offline a bit more -

It seems like this logic would be more comfortable living more in the widgets layer. Going to experiment with that to see if I can clean up some of the rough edges here.

@Hixie
Copy link
Contributor

Hixie commented Jan 22, 2020

Further discussion with @dnfield suggested that maybe what we should do is move the new image provider into the widgets layer, so it can know about BuildContext and Scrollable, and have it create a new ImageStream or ImageStreamCompleter that figures out whether to cancel based on when it runs out of listeners.

@dnfield dnfield mentioned this pull request Jan 22, 2020
9 tasks
@dnfield
Copy link
Contributor Author

dnfield commented Jan 23, 2020

I'm going to close this - the approach I'm going to take has some similarities to this but should be an improved API. At this point, there's lots of out of date comments on this PR and the approach is different enough that I'd like to not lose all the context here. Will follow up with the next PR after #49319 lands (second PR will depend on that one)

@dnfield
Copy link
Contributor Author

dnfield commented Jan 24, 2020

Continued in #49389

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
a: images Loading, displaying, rendering images c: API break Backwards-incompatible API changes f: scrolling Viewports, list views, slivers, etc. framework flutter/packages/flutter repository. See also f: labels. perf: memory Performance issues related to memory
Projects
None yet
Development

Successfully merging this pull request may close these issues.

How to stop image load when scrolling list ?
7 participants