基本上,我正在实现绘画应用程序,因此我创建了一个自定义视图来处理事件并在画布上绘制。现在,我想实时显示在TextView上绘制的点的坐标。因此,我尝试将ViewModel与LiveData一起使用来进行更新。但是我想不出一种方法,可以将自定义View中的信息提供给ViewModel。你会怎么做?我也尝试过使用BindingAdapter,但是找不到解决方法。
以下是我的自定义视图的代码:
class PaintView(context: Context, attrs: AttributeSet): View(context,attrs) {
var params: LayoutParams
private val path : Path = Path()
private val brush: Paint = Paint()
init {
params = LayoutParams(LayoutParams.MATCH_PARENT,
LayoutParams.WRAP_CONTENT)
brush.isAntiAlias = true
brush.color = Color.BLACK
brush.style = Paint.Style.STROKE
brush.strokeJoin = Paint.Join.ROUND
brush.strokeWidth = 8f
}
override fun onTouchEvent(event: MotionEvent): Boolean {
val pointX = event.x
val pointY = event.y
when (event.action) {
MotionEvent.ACTION_DOWN -> {
path.moveTo(pointX, pointY)
return true
}
MotionEvent.ACTION_MOVE -> {
path.lineTo(pointX, pointY)
}
else -> return false
}
postInvalidate()
return false
}
fun clearPath(){
path.reset()
invalidate()
}
public override fun onDraw(canvas: Canvas) {
canvas.drawPath(path, brush)
}
}
这是我在这篇文章之后发现的绑定适配器:Android data binding view.onTouchListener。我可以记录点的坐标,但是无法将其发送到ViewModel,所以它现在很空。
@BindingAdapter("touchListener")
fun setTouchListener(self: View, bool: Boolean) {
self.setOnTouchListener(
object : View.OnTouchListener {
override fun onTouch(view: View, event: MotionEvent): Boolean {
val pointX = event.x
val pointY = event.y
when (event.action) {
MotionEvent.ACTION_DOWN -> {
Log.i("Points", " ($pointX, $pointY)")
}
MotionEvent.ACTION_MOVE -> {
Log.i("Points", " ($pointX, $pointY)")
}
MotionEvent.ACTION_UP -> {
Log.i("Points", " ($pointX, $pointY)")
}
}
return false
}
})
}
答案 0 :(得分:1)
很抱歉在JAVA中分享此内容...但是我想您可以抓住想法。
在CustomView上,创建一个自定义界面:
public class PaintView {
private OnCoordinateUpdate mCoordinatesListener;
override fun onTouchEvent(event: MotionEvent): Boolean {
val pointX = event.x
val pointY = event.y
....
if(listener != null) {
listener.onUpdate(pointX, pointY);
}
}
...
public void setCoordinatesListener(OnCoordinateUpdate listener) {
mCoordinatesListener = listener;
}
public interface OnCoordinateUpdate {
void onUpdate(int x, int y);
}
}
然后,关于您的活动:
public class MainActivity {
...
mPaintView.setCoordinatesListener(new PaintView.OnCoordinateUpdate {
@Override
void onUpdate(int x, int y) {
if(mTextView != null) {
mTextView.setText("X: " + x + " Y: " + y);
}
}
});
}
这样,您的PaintView
可以随时调用onUpdate
。然后,在您的活动中,每次调用onUpdate
时,TextView的内容都会更新。
我想这是一个好方法,因为您创建了一个自定义界面(OnCoordinateUpdate
),它仅对您的自定义视图有意义。
答案 1 :(得分:0)
最后,它以下列方式工作。 我从前一篇文章中学到了灵感,只是使用绑定的viewmodel调用一个函数然后获取数据。
这是MainActivity代码:
class MainActivity : AppCompatActivity () {
private lateinit var binding: ActivityMainBinding
private lateinit var viewModel: PaintViewModel
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
viewModel = ViewModelProviders.of(this).get(PaintViewModel::class.java)
binding.viewmodel = viewModel
binding.paintView.setCoordinatesListener(object :
PaintView.OnCoordinateUpdate {
override fun onUpdate(x: Float, y: Float) {
binding.textView.text = "X: $x \nY: $y"
binding.viewmodel?.onUpdateCoordinate(x,y)
}
})
binding.clearButton.setOnClickListener { binding.paintView.clearPath() }
}
}
这是ViewModel代码,面临的挑战是在ViewModel中直接调用onUpdate函数。
class PaintViewModel : ViewModel() {
private val _pointX = MutableLiveData<Float>()
private val _pointY = MutableLiveData<Float>()
val pointX : LiveData<Float> = _pointX
val pointY : LiveData<Float> = _pointY
init {
_pointX.value = 0f
_pointY.value = 0f
}
fun onUpdateCoordinate(x: Float, y: Float) {
_pointX.value = x
_pointY.value = y
}
}