管灌系统巡查员智能手机App
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
package cc.shinichi.library
 
import android.annotation.SuppressLint
import android.app.Activity
import android.text.TextUtils
import androidx.annotation.DrawableRes
import androidx.annotation.LayoutRes
import cc.shinichi.library.bean.ImageInfo
import cc.shinichi.library.bean.Type
import cc.shinichi.library.tool.common.SLog
import cc.shinichi.library.view.ImagePreviewActivity
import cc.shinichi.library.view.listener.*
import java.lang.ref.WeakReference
 
/**
 * @author 工藤
 * @email qinglingou@gmail.com
 * cc.shinichi.library
 * create at 2018/5/22  09:06
 * description:
 */
class ImagePreview {
    private var contextWeakReference: WeakReference<Activity> = WeakReference(null)
 
    // 图片数据集合
    private var imageInfoList: MutableList<ImageInfo> = mutableListOf()
    private var resImageList: MutableList<Int> = mutableListOf()
 
    // 默认显示第几个
    var index = 0
        private set
 
    // 下载到的文件夹名(根目录中)
    var folderName = ""
        get() {
            if (TextUtils.isEmpty(field)) {
                field = "Download"
            }
            return field
        }
        private set
 
    // 最小缩放倍数
    var minScale = 1.0f
        private set
 
    // 中等缩放倍数
    var mediumScale = 3.0f
        private set
 
    // 最大缩放倍数
    var maxScale = 5.0f
        private set
 
    // 是否显示图片指示器(1/9)
    var isShowIndicator = true
        private set
 
    // 是否显示关闭页面按钮
    var isShowCloseButton = false
        private set
 
    // 是否显示下载按钮
    var isShowDownButton = true
        private set
 
    // 动画持续时间 单位毫秒 ms
    var zoomTransitionDuration = 200
        private set
 
    // 是否启用下拉关闭,默认启用
    var isEnableDragClose = true
        private set
 
    // 是否启用上拉关闭,默认启用
    var isEnableUpDragClose = true
        private set
 
    // 是否忽略缩放启用拉动关闭,默认false,true即忽略
    var isEnableDragCloseIgnoreScale = true
        private set
 
    // 是否启用点击关闭,默认启用
    var isEnableClickClose = true
        private set
 
    // 是否在加载失败时显示toast
    var isShowErrorToast = false
        private set
 
    // 加载策略
    var loadStrategy = LoadStrategy.Auto
        private set
 
    // 长图的展示模式
    var longPicDisplayMode = LongPicDisplayMode.Default
        private set
 
    @LayoutRes
    var previewLayoutResId = R.layout.sh_layout_preview
        private set
 
    var onCustomLayoutCallback: OnCustomLayoutCallback? = null
        private set
 
    @DrawableRes
    var indicatorShapeResId = R.drawable.shape_indicator_bg
        private set
 
    @DrawableRes
    var closeIconResId = R.drawable.ic_action_close
        private set
 
    @DrawableRes
    var closeIconBackgroundResId = -1
        private set
 
    @DrawableRes
    var downIconResId = R.drawable.icon_download_new
        private set
 
    @DrawableRes
    var downIconBackgroundResId = -1
        private set
 
    // 加载失败时的占位图
    @DrawableRes
    var errorPlaceHolder = R.drawable.load_failed
        private set
 
    // 点击和长按事件接口
    var bigImageClickListener: OnBigImageClickListener? = null
        private set
    var bigImageLongClickListener: OnBigImageLongClickListener? = null
        private set
    var bigImagePageChangeListener: OnBigImagePageChangeListener? = null
        private set
    var downloadClickListener: OnDownloadClickListener? = null
        private set
    var downloadListener: OnDownloadListener? = null
        private set
    var onOriginProgressListener: OnOriginProgressListener? = null
        private set
    var onPageFinishListener: OnPageFinishListener? = null
        private set
    var onPageDragListener: OnPageDragListener? = null
        private set
    var finishListener: OnFinishListener? = null
        private set
 
    // 自定义百分比布局layout id
    @LayoutRes
    var progressLayoutId = -1
        private set
 
    // 防止多次快速点击,记录上次打开的时间戳
    private var lastClickTime: Long = 0
 
    fun with(context: Activity): ImagePreview {
        contextWeakReference = WeakReference(context)
        return this
    }
 
    @Deprecated("请使用with(Context context)代替")
    fun setContext(context: Activity): ImagePreview {
        with(context)
        return this
    }
 
    fun getImageInfoList(): MutableList<ImageInfo> {
        return imageInfoList
    }
 
    @Deprecated("请使用setMediaInfoList(List<ImageInfo> imageInfoList)代替")
    fun setImageInfoList(imageInfoList: MutableList<ImageInfo>): ImagePreview {
        setMediaInfoList(imageInfoList)
        return this
    }
 
    /**
     * 支持图片视频混合
     */
    fun setMediaInfoList(mediaList: MutableList<ImageInfo>): ImagePreview {
        this.imageInfoList.clear()
        this.imageInfoList.addAll(mediaList)
        return this
    }
 
    /**
     * 仅支持图片类型
     */
    fun setImageUrlList(imageList: MutableList<String>): ImagePreview {
        var imageInfo: ImageInfo
        imageInfoList.clear()
        for (i in imageList.indices) {
            imageInfo = ImageInfo()
            imageInfo.type = Type.IMAGE
            imageInfo.thumbnailUrl = imageList[i]
            imageInfo.originUrl = imageList[i]
            imageInfoList.add(imageInfo)
        }
        return this
    }
 
    fun setImage(image: String): ImagePreview {
        imageInfoList.clear()
        val imageInfo = ImageInfo()
        imageInfo.type = Type.IMAGE
        imageInfo.thumbnailUrl = image
        imageInfo.originUrl = image
        imageInfoList.add(imageInfo)
        return this
    }
 
//    fun setImageRes(imageResId: Int): ImagePreview {
//        resImageList.clear()
//        resImageList.add(imageResId)
//        return this
//    }
//
//    fun setImageResList(imageResIdList: MutableList<Int>): ImagePreview {
//        resImageList.clear()
//        resImageList.addAll(imageResIdList)
//        return this
//    }
 
    fun setIndex(index: Int): ImagePreview {
        this.index = index
        return this
    }
 
    fun setShowDownButton(showDownButton: Boolean): ImagePreview {
        isShowDownButton = showDownButton
        return this
    }
 
    fun setShowCloseButton(showCloseButton: Boolean): ImagePreview {
        isShowCloseButton = showCloseButton
        return this
    }
 
    fun isShowOriginButton(index: Int): Boolean {
        if (getImageInfoList().isEmpty()) {
            return false
        }
        // 根据不同加载策略,自行判断是否显示查看原图按钮
        val originUrl = imageInfoList[index].originUrl
        val thumbUrl = imageInfoList[index].thumbnailUrl
        // 原图、缩略图url一样,不显示查看原图按钮
        if (originUrl.equals(thumbUrl, ignoreCase = true)) {
            return false
        }
        return when (loadStrategy) {
            LoadStrategy.Default -> {
                true // 手动模式时,根据是否有原图缓存来决定是否显示查看原图按钮
            }
 
            LoadStrategy.NetworkAuto -> {
                false // 强制隐藏查看原图按钮
            }
 
            LoadStrategy.AlwaysThumb -> {
                false // 强制隐藏查看原图按钮
            }
 
            LoadStrategy.AlwaysOrigin -> {
                false // 强制隐藏查看原图按钮
            }
 
            LoadStrategy.Auto -> {
                true // 显示查看原图按钮
            }
        }
    }
 
    /**
     * 不再有效,是否显示查看原图按钮,取决于加载策略,LoadStrategy,会自行判断是否显示。
     */
    @Deprecated("不再支持")
    fun setShowOriginButton(showOriginButton: Boolean): ImagePreview {
        //isShowOriginButton = showOriginButton;
        return this
    }
 
    fun setFolderName(folderName: String): ImagePreview {
        this.folderName = folderName
        return this
    }
 
    /**
     * 当前版本不再支持本设置,双击会在最小和中等缩放值之间进行切换,可手动放大到最大。
     */
    @Deprecated("不再支持")
    fun setScaleMode(scaleMode: Int): ImagePreview {
        //if (scaleMode != MODE_SCALE_TO_MAX_TO_MIN
        //    && scaleMode != MODE_SCALE_TO_MEDIUM_TO_MAX_TO_MIN
        //    && scaleMode != MODE_SCALE_TO_MEDIUM_TO_MIN) {
        //    throw new IllegalArgumentException("only can use one of( MODE_SCALE_TO_MAX_TO_MIN、MODE_SCALE_TO_MEDIUM_TO_MAX_TO_MIN、MODE_SCALE_TO_MEDIUM_TO_MIN )");
        //}
        //this.scaleMode = scaleMode;
        return this
    }
 
    @Deprecated("不再支持,每张图片的缩放由本身的尺寸决定")
    fun setScaleLevel(min: Int, medium: Int, max: Int): ImagePreview {
        if (medium in (min + 1) until max && min > 0) {
            minScale = min.toFloat()
            mediumScale = medium.toFloat()
            maxScale = max.toFloat()
        } else {
            throw IllegalArgumentException("max must greater to medium, medium must greater to min!")
        }
        return this
    }
 
    fun setZoomTransitionDuration(zoomTransitionDuration: Int): ImagePreview {
        require(zoomTransitionDuration >= 0) { "zoomTransitionDuration must greater 0" }
        this.zoomTransitionDuration = zoomTransitionDuration
        return this
    }
 
    fun setLoadStrategy(loadStrategy: LoadStrategy): ImagePreview {
        this.loadStrategy = loadStrategy
        return this
    }
 
    fun setLongPicDisplayMode(longPicDisplayMode: LongPicDisplayMode): ImagePreview {
        this.longPicDisplayMode = longPicDisplayMode
        return this
    }
 
    fun setEnableDragClose(enableDragClose: Boolean): ImagePreview {
        isEnableDragClose = enableDragClose
        return this
    }
 
    fun setEnableUpDragClose(enableUpDragClose: Boolean): ImagePreview {
        isEnableUpDragClose = enableUpDragClose
        return this
    }
 
    fun setEnableDragCloseIgnoreScale(enableDragCloseIgnoreScale: Boolean): ImagePreview {
        isEnableDragCloseIgnoreScale = enableDragCloseIgnoreScale
        return this
    }
 
    fun setEnableClickClose(enableClickClose: Boolean): ImagePreview {
        isEnableClickClose = enableClickClose
        return this
    }
 
    fun setShowErrorToast(showErrorToast: Boolean): ImagePreview {
        isShowErrorToast = showErrorToast
        return this
    }
 
    fun setIndicatorShapeResId(indicatorShapeResId: Int): ImagePreview {
        this.indicatorShapeResId = indicatorShapeResId
        return this
    }
 
    fun setCloseIconResId(@DrawableRes closeIconResId: Int): ImagePreview {
        this.closeIconResId = closeIconResId
        return this
    }
 
    fun setDownIconResId(@DrawableRes downIconResId: Int): ImagePreview {
        this.downIconResId = downIconResId
        return this
    }
 
    fun setCloseIconBackgroundResId(@DrawableRes closeIconBackgroundResId: Int): ImagePreview {
        this.closeIconBackgroundResId = closeIconBackgroundResId
        return this
    }
 
    fun setDownIconBackgroundResId(@DrawableRes downIconBackgroundResId: Int): ImagePreview {
        this.downIconBackgroundResId = downIconBackgroundResId
        return this
    }
 
    fun setShowIndicator(showIndicator: Boolean): ImagePreview {
        isShowIndicator = showIndicator
        return this
    }
 
    fun setErrorPlaceHolder(errorPlaceHolderResId: Int): ImagePreview {
        errorPlaceHolder = errorPlaceHolderResId
        return this
    }
 
    fun setBigImageClickListener(bigImageClickListener: OnBigImageClickListener?): ImagePreview {
        this.bigImageClickListener = bigImageClickListener
        return this
    }
 
    fun setBigImageLongClickListener(bigImageLongClickListener: OnBigImageLongClickListener?): ImagePreview {
        this.bigImageLongClickListener = bigImageLongClickListener
        return this
    }
 
    fun setBigImagePageChangeListener(bigImagePageChangeListener: OnBigImagePageChangeListener?): ImagePreview {
        this.bigImagePageChangeListener = bigImagePageChangeListener
        return this
    }
 
    fun setDownloadClickListener(downloadClickListener: OnDownloadClickListener?): ImagePreview {
        this.downloadClickListener = downloadClickListener
        return this
    }
 
    fun setDownloadListener(downloadListener: OnDownloadListener?): ImagePreview {
        this.downloadListener = downloadListener
        return this
    }
 
    fun setOnPageFinishListener(onPageFinishListener: OnPageFinishListener): ImagePreview {
        this.onPageFinishListener = onPageFinishListener
        return this
    }
 
    fun setOnPageDragListener(onPageDragListener: OnPageDragListener): ImagePreview {
        this.onPageDragListener = onPageDragListener
        return this
    }
 
    fun setOnFinishListener(finishListener: OnFinishListener): ImagePreview {
        this.finishListener = finishListener
        return this
    }
 
    private fun setOnOriginProgressListener(onOriginProgressListener: OnOriginProgressListener): ImagePreview {
        this.onOriginProgressListener = onOriginProgressListener
        return this
    }
 
    fun setProgressLayoutId(
        progressLayoutId: Int,
        onOriginProgressListener: OnOriginProgressListener
    ): ImagePreview {
        setOnOriginProgressListener(onOriginProgressListener)
        this.progressLayoutId = progressLayoutId
        return this
    }
 
    /**
     * 完全自定义预览界面,请参考:R.layout.sh_layout_preview
     * 并保持控件类型、id和其中一致,否则会找不到控件而报错
     */
    fun setPreviewLayoutResId(
        previewLayoutResId: Int,
        onCustomLayoutCallback: OnCustomLayoutCallback?
    ): ImagePreview {
        this.previewLayoutResId = previewLayoutResId
        this.onCustomLayoutCallback = onCustomLayoutCallback
        return this
    }
 
    fun reset() {
        imageInfoList.clear()
        resImageList.clear()
 
        index = 0
 
        folderName = "Download"
 
        minScale = 1.0f
        mediumScale = 3.0f
        maxScale = 5.0f
 
        isShowIndicator = true
        isShowCloseButton = false
        isShowDownButton = true
 
        zoomTransitionDuration = 200
 
        isEnableDragClose = true
        isEnableUpDragClose = true
        isEnableDragCloseIgnoreScale = true
        isEnableClickClose = true
 
        isShowErrorToast = false
 
        loadStrategy = LoadStrategy.Default
        longPicDisplayMode = LongPicDisplayMode.Default
 
        previewLayoutResId = R.layout.sh_layout_preview
 
        onCustomLayoutCallback = null
 
        indicatorShapeResId = R.drawable.shape_indicator_bg
        closeIconResId = R.drawable.ic_action_close
        downIconResId = R.drawable.icon_download_new
        closeIconBackgroundResId = -1
        downIconBackgroundResId = -1
        errorPlaceHolder = R.drawable.load_failed
 
        bigImageClickListener = null
        bigImageLongClickListener = null
        bigImagePageChangeListener = null
        downloadClickListener = null
        downloadListener = null
        onOriginProgressListener = null
        onPageFinishListener = null
        onPageDragListener = null
        finishListener = null
 
        progressLayoutId = -1
        lastClickTime = 0
 
        contextWeakReference.clear()
    }
 
    fun start() {
        if (System.currentTimeMillis() - lastClickTime <= MIN_DOUBLE_CLICK_TIME) {
            SLog.e("ImagePreview", "---忽略多次快速点击---")
            return
        }
        val context = contextWeakReference.get() ?: throw IllegalArgumentException("You must call 'setContext(Context context)' first!")
        if (context.isFinishing || context.isDestroyed) {
            reset()
            return
        }
        require(imageInfoList.isNotEmpty() || resImageList.isNotEmpty()) { "没有数据源!" }
        require(index < imageInfoList.size || index < resImageList.size) { "index out of bound!" }
        lastClickTime = System.currentTimeMillis()
        ImagePreviewActivity.activityStart(context)
    }
 
    /**
     * 手动关闭页面
     */
    fun finish() {
        finishListener?.onFinish()
    }
 
    enum class LoadStrategy {
        /**
         * 仅加载原图;会强制隐藏查看原图按钮
         */
        AlwaysOrigin,
 
        /**
         * 仅加载普清;会强制隐藏查看原图按钮
         */
        AlwaysThumb,
 
        /**
         * 根据网络自适应加载,WiFi原图,流量普清;会强制隐藏查看原图按钮
         */
        NetworkAuto,
 
        /**
         * 手动模式:默认普清,点击按钮再加载原图;会根据原图、缩略图url是否一样来判断是否显示查看原图按钮
         */
        Default,
 
        /**
         * 全自动模式:WiFi原图,流量下默认普清,可点击按钮查看原图
         */
        Auto
    }
 
    enum class LongPicDisplayMode {
        /**
         * 缩小填充,双击拉满,可手动缩放
         */
        Default,
 
        /**
         * 左右拉满,双击缩小,可手动缩放
         * 一般竖屏手机使用
         */
        FillWidth,
    }
 
    private object InnerClass {
        @SuppressLint("StaticFieldLeak")
        val instance = ImagePreview()
    }
 
    companion object {
        @JvmField
        @LayoutRes
        val PROGRESS_THEME_CIRCLE_TEXT = R.layout.sh_default_progress_layout
 
        // 触发双击的最短时间,小于这个时间的直接返回
        private const val MIN_DOUBLE_CLICK_TIME = 1500
 
        @JvmStatic
        val instance: ImagePreview
            get() = InnerClass.instance
    }
}