davemorrissey | e7e9a6a | 2015-06-30 15:11:43 +0100 | [diff] [blame] | 1 | Subsampling Scale Image View |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 2 | =========================== |
davemorrissey | b064b0b | 2013-08-26 09:45:26 -0700 | [diff] [blame] | 3 | |
David Morrissey | cf2f5e9 | 2017-11-02 19:12:41 +0000 | [diff] [blame] | 4 | [](https://travis-ci.org/davemorrissey/subsampling-scale-image-view) |
| 5 | |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 6 | A custom image view for Android, designed for photo galleries and displaying huge images (e.g. maps and building plans) without `OutOfMemoryError`s. Includes pinch to zoom, panning, rotation and animation support, and allows easy extension so you can add your own overlays and touch event detection. |
David Morrissey | d419186 | 2014-10-18 23:14:07 +0100 | [diff] [blame] | 7 | |
davemorrissey | 4fcb613 | 2015-03-23 21:04:41 +0000 | [diff] [blame] | 8 | The view optionally uses subsampling and tiles to support very large images - a low resolution base layer is loaded and as you zoom in, it is overlaid with smaller high resolution tiles for the visible area. This avoids holding too much data in memory. It's ideal for displaying large images while allowing you to zoom in to the high resolution details. You can disable tiling for smaller images and when displaying a bitmap object. There are some advantages and disadvantages to disabling tiling so to decide which is best, see [the wiki](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/02.-Displaying-images). |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 9 | |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 10 | #### Guides |
| 11 | |
davemorrissey | 126eefa | 2015-03-22 19:18:54 +0000 | [diff] [blame] | 12 | * [Releases & downloads](https://github.com/davemorrissey/subsampling-scale-image-view/releases) |
davemorrissey | 0f4cd93 | 2015-03-23 20:12:12 +0000 | [diff] [blame] | 13 | * [Installation and setup](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/01.-Setup) |
| 14 | * [Image display notes & limitations](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/02.-Displaying-images) |
| 15 | * [Using preview images](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/03.-Preview-images) |
| 16 | * [Handling orientation changes](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/05.-Orientation-changes) |
| 17 | * [Advanced configuration](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/07.-Configuration) |
| 18 | * [Event handling](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/09.-Events) |
| 19 | * [Animation](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/08.-Animation) |
davemorrissey | 49fb934 | 2015-03-23 22:24:29 +0000 | [diff] [blame] | 20 | * [Extension](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/10.-Extension) |
David Morrissey | 3fccc74 | 2017-11-16 14:57:46 +0000 | [diff] [blame] | 21 | * [Reference (JavaDocs)](http://davemorrissey.github.io/subsampling-scale-image-view/javadoc/) |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 22 | |
David Morrissey | b11947f | 2017-12-02 08:40:06 +0000 | [diff] [blame] | 23 | #### Migration guides |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 24 | |
David Morrissey | 8181450 | 2017-12-12 08:36:25 +0000 | [diff] [blame] | 25 | Versions 3.9.0, 3.8.0 and 3.0.0 contain breaking changes. Migration instructions can be found [in the wiki](https://github.com/davemorrissey/subsampling-scale-image-view/wiki/X.--Migration-guides). |
David Morrissey | cf188f8 | 2017-11-16 12:58:43 +0000 | [diff] [blame] | 26 | |
David Morrissey | ef41d8a | 2014-06-10 01:00:09 +0100 | [diff] [blame] | 27 | #### Download the sample app |
| 28 | |
David Morrissey | ed7bd39 | 2017-11-27 10:52:56 +0000 | [diff] [blame] | 29 | [](https://play.google.com/store/apps/details?id=com.davemorrissey.labs.subscaleview.sample) |
David Morrissey | ef41d8a | 2014-06-10 01:00:09 +0100 | [diff] [blame] | 30 | |
David Morrissey | cac4407 | 2017-11-27 10:54:24 +0000 | [diff] [blame] | 31 | [Kotlin Sample App on GitHub](https://github.com/davemorrissey/ssiv-kotlin-sample) |
| 32 | |
David Morrissey | 0102865 | 2017-11-17 18:03:50 +0000 | [diff] [blame] | 33 | #### Demo |
| 34 | |
| 35 |  |
| 36 | |
David Morrissey | 36cc125 | 2015-01-12 21:42:02 +0000 | [diff] [blame] | 37 | ## Features |
David Morrissey | 0e895c2 | 2013-08-26 20:07:35 +0100 | [diff] [blame] | 38 | |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 39 | #### Image display |
David Morrissey | 365ccab | 2014-07-31 00:08:41 +0100 | [diff] [blame] | 40 | |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 41 | * Display images from assets, resources, the file system or bitmaps |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 42 | * Automatically rotate images from the file system (e.g. the camera or gallery) according to EXIF |
| 43 | * Manually rotate images in 90° increments |
David Morrissey | 0d041ff | 2015-03-08 21:44:44 +0000 | [diff] [blame] | 44 | * Display a region of the source image |
| 45 | * Use a preview image while large images load |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 46 | * Swap images at runtime |
David Morrissey | 36cc125 | 2015-01-12 21:42:02 +0000 | [diff] [blame] | 47 | * Use a custom bitmap decoder |
David Morrissey | 0e895c2 | 2013-08-26 20:07:35 +0100 | [diff] [blame] | 48 | |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 49 | *With tiling enabled:* |
David Morrissey | 365ccab | 2014-07-31 00:08:41 +0100 | [diff] [blame] | 50 | |
| 51 | * Display huge images, larger than can be loaded into memory |
| 52 | * Show high resolution detail on zooming in |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 53 | * Tested up to 20,000x20,000px, though larger images are slower |
David Morrissey | 365ccab | 2014-07-31 00:08:41 +0100 | [diff] [blame] | 54 | |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 55 | #### Gesture detection |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 56 | |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 57 | * One finger pan |
| 58 | * Two finger pinch to zoom |
davemorrissey | a033698 | 2015-02-23 21:49:23 +0000 | [diff] [blame] | 59 | * Quick scale (one finger zoom) |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 60 | * Pan while zooming |
| 61 | * Seamless switch between pan and zoom |
| 62 | * Fling momentum after panning |
David Morrissey | 77096ba | 2014-06-05 21:22:44 +0100 | [diff] [blame] | 63 | * Double tap to zoom in and out |
David Morrissey | 02ceb3d | 2014-05-30 20:48:51 +0100 | [diff] [blame] | 64 | * Options to disable pan and/or zoom gestures |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 65 | |
David Morrissey | 9f3fad1 | 2014-06-08 10:16:49 +0100 | [diff] [blame] | 66 | #### Animation |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 67 | |
David Morrissey | 9f3fad1 | 2014-06-08 10:16:49 +0100 | [diff] [blame] | 68 | * Public methods for animating the scale and center |
| 69 | * Customisable duration and easing |
| 70 | * Optional uninterruptible animations |
| 71 | |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 72 | #### Overridable event detection |
| 73 | * Supports `OnClickListener` and `OnLongClickListener` |
| 74 | * Supports interception of events using `GestureDetector` and `OnTouchListener` |
| 75 | * Extend to add your own gestures |
| 76 | |
| 77 | #### Easy integration |
| 78 | * Use within a `ViewPager` to create a photo gallery |
| 79 | * Easily restore scale, center and orientation after screen rotation |
| 80 | * Can be extended to add overlay graphics that move and scale with the image |
| 81 | * Handles view resizing and `wrap_content` layout |
| 82 | |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 83 | ## Quick start |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 84 | |
David Morrissey | b11947f | 2017-12-02 08:40:06 +0000 | [diff] [blame] | 85 | **1)** Add `com.davemorrissey.labs:subsampling-scale-image-view:3.9.0` as a dependency in your build.gradle file. |
David Morrissey | 365ccab | 2014-07-31 00:08:41 +0100 | [diff] [blame] | 86 | |
davemorrissey | 55dadbb | 2015-03-22 19:16:07 +0000 | [diff] [blame] | 87 | **2)** Add the view to your layout XML. |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 88 | |
| 89 | <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
| 90 | android:layout_width="match_parent" |
| 91 | android:layout_height="match_parent" > |
| 92 | |
| 93 | <com.davemorrissey.labs.subscaleview.SubsamplingScaleImageView |
| 94 | android:id="@+id/imageView" |
David Morrissey | e11ee3e | 2014-05-30 14:09:41 +0100 | [diff] [blame] | 95 | android:layout_width="match_parent" |
| 96 | android:layout_height="match_parent"/> |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 97 | |
Ed George | 86af5fa | 2015-06-29 17:34:47 +0100 | [diff] [blame] | 98 | </LinearLayout> |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 99 | |
David Morrissey | 9309180 | 2016-11-04 19:13:52 +0000 | [diff] [blame] | 100 | **3a)** Now, in your fragment or activity, set the image resource, asset name or file path. |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 101 | |
| 102 | SubsamplingScaleImageView imageView = (SubsamplingScaleImageView)findViewById(id.imageView); |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 103 | imageView.setImage(ImageSource.resource(R.drawable.monkey)); |
David Morrissey | 36cc125 | 2015-01-12 21:42:02 +0000 | [diff] [blame] | 104 | // ... or ... |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 105 | imageView.setImage(ImageSource.asset("map.png")) |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 106 | // ... or ... |
David Morrissey | 71b94c2 | 2015-03-04 23:10:45 +0000 | [diff] [blame] | 107 | imageView.setImage(ImageSource.uri("/sdcard/DCIM/DSCM00123.JPG")); |
David Morrissey | a0bf802 | 2014-05-30 14:08:31 +0100 | [diff] [blame] | 108 | |
David Morrissey | 9309180 | 2016-11-04 19:13:52 +0000 | [diff] [blame] | 109 | **3b)** Or, if you have a `Bitmap` object in memory, load it into the view. This is unsuitable for large images because it bypasses subsampling - you may get an `OutOfMemoryError`. |
| 110 | |
David Morrissey | 3d3df1c | 2016-11-04 19:14:54 +0000 | [diff] [blame] | 111 | SubsamplingScaleImageView imageView = (SubsamplingScaleImageView)findViewById(id.imageView); |
| 112 | imageView.setImage(ImageSource.bitmap(bitmap)); |
David Morrissey | 9309180 | 2016-11-04 19:13:52 +0000 | [diff] [blame] | 113 | |
David Morrissey | f953721 | 2017-12-11 09:00:33 +0000 | [diff] [blame] | 114 | ## Photo credits |
| 115 | |
| 116 | * San Martino by Luca Bravo, via [unsplash.com](https://unsplash.com/photos/lWAOc0UuJ-A) |
| 117 | * Swiss Road by Ludovic Fremondiere, via [unsplash.com](https://unsplash.com/photos/3XN-BNRDUyY) |
| 118 | |
David Morrissey | 5833e30 | 2014-06-06 22:08:14 +0100 | [diff] [blame] | 119 | ## About |
| 120 | |
David Morrissey | cb93ba0 | 2018-01-19 08:20:05 +0000 | [diff] [blame] | 121 | Copyright 2018 David Morrissey, and licensed under the Apache License, Version 2.0. No attribution is necessary but it's very much appreciated. Star this project if you like it! |