-
Notifications
You must be signed in to change notification settings - Fork 23
/
MaskActivity.kt
202 lines (168 loc) · 7.47 KB
/
MaskActivity.kt
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
package net.aiscope.gdd_app.ui.mask
import android.content.Intent
import android.graphics.Bitmap
import android.graphics.drawable.LayerDrawable
import android.os.Bundle
import android.view.Gravity
import android.view.View
import android.widget.AdapterView
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import androidx.lifecycle.lifecycleScope
import dagger.android.AndroidInjection
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext
import net.aiscope.gdd_app.R
import net.aiscope.gdd_app.databinding.ActivityMaskBinding
import net.aiscope.gdd_app.extensions.writeToFile
import net.aiscope.gdd_app.ui.CaptureFlow
import net.aiscope.gdd_app.ui.attachCaptureFlowToolbar
import net.aiscope.gdd_app.ui.metadata.MetadataActivity
import net.aiscope.gdd_app.ui.showConfirmBackDialog
import net.aiscope.gdd_app.ui.util.BitmapReader
import net.aiscope.gdd_app.ui.util.BitmapReader.MAX_TEXTURE_SIZE
import net.aiscope.gdd_app.ui.util.MaximumSizeDownSampling
import java.io.File
import javax.inject.Inject
@Suppress("TooManyFunctions")
class MaskActivity : AppCompatActivity(), MaskView, CaptureFlow {
companion object {
const val EXTRA_DISEASE_NAME = "net.aiscope.gdd_app.ui.mask.MaskActivity.EXTRA_DISEASE_NAME"
const val EXTRA_IMAGE_NAME = "net.aiscope.gdd_app.ui.mask.MaskActivity.EXTRA_IMAGE_NAME"
const val EXTRA_MASK_NAME = "net.aiscope.gdd_app.ui.mask.MaskActivity.EXTRA_MASK_NAME"
const val EXTRA_MASK_PATH = "net.aiscope.gdd_app.ui.mask.MaskActivity.EXTRA_MASK_PATH"
}
@Inject
lateinit var presenter: MaskPresenter
private val brushDiseaseStages by lazy { presenter.brushDiseaseStages }
private var currentBrushColor: Int = 0
private val selectStagePopup by lazy { composeSelectStagePopup() }
private lateinit var binding: ActivityMaskBinding
override fun onCreate(savedInstanceState: Bundle?) {
AndroidInjection.inject(this)
super.onCreate(savedInstanceState)
val diseaseName = checkNotNull(intent.getStringExtra(EXTRA_DISEASE_NAME))
val imageNameExtra = checkNotNull(intent.getStringExtra(EXTRA_IMAGE_NAME))
val maskNameExtra = checkNotNull(intent.getStringExtra(EXTRA_MASK_NAME))
val maskPathExtra = intent.getStringExtra(EXTRA_MASK_PATH)
binding = ActivityMaskBinding.inflate(layoutInflater)
with(binding) {
setContentView(root)
setSupportActionBar(toolbarLayout.toolbar)
attachCaptureFlowToolbar(toolbarLayout.toolbar)
presenter.start(diseaseName, imageNameExtra, maskPathExtra)
currentBrushColor = savedInstanceState?.getInt("currentBrushColor")
?: brushDiseaseStages[0].maskColor
refreshBrushDrawableColor()
photoMaskView.initBrushColor(currentBrushColor)
toolsRadioGroup.setOnCheckedChangeListener { _, checkedId ->
when (checkedId) {
drawBtn.id -> photoMaskView.drawMode()
zoomBtn.id -> photoMaskView.zoomMode()
eraseBtn.id -> photoMaskView.eraseMode()
}
}
undoBtn.setOnClickListener {
photoMaskView.undo()
setEnabled(undoBtn, photoMaskView.undoAvailable())
setEnabled(redoBtn, true)
}
redoBtn.setOnClickListener {
photoMaskView.redo()
setEnabled(undoBtn, true)
setEnabled(redoBtn, photoMaskView.redoAvailable())
}
stagesBtn.setOnClickListener { selectStagePopup.show() }
getBitmapBtn.setOnClickListener {
presenter.handleCaptureBitmap(maskNameExtra, isEmptyMaskBitmap())
}
photoMaskView.onMaskingActionFinishedListener = View.OnTouchListener { _, _ ->
setEnabled(undoBtn, photoMaskView.undoAvailable())
setEnabled(redoBtn, photoMaskView.redoAvailable())
false
}
}
}
override fun takeMask(maskName: String, onPhotoReceived: suspend (File?) -> Unit) {
val bmp = binding.photoMaskView.getMaskBitmap()
lifecycleScope.launch {
val dest = File(this@MaskActivity.filesDir, "${maskName}.png")
bmp.writeToFile(dest, Bitmap.CompressFormat.PNG)
onPhotoReceived(dest)
}
}
override fun goToMetadata() {
val intent = Intent(this, MetadataActivity::class.java)
startActivity(intent)
//Finish the activity to avoid keeping all the bitmaps in memory
finish()
}
override fun notifyImageCouldNotBeTaken() {
Toast.makeText(this, getString(R.string.image_could_not_be_taken), Toast.LENGTH_SHORT)
.show()
}
override fun initPhotoMaskView(imagePath: String, maskPath: String?) {
lifecycleScope.launch {
val bmp = readImage(imagePath)
binding.photoMaskView.setImageBitmap(bmp)
maskPath?.let {
val mask = readImage(maskPath, mutable = true)
mask.let { binding.photoMaskView.setMaskBitmap(it) }
}
}
}
override fun onSaveInstanceState(outState: Bundle) {
super.onSaveInstanceState(outState)
outState.putInt("currentBrushColor", currentBrushColor)
outState.putInt("undo_btn-visibility", binding.undoBtn.visibility)
outState.putInt("redo_btn-visibility", binding.redoBtn.visibility)
}
override fun onRestoreInstanceState(savedInstanceState: Bundle) {
super.onRestoreInstanceState(savedInstanceState)
currentBrushColor = savedInstanceState.getInt("currentBrushColor")
binding.undoBtn.visibility = savedInstanceState.getInt("undo_btn-visibility", View.INVISIBLE)
binding.redoBtn.visibility = savedInstanceState.getInt("redo_btn-visibility", View.INVISIBLE)
}
override fun onBackPressed() {
showConfirmBackDialog()
}
private fun setBrushColor(color: Int) {
currentBrushColor = color
refreshBrushDrawableColor()
binding.photoMaskView.setBrushColor(color)
}
private fun composeSelectStagePopup(): SelectStagePopup {
return SelectStagePopup(
this,
presenter.brushDiseaseStages,
binding.stagesBtn,
AdapterView.OnItemClickListener() { _, _, position, _ ->
setBrushColor(presenter.brushDiseaseStages[position].maskColor)
}).apply {
setDropDownGravity(Gravity.TOP)
}
}
private fun refreshBrushDrawableColor() {
val stageBtnDrawable = binding.drawBtn.compoundDrawables[0] as LayerDrawable
stageBtnDrawable.getDrawable(0).apply {
setTint(currentBrushColor)
}
}
private fun setEnabled(view: View, enabled: Boolean) {
view.visibility = if (enabled) View.VISIBLE else View.INVISIBLE
}
private suspend fun readImage(filepath: String, mutable: Boolean = false): Bitmap = withContext( Dispatchers.IO) {
BitmapReader.decodeSampledBitmapFromResource(
File(filepath),
MaximumSizeDownSampling(MAX_TEXTURE_SIZE, MAX_TEXTURE_SIZE),
mutable = mutable
)
}
private fun isEmptyMaskBitmap(): Boolean {
val maskBitmap = binding.photoMaskView.getMaskBitmap()
val emptyBitmap: Bitmap =
Bitmap.createBitmap(maskBitmap.width, maskBitmap.height, maskBitmap.config)
return maskBitmap.sameAs(emptyBitmap)
}
}