<h1 id="how-to-cache-network-images-in-flutter">How to Cache Network Images in Flutter</h1> <p>Loading and displaying images from the network is a common requirement in most apps. However, repeatedly downloading the same images is inefficient and can lead to a poor user experience. In this guide, we'll explore how to effectively cache network images in Flutter using the popular <code>cached_network_image</code> package.</p> <h2 id="why-cache-images">Why Cache Images?</h2> <p>Caching network images provides several benefits:</p> <ol> <li><strong>Reduced Data Usage</strong>: Images are downloaded only once and stored locally</li> <li><strong>Faster Load Times</strong>: Cached images load instantly from local storage</li> <li><strong>Offline Support</strong>: Previously loaded images can be displayed without internet</li> <li><strong>Improved Performance</strong>: Reduces network requests and memory usage</li> </ol> <p><img src="" alt="Image Caching Visualization" /></p> <h2 id="getting-started-with-cached_network_image">Getting Started with cached_network_image</h2> <h3 id="add-the-package">1. Add the Package</h3> <p>First, add the package to your <code>pubspec.yaml</code> file:</p> <pre>dependencies: flutter: sdk: flutter cached_network_image: ^3.2.3 </pre> <p>Run <code>flutter pub get</code> to install the package.</p> <h3 id="basic-usage">2. Basic Usage</h3> <p>The simplest way to use the package is to replace your <code>Image.network</code> widgets with <code>CachedNetworkImage</code>:</p> <pre>import 'package:flutter/material.dart'; import 'package:cached_network_image/cached_network_image.dart';
class BasicExample extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('Cached Network Image'), ), body: Center( child: CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), ), ), ); } } </pre> <p>This will:</p> <ul> <li>Display a loading spinner while the image loads</li> <li>Show the image once downloaded (and cache it)</li> <li>Show an error icon if the image fails to load</li> </ul> <h2 id="customizing-cachednetworkimage">Customizing CachedNetworkImage</h2> <h3 id="adding-loading-placeholder">Adding Loading Placeholder</h3> <p>You can customize the placeholder widget shown during loading:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", placeholder: (context, url) => Container( color: Colors.grey[300], child: Center( child: CircularProgressIndicator( valueColor: AlwaysStoppedAnimation<Color>(Colors.blue), ), ), ), errorWidget: (context, url, error) => Icon( Icons.error, color: Colors.red, size: 30, ), ) </pre> <h3 id="configuring-image-properties">Configuring Image Properties</h3> <p>You can use most of the same properties available with regular Image widgets:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", imageBuilder: (context, imageProvider) => Container( height: 200, width: 200, decoration: BoxDecoration( borderRadius: BorderRadius.circular(20), image: DecorationImage( image: imageProvider, fit: BoxFit.cover, ), ), ), placeholder: (context, url) => Container( height: 200, width: 200, child: Center( child: CircularProgressIndicator(), ), ), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h3 id="using-fade-in-animation">Using Fade-In Animation</h3> <p>You can animate the appearance of images as they load:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), fadeInDuration: Duration(milliseconds: 500), fadeInCurve: Curves.easeIn, ) </pre> <h2 id="advanced-configuration">Advanced Configuration</h2> <h3 id="configuring-cache-settings">Configuring Cache Settings</h3> <p>You can configure the caching behavior by using the <code>CacheManager</code>:</p> <pre>import 'package:cached_network_image/cached_network_image.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart';
// Create a custom cache manager class CustomCacheManager { static const key = 'customCacheKey'; static CacheManager instance = CacheManager( Config( key, stalePeriod: Duration(days: 7), maxNrOfCacheObjects: 100, repo: JsonCacheInfoRepository(databaseName: key), fileService: HttpFileService(), ), ); }
// Use the custom cache manager CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", cacheManager: CustomCacheManager.instance, placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h3 id="handling-image-progress">Handling Image Progress</h3> <p>You can track the download progress of an image:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", progressIndicatorBuilder: (context, url, downloadProgress) => CircularProgressIndicator(value: downloadProgress.progress), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h2 id="example-image-gallery-with-caching">Example: Image Gallery with Caching</h2> <p>Here's a complete example of an image gallery that uses cached network images:</p> <pre>import 'package:flutter/material.dart'; import 'package:cached_network_image/cached_network_image.dart';
class ImageGallery extends StatelessWidget { final List<String> imageUrls = [ "https://picsum.photos/id/1001/800/800", "https://picsum.photos/id/1002/800/800", "https://picsum.photos/id/1003/800/800", "https://picsum.photos/id/1004/800/800", "https://picsum.photos/id/1005/800/800", "https://picsum.photos/id/1006/800/800", ];
@override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('Cached Image Gallery'), ), body: GridView.builder( padding: EdgeInsets.all(8), gridDelegate: SliverGridDelegateWithFixedCrossAxisCount( crossAxisCount: 2, crossAxisSpacing: 8, mainAxisSpacing: 8, ), itemCount: imageUrls.length, itemBuilder: (context, index) { return ClipRRect( borderRadius: BorderRadius.circular(12), child: CachedNetworkImage( imageUrl: imageUrls[index], fit: BoxFit.cover, placeholder: (context, url) => Container( color: Colors.grey[300], child: Center( child: CircularProgressIndicator(), ), ), errorWidget: (context, url, error) => Container( color: Colors.grey[300], child: Icon( Icons.error, color: Colors.red, ), ), ), ); }, ), ); } } </pre> <h2 id="clearing-the-cache">Clearing the Cache</h2> <p>Sometimes you might need to clear the image cache:</p> <pre>// Clear a specific image cache await DefaultCacheManager().removeFile("https://picsum.photos/id/237/200/300");
// Clear all image caches await DefaultCacheManager().emptyCache(); </pre> <h2 id="performance-tips-and-best-practices">Performance Tips and Best Practices</h2> <h3 id="specify-image-dimensions">1. Specify Image Dimensions</h3> <p>Always specify image dimensions when possible to avoid layout shifts:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/200/300", width: 200, height: 300, fit: BoxFit.cover, placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h3 id="use-memcachewidth-and-memcacheheight">2. Use memCacheWidth and memCacheHeight</h3> <p>Reduce memory usage by specifying the cached image dimensions:</p> <pre>CachedNetworkImage( imageUrl: "https://picsum.photos/id/237/1000/1500", memCacheWidth: 400, // Resize the image in memory to 400 pixels wide memCacheHeight: 600, // Resize the image in memory to 600 pixels high placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h3 id="handle-different-screen-sizes">3. Handle Different Screen Sizes</h3> <p>Fetch images appropriate to screen size:</p> <pre>CachedNetworkImage( imageUrl: MediaQuery.of(context).size.width > 800 ? "https://picsum.photos/id/237/1000/1000" // High-res : "https://picsum.photos/id/237/500/500", // Low-res placeholder: (context, url) => CircularProgressIndicator(), errorWidget: (context, url, error) => Icon(Icons.error), ) </pre> <h3 id="clear-old-caches-for-long-lived-apps">4. Clear Old Caches for Long-lived Apps</h3> <p>For long-lived apps, consider clearing old caches periodically:</p> <pre>void clearOldCache() async { await DefaultCacheManager().emptyCache(); print('Cache cleared!'); }
// Call this method when appropriate, e.g., on app start @override void initState() { super.initState();
// Check the last cache clear date and clear if older than a week final lastClearDate = await getLastClearDate(); final oneWeek = Duration(days: 7);
if (DateTime.now().difference(lastClearDate) > oneWeek) { clearOldCache(); await saveLastClearDate(DateTime.now()); } } </pre> <h2 id="troubleshooting-common-issues">Troubleshooting Common Issues</h2> <h3 id="images-not-caching">Images not caching</h3> <ul> <li>Check if you have sufficient storage permissions</li> <li>Verify the URLs are accessible and consistent</li> <li>Make sure the device has enough storage space</li> </ul> <h3 id="memory-issues-with-large-images">Memory issues with large images</h3> <ul> <li>Use <code>memCacheWidth</code> and <code>memCacheHeight</code> parameters</li> <li>Consider using placeholder images that match the expected size</li> <li>Load lower resolution images when appropriate</li> </ul> <h3 id="images-not-showing-offline">Images not showing offline</h3> <ul> <li>Ensure the images were successfully cached during online access</li> <li>Check if the cache has been cleared</li> <li>Verify the cache duration settings</li> </ul> <h2 id="conclusion">Conclusion</h2> <p>Implementing image caching in Flutter apps is straightforward with the <code>cached_network_image</code> package. This approach significantly improves user experience by reducing loading times, minimizing data usage, and enabling offline access to previously viewed images.</p> <p>By following the best practices outlined in this guide, you can ensure your app handles images efficiently and provides a smooth experience for your users.</p> <p>Happy coding!</p>