@ -2,10 +2,13 @@ package org.thoughtcrime.securesms.conversation.v2.utilities
import android.content.Context
import android.graphics.Bitmap
import android.graphics.Outline
import android.graphics.drawable.Drawable
import android.net.Uri
import android.util.AttributeSet
import android.util.TypedValue
import android.view.View
import android.view.ViewOutlineProvider
import android.widget.FrameLayout
import androidx.core.view.isVisible
import com.bumptech.glide.load.engine.DiskCacheStrategy
@ -21,18 +24,17 @@ import org.session.libsignal.utilities.ListenableFuture
import org.session.libsignal.utilities.SettableFuture
import org.thoughtcrime.securesms.components.GlideBitmapListeningTarget
import org.thoughtcrime.securesms.components.GlideDrawableListeningTarget
import org.thoughtcrime.securesms.database.model.MmsMessageRecord
import org.thoughtcrime.securesms.mms.DecryptableStreamUriLoader.DecryptableUri
import org.thoughtcrime.securesms.mms.GlideRequest
import org.thoughtcrime.securesms.mms.GlideRequests
import org.thoughtcrime.securesms.mms.Slide
import kotlin.Boolean
import kotlin.Int
import kotlin.getValue
import kotlin.lazy
import kotlin.let
open class ThumbnailView : FrameLayout {
open class ThumbnailView @JvmOverloads constructor (
context : Context ,
attrs : AttributeSet ? = null ,
defStyleAttr : Int = 0
) : FrameLayout ( context , attrs , defStyleAttr ) {
companion object {
private const val WIDTH = 0
private const val HEIGHT = 1
@ -41,29 +43,28 @@ open class ThumbnailView: FrameLayout {
private val binding : ThumbnailViewBinding by lazy { ThumbnailViewBinding . bind ( this ) }
// region Lifecycle
constructor ( context : Context ) : super ( context ) { initialize ( null ) }
constructor ( context : Context , attrs : AttributeSet ) : super ( context , attrs ) { initialize ( attrs ) }
constructor ( context : Context , attrs : AttributeSet , defStyleAttr : Int ) : super ( context , attrs , defStyleAttr ) { initialize ( attrs ) }
val loadIndicator : View by lazy { binding . thumbnailLoadIndicator }
private val dimensDelegate = ThumbnailDimensDelegate ( )
private var slide : Slide ? = null
var radius : Int = 0
private fun initialize ( attrs : AttributeSet ? ) {
if ( attrs != null ) {
val typedArray = context . theme . obtainStyledAttributes ( attrs , R . styleable . ThumbnailView , 0 , 0 )
dimensDelegate . setBounds ( typedArray . getDimensionPixelSize ( R . styleable . ThumbnailView _minWidth , 0 ) ,
typedArray . getDimensionPixelSize ( R . styleable . ThumbnailView _minHeight , 0 ) ,
typedArray . getDimensionPixelSize ( R . styleable . ThumbnailView _maxWidth , 0 ) ,
typedArray . getDimensionPixelSize ( R . styleable . ThumbnailView _maxHeight , 0 ) )
init {
attrs ?. let { context . theme . obtainStyledAttributes ( it , R . styleable . ThumbnailView , 0 , 0 ) }
?. apply {
dimensDelegate . setBounds (
getDimensionPixelSize ( R . styleable . ThumbnailView _minWidth , 0 ) ,
getDimensionPixelSize ( R . styleable . ThumbnailView _minHeight , 0 ) ,
getDimensionPixelSize ( R . styleable . ThumbnailView _maxWidth , 0 ) ,
getDimensionPixelSize ( R . styleable . ThumbnailView _maxHeight , 0 )
)
radius = typedArray . getDimensionPixelSize ( R . styleable . ThumbnailView _thumbnail _radius , 0 )
setRoundedCorners (
getDimensionPixelSize ( R . styleable . ThumbnailView _thumbnail _radius , 0 )
)
typedArray . recycle ( )
recycle ( )
}
}
@ -84,114 +85,118 @@ open class ThumbnailView: FrameLayout {
private fun getDefaultWidth ( ) = maxOf ( layoutParams ?. width ?: 0 , 0 )
private fun getDefaultHeight ( ) = maxOf ( layoutParams ?. height ?: 0 , 0 )
// endregion
// region Interaction
fun setImageResource ( glide : GlideRequests , slide : Slide , isPreview : Boolean , mms : MmsMessageRecord ? ) : ListenableFuture < Boolean > {
return setImageResource ( glide , slide , isPreview , 0 , 0 , mms )
fun setRoundedCorners ( radius : Int ) {
// create an outline provider and clip the whole view to that shape
// that way we can round the image and the background ( and any other artifacts that the view may contain )
val mOutlineProvider = object : ViewOutlineProvider ( ) {
override fun getOutline ( view : View , outline : Outline ) {
// all corners
outline . setRoundRect ( 0 , 0 , view . width , view . height , radius . toFloat ( ) )
}
}
fun setImageResource ( glide : GlideRequests , slide : Slide ,
isPreview : Boolean , naturalWidth : Int ,
naturalHeight : Int , mms : MmsMessageRecord ? ) : ListenableFuture < Boolean > {
outlineProvider = mOutlineProvider
clipToOutline = true
}
val currentSlide = this . slide
fun setImageResource (
glide : GlideRequests ,
slide : Slide ,
isPreview : Boolean
) : ListenableFuture < Boolean > = setImageResource ( glide , slide , isPreview , 0 , 0 )
fun setImageResource (
glide : GlideRequests , slide : Slide ,
isPreview : Boolean , naturalWidth : Int ,
naturalHeight : Int
) : ListenableFuture < Boolean > {
binding . playOverlay . isVisible = ( slide . thumbnailUri != null && slide . hasPlayOverlay ( ) &&
( slide . transferState == AttachmentTransferProgress . TRANSFER _PROGRESS _DONE || isPreview ) )
if ( equals ( currentSlide , slide ) ) {
if ( equals ( this . s lide, slide ) ) {
// don't re-load slide
return SettableFuture ( false )
}
if ( currentSlide != null && currentSlide . fastPreflightId != null && currentSlide . fastPreflightId == slide . fastPreflightId ) {
// not reloading slide for fast preflight
this . slide = slide
}
this . slide = slide
binding . thumbnailLoadIndicator . isVisible = slide . isInProgress
binding . thumbnailDownloadIcon . isVisible = slide . transferState == AttachmentTransferProgress . TRANSFER _PROGRESS _FAILED
binding . thumbnailDownloadIcon . isVisible =
slide . transferState == AttachmentTransferProgress . TRANSFER _PROGRESS _FAILED
dimensDelegate . setDimens ( naturalWidth , naturalHeight )
invalidate ( )
val result = SettableFuture < Boolean > ( )
return SettableFuture < Boolean > ( ) . also {
when {
slide . thumbnailUri != null -> {
buildThumbnailGlideRequest ( glide , slide ) . into ( GlideDrawableListeningTarget ( binding . thumbnailImage , binding . thumbnailLoadIndicator , result ) )
buildThumbnailGlideRequest ( glide , slide ) . into (
GlideDrawableListeningTarget ( binding . thumbnailImage , binding . thumbnailLoadIndicator , it )
)
}
slide . hasPlaceholder ( ) -> {
buildPlaceholderGlideRequest ( glide , slide ) . into ( GlideBitmapListeningTarget ( binding . thumbnailImage , null , result ) )
buildPlaceholderGlideRequest ( glide , slide ) . into (
GlideBitmapListeningTarget ( binding . thumbnailImage , null , it )
)
}
else -> {
glide . clear ( binding . thumbnailImage )
result . set ( false )
it . set ( false )
}
}
}
return result
}
fun buildThumbnailGlideRequest ( glide : GlideRequests , slide : Slide ) : GlideRequest < Drawable > {
val dimens = dimensDelegate . resourceSize ( )
val request = glide . load ( DecryptableUri ( slide . thumbnailUri !! ) )
private fun buildThumbnailGlideRequest (
glide : GlideRequests ,
slide : Slide
) : GlideRequest < Drawable > = glide . load ( DecryptableUri ( slide . thumbnailUri !! ) )
. diskCacheStrategy ( DiskCacheStrategy . NONE )
. let { request ->
if ( dimens [ WIDTH ] == 0 || dimens [ HEIGHT ] == 0 ) {
request . override ( getDefaultWidth ( ) , getDefaultHeight ( ) )
} else {
request . override ( dimens [ WIDTH ] , dimens [ HEIGHT ] )
}
}
. overrideDimensions ( )
. transition ( DrawableTransitionOptions . withCrossFade ( ) )
. centerCrop ( )
return if ( slide . isInProgress ) request else request . apply ( RequestOptions . errorOf ( R . drawable . ic _missing _thumbnail _picture ) )
}
fun buildPlaceholderGlideRequest ( glide : GlideRequests , slide : Slide ) : GlideRequest < Bitmap > {
. transform ( CenterCrop ( ) )
. missingThumbnailPicture ( slide . isInProgress )
val dimens = dimensDelegate . resourceSize ( )
return glide . asBitmap ( )
private fun buildPlaceholderGlideRequest (
glide : GlideRequests ,
slide : Slide
) : GlideRequest < Bitmap > = glide . asBitmap ( )
. load ( slide . getPlaceholderRes ( context . theme ) )
. diskCacheStrategy ( DiskCacheStrategy . NONE )
. let { request ->
if ( dimens [ WIDTH ] == 0 || dimens [ HEIGHT ] == 0 ) {
request . override ( getDefaultWidth ( ) , getDefaultHeight ( ) )
} else {
request . override ( dimens [ WIDTH ] , dimens [ HEIGHT ] )
}
}
. overrideDimensions ( )
. fitCenter ( )
}
open fun clear ( glideRequests : GlideRequests ) {
glideRequests . clear ( binding . thumbnailImage )
slide = null
}
fun setImageResource ( glideRequests : GlideRequests , uri : Uri ) : ListenableFuture < Boolean > {
val future = SettableFuture < Boolean > ( )
var request : GlideRequest < Drawable > = glideRequests . load ( DecryptableUri ( uri ) )
fun setImageResource (
glideRequests : GlideRequests ,
uri : Uri
) : ListenableFuture < Boolean > = glideRequests . load ( DecryptableUri ( uri ) )
. diskCacheStrategy ( DiskCacheStrategy . NONE )
. transition ( DrawableTransitionOptions . withCrossFade ( ) )
. transform ( CenterCrop ( ) )
. intoDrawableTargetAsFuture ( )
request = if ( radius > 0 ) {
request . transforms ( CenterCrop ( ) , RoundedCorners ( radius ) )
} else {
request . transforms ( CenterCrop ( ) )
private fun GlideRequest < Drawable > . intoDrawableTargetAsFuture ( ) =
SettableFuture < Boolean > ( ) . also {
binding . run {
GlideDrawableListeningTarget ( thumbnailImage , thumbnailLoadIndicator , it )
} . let { into ( it ) }
}
request . into ( GlideDrawableListeningTarget ( binding . thumbnailImage , binding . thumbnailLoadIndicator , future ) )
return future
}
private fun < T > GlideRequest < T > . overrideDimensions ( ) =
dimensDelegate . resourceSize ( ) . takeIf { 0 !in it }
?. let { override ( it [ WIDTH ] , it [ HEIGHT ] ) }
?: override ( getDefaultWidth ( ) , getDefaultHeight ( ) )
}
private fun < T > GlideRequest < T > . missingThumbnailPicture (
inProgress : Boolean
) = takeIf { inProgress } ?: apply ( RequestOptions . errorOf ( R . drawable . ic _missing _thumbnail _picture ) )