Android waterfall photo wall implementation, experience the beauty of irregular arrangement Demo

Android waterfall photo wall implementation, experience the beauty of irregular arrangement Demo

Source code introduction

Android waterfall photo wall implementation, experience the beauty of irregular arrangement Demo.
Source code running screenshot

Source code snippet

  1. /**
  2. * Custom ScrollView, in which images are added dynamically.
  3. *
  4. * @author guolin
  5. */  
  6. public   class MyScrollView extends ScrollView implements OnTouchListener {
  7.   
  8. /**
  9. * The number of images to load per page
  10. */  
  11. public   static   final   int PAGE_SIZE = 15 ;
  12.   
  13. /**
  14. * Record the page currently loaded
  15. */  
  16. private   int page;
  17.   
  18. /**
  19. * The width of each column
  20. */  
  21. private   int columnWidth;
  22.   
  23. /**
  24. * The current height of the first column
  25. */  
  26. private   int firstColumnHeight;
  27.   
  28. /**
  29. * The current height of the second column
  30. */  
  31. private   int secondColumnHeight;
  32.   
  33. /**
  34. * The current height of the third column
  35. */  
  36. private   int thirdColumnHeight;
  37.   
  38. /**
  39. * Whether the layout has been loaded once. The initialization in onLayout only needs to be loaded once.
  40. */  
  41. private   boolean loadOnce;
  42.   
  43. /**
  44. * Tools for managing pictures
  45. */  
  46. private ImageLoader imageLoader;
  47.   
  48. /**
  49. * Layout of the first column
  50. */  
  51. private LinearLayout firstColumn;
  52.   
  53. /**
  54. * Layout of the second column
  55. */  
  56. private LinearLayout secondColumn;
  57.   
  58. /**
  59. * Layout of the third column
  60. */  
  61. private LinearLayout thirdColumn;
  62.   
  63. /**
  64. * Record all tasks that are downloading or waiting to download.
  65. */  
  66. private   static Set<loadimagetask> taskCollection;
  67.   
  68. /**
  69. * The direct child layout under MyScrollView.
  70. */  
  71. private   static View scrollLayout;
  72.   
  73. /**
  74. * The height of the MyScrollView layout.
  75. */  
  76. private   static   int scrollViewHeight;
  77.   
  78. /**
  79. * Record the vertical scroll distance.
  80. */  
  81. private   static   int lastScrollY = - 1 ;
  82.   
  83. /**
  84. * Record all images on the interface so that you can control the release of images at any time.
  85. */  
  86. private List<imageview> imageViewList = new ArrayList<imageview>();
  87.   
  88. /**
  89. * In the Handler, check the image visibility and load more images.
  90. */  
  91. private   static Handler handler = new Handler() {
  92.   
  93. public   void handleMessage(android.os.Message msg) {
  94. MyScrollView myScrollView = (MyScrollView) msg.obj;
  95. int scrollY = myScrollView.getScrollY();
  96. // If the current scroll position is the same as the last time, it means that scrolling has stopped  
  97. if (scrollY == lastScrollY) {
  98. // When you scroll to the bottom and there is no downloading task currently, start loading the next page of images  
  99. if (scrollViewHeight + scrollY >= scrollLayout.getHeight()
  100. && taskCollection.isEmpty()) {
  101. myScrollView.loadMoreImages();
  102. }
  103. myScrollView.checkVisibility();
  104. } else {
  105. lastScrollY = scrollY;
  106. Message message = new Message();
  107. message.obj = myScrollView;
  108. // Determine the scroll position again after 5 milliseconds  
  109. handler.sendMessageDelayed(message, 5 );
  110. }
  111. };
  112.   
  113. };
  114.   
  115. /**
  116. * Constructor of MyScrollView.
  117. *
  118. * @param context
  119. * @param attrs
  120. */  
  121. public MyScrollView(Context context, AttributeSet attrs) {
  122. super (context, attrs);
  123. imageLoader = ImageLoader.getInstance();
  124. taskCollection = new HashSet<loadimagetask>();
  125. setOnTouchListener( this );
  126. }
  127.   
  128. /**
  129. * Perform some key initialization operations, get the height of MyScrollView, and get the width of the first column. And start loading the first page of pictures here.
  130. */  
  131. @Override  
  132. protected   void onLayout( boolean changed, int l, int t, int r, int b) {
  133. super .onLayout(changed, l, t, r, b);
  134. if (changed && !loadOnce) {
  135. scrollViewHeight = getHeight();
  136. scrollLayout = getChildAt( 0 );
  137. firstColumn = (LinearLayout) findViewById(R.id.first_column);
  138. secondColumn = (LinearLayout) findViewById(R.id.second_column);
  139. thirdColumn = (LinearLayout) findViewById(R.id.third_column);
  140. columnWidth = firstColumn.getWidth();
  141. loadOnce = true ;
  142. loadMoreImages();
  143. }
  144. }
  145.   
  146. /**
  147. * Listen for user touch screen events and start scrolling detection if the user's finger leaves the screen.
  148. */  
  149. @Override  
  150. public   boolean onTouch(View v, MotionEvent event) {
  151. if (event.getAction() == MotionEvent.ACTION_UP) {
  152. Message message = new Message();
  153. message.obj = this ;
  154. handler.sendMessageDelayed(message, 5 );
  155. }
  156. return   false ;
  157. }
  158.   
  159. /**
  160. * Start loading the next page of pictures. Each picture will start an asynchronous thread to download.
  161. */  
  162. public   void loadMoreImages() {
  163. if (hasSDCard()) {
  164. int startIndex = page * PAGE_SIZE;
  165. int endIndex = page * PAGE_SIZE + PAGE_SIZE;
  166. if (startIndex < Images.imageUrls.length) {
  167. Toast.makeText(getContext(), "Loading..." , Toast.LENGTH_SHORT)
  168. .show();
  169. if (endIndex > Images.imageUrls.length) {
  170. endIndex = Images.imageUrls.length;
  171. }
  172. for ( int i = startIndex; i < endIndex; i++) {
  173. LoadImageTask task = new LoadImageTask();
  174. taskCollection.add(task);
  175. task.execute(Images.imageUrls[i]);
  176. }
  177. page++;
  178. } else {
  179. Toast.makeText(getContext(), "No more pictures" , Toast.LENGTH_SHORT)
  180. .show();
  181. }
  182. } else {
  183. Toast.makeText(getContext(), "SD card not found" , Toast.LENGTH_SHORT).show();
  184. }
  185. }
  186.   
  187. /**
  188. * Traverse each image in imageViewList and check the visibility of the image. If the image has left the visible range of the screen, replace the image with an empty image.
  189. */  
  190. public   void checkVisibility() {
  191. for ( int i = 0 ; i < imageViewList.size(); i++) {
  192. ImageView imageView = imageViewList.get(i);
  193. int borderTop = (Integer) imageView.getTag(R.string.border_top);
  194. int borderBottom = (Integer) imageView
  195. .getTag(R.string.border_bottom);
  196. if (borderBottom > getScrollY()
  197. && borderTop < getScrollY() + scrollViewHeight) {
  198. String imageUrl = (String) imageView.getTag(R.string.image_url);
  199. Bitmap bitmap = imageLoader.getBitmapFromMemoryCache(imageUrl);
  200. if (bitmap != null ) {
  201. imageView.setImageBitmap(bitmap);
  202. } else {
  203. LoadImageTask task = new LoadImageTask(imageView);
  204. task.execute(imageUrl);
  205. }
  206. } else {
  207. imageView.setImageResource(R.drawable.empty_photo);
  208. }
  209. }
  210. }
  211.   
  212. /**
  213. * Determine whether the phone has an SD card.
  214. *
  215. * @return Returns true if there is an SD card, otherwise returns false.
  216. */  
  217. private   boolean hasSDCard() {
  218. return Environment.MEDIA_MOUNTED.equals(Environment
  219. .getExternalStorageState());
  220. }
  221.   
  222. /**
  223. * The task of asynchronously downloading images.
  224. *
  225. * @author guolin
  226. */  
  227. class LoadImageTask extends AsyncTask<string, void ,= "" bitmap= "" > {
  228.   
  229. /**
  230. * URL of the image
  231. */  
  232. private String mImageUrl;
  233.   
  234. /**
  235. * Reusable ImageView
  236. */  
  237. private ImageView mImageView;
  238.   
  239. public LoadImageTask() {
  240. }
  241.   
  242. /**
  243. * Pass in a reusable ImageView
  244. *
  245. * @param imageView
  246. */  
  247. public LoadImageTask(ImageView imageView) {
  248. mImageView = imageView;
  249. }
  250.   
  251. @Override  
  252. protected Bitmap doInBackground(String... params) {
  253. mImageUrl = params[ 0 ];
  254. Bitmap imageBitmap = imageLoader
  255. .getBitmapFromMemoryCache(mImageUrl);
  256. if (imageBitmap == null ) {
  257. imageBitmap = loadImage(mImageUrl);
  258. }
  259. return imageBitmap;
  260. }
  261.   
  262. @Override  
  263. protected   void onPostExecute(Bitmap bitmap) {
  264. if (bitmap != null ) {
  265. double ratio = bitmap.getWidth() / (columnWidth * 1.0 );
  266. int scaledHeight = ( int ) (bitmap.getHeight() / ratio);
  267. addImage(bitmap, columnWidth, scaledHeight);
  268. }
  269. taskCollection.remove( this );
  270. }
  271.   
  272. /**
  273. * Load the image according to the URL passed in. If the image already exists in the SD card, read it directly from the SD card, otherwise download it from the Internet.
  274. *
  275. * @param imageUrl
  276. * URL of the image
  277. * @return The image loaded into memory.
  278. */  
  279. private Bitmap loadImage(String imageUrl) {
  280. File imageFile = new File(getImagePath(imageUrl));
  281. if (!imageFile.exists()) {
  282. downloadImage(imageUrl);
  283. }
  284. if (imageUrl != null ) {
  285. Bitmap bitmap = ImageLoader.decodeSampledBitmapFromResource(
  286. imageFile.getPath(), columnWidth);
  287. if (bitmap != null ) {
  288. imageLoader.addBitmapToMemoryCache(imageUrl, bitmap);
  289. return bitmap;
  290. }
  291. }
  292. return   null ;
  293. }
  294.   
  295. /**
  296. * Add an image to the ImageView
  297. *
  298. * @param bitmap
  299. * Pictures to be added
  300. * @param imageWidth
  301. * The width of the image
  302. * @param imageHeight
  303. * The height of the image
  304. */  
  305. private   void addImage(Bitmap bitmap, int imageWidth, int imageHeight) {
  306. LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(
  307. imageWidth, imageHeight);
  308. if (mImageView != null ) {
  309. mImageView.setImageBitmap(bitmap);
  310. } else {
  311. ImageView imageView = new ImageView(getContext());
  312. imageView.setLayoutParams(params);
  313. imageView.setImageBitmap(bitmap);
  314. imageView.setScaleType(ScaleType.FIT_XY);
  315. imageView.setPadding( 5 , 5 , 5 , 5 );
  316. imageView.setTag(R.string.image_url, mImageUrl);
  317. findColumnToAdd(imageView, imageHeight).addView(imageView);
  318. imageViewList.add(imageView);
  319. }
  320. }
  321.   
  322. /**
  323. * Find the column where the image should be added. The principle is to judge the height of the three columns, and the column with the smallest current height is the one that should be added.
  324. *
  325. * @param imageView
  326. * @param imageHeight
  327. * @return The column where the image should be added
  328. */  
  329. private LinearLayout findColumnToAdd(ImageView imageView,
  330. int imageHeight) {
  331. if (firstColumnHeight <= secondColumnHeight) {
  332. if (firstColumnHeight <= thirdColumnHeight) {
  333. imageView.setTag(R.string.border_top, firstColumnHeight);
  334. firstColumnHeight += imageHeight;
  335. imageView.setTag(R.string.border_bottom, firstColumnHeight);
  336. return firstColumn;
  337. }
  338. imageView.setTag(R.string.border_top, thirdColumnHeight);
  339. thirdColumnHeight += imageHeight;
  340. imageView.setTag(R.string.border_bottom, thirdColumnHeight);
  341. return thirdColumn;
  342. } else {
  343. if (secondColumnHeight <= thirdColumnHeight) {
  344. imageView.setTag(R.string.border_top, secondColumnHeight);
  345. secondColumnHeight += imageHeight;
  346. imageView
  347. .setTag(R.string.border_bottom, secondColumnHeight);
  348. return secondColumn;
  349. }
  350. imageView.setTag(R.string.border_top, thirdColumnHeight);
  351. thirdColumnHeight += imageHeight;
  352. imageView.setTag(R.string.border_bottom, thirdColumnHeight);
  353. return thirdColumn;
  354. }
  355. }
  356.   
  357. /**
  358. * Download the image to the SD card for cache.
  359. *
  360. * @param imageUrl
  361. * The URL of the image.
  362. */  
  363. private   void downloadImage(String imageUrl) {
  364. if (Environment.getExternalStorageState().equals(
  365. Environment.MEDIA_MOUNTED)) {
  366. Log.d( "TAG" , "monted sdcard" );
  367. } else {
  368. Log.d( "TAG" , "has no sdcard" );
  369. }
  370. HttpURLConnection con = null ;
  371. FileOutputStream fos = null ;
  372. BufferedOutputStream bos = null ;
  373. BufferedInputStream bis = null ;
  374. File imageFile = null ;
  375. try {
  376. URL url = new URL(imageUrl);
  377. con = (HttpURLConnection) url.openConnection();
  378. con.setConnectTimeout( 5 * 1000 );
  379. con.setReadTimeout( 15 * 1000 );
  380. con.setDoInput( true );
  381. con.setDoOutput( true );
  382. bis = new BufferedInputStream(con.getInputStream());
  383. imageFile = new File(getImagePath(imageUrl));
  384. fos = new FileOutputStream(imageFile);
  385. bos = new BufferedOutputStream(fos);
  386. byte [] b = new   byte [ 1024 ];
  387. int length;
  388. while ((length = bis.read(b)) != - 1 ) {
  389. bos.write(b, 0 , length);
  390. bos.flush();
  391. }
  392. } catch (Exception e) {
  393. e.printStackTrace();
  394. finally
  395. try {
  396. if (bis != null ) {
  397. bis.close();
  398. }
  399. if (bos != null ) {
  400. bos.close();
  401. }
  402. if (con != null ) {
  403. con.disconnect();
  404. }
  405. } catch (IOException e) {
  406. e.printStackTrace();
  407. }
  408. }
  409. if (imageFile != null ) {
  410. Bitmap bitmap = ImageLoader.decodeSampledBitmapFromResource(
  411. imageFile.getPath(), columnWidth);
  412. if (bitmap != null ) {
  413. imageLoader.addBitmapToMemoryCache(imageUrl, bitmap);
  414. }
  415. }
  416. }
  417.   
  418. /**
  419. * Get the local storage path of the image.
  420. *
  421. * @param imageUrl
  422. * The URL of the image.
  423. * @return The local storage path of the image.
  424. */  
  425. private String getImagePath(String imageUrl) {
  426. int lastSlashIndex = imageUrl.lastIndexOf( "/" );
  427. String imageName = imageUrl.substring(lastSlashIndex + 1 );
  428. String imageDir = Environment.getExternalStorageDirectory()
  429. .getPath() + "/PhotoWallFalls/" ;
  430. File file = new File(imageDir);
  431. if (!file.exists()) {
  432. file.mkdirs();
  433. }
  434. String imagePath = imageDir + imageName;
  435. return imagePath;
  436. }
  437. }
  438.   
  439. }</string,></loadimagetask></imageview></imageview></loadimagetask>

Source code download: http://download..com/data/1980596

<<:  Chukong Technology officially releases "One-stop solution for game development"

>>:  Don't step on the white block game source code

Recommend

How much does it cost to make a transportation mini program in Hegang?

The factors affecting the quotation of the Hegang...

Develop Android Wear Apps with Xamarin and Visual Studio

Build a development environment The first thing w...

What is the core of the brand breakthrough strategy of the To B industry?

The influence of brands has been weak for a long ...

Definition of server hosting

Definition of server rental: Server rental means ...

New analysis of Tik Tok’s marketing strategy!

1. Long-term management returns to its essence In...

Ps+Ai double major! The first compulsory course for designers

Ps+Ai double major! The first compulsory course f...

Facebook low-cost traffic generation skills!

Marketers often look for ways to get more Faceboo...

The pitfalls of the App Store that you don’t know about!

Preface Generally, game developers and game publi...