开发中遇到这样一个问题。在用户按住录音按钮开始录音时,有些权限控制比较严格的手机,测试时发现装了360或者其他安全软件的时候。使用录音时就会弹出权限确认框。但此时由于OnTouch事件已经触发,录音的动画已经执行,在确认完权限手指离开后,动画仍然在执行。程序并没有执行Up的操作。
后经过测试发现,在用户按下录音按钮后在ACTION—DWON中的录音操作触发权限弹窗,此时OnTouch被强行中止,但是并没有被消费。在用户操作完权限弹窗后继续执行了ACTION—MOVE,ACTION—CANCEL。并没有执行ACTION—UP。因此写在UP中取消动画的操作没有执行。导致手指离开后动画仍然在继续。
但是后来发现一个比较奇怪的现象。并不是每次权限弹窗结束后都会跳转到ACTION-CANCEL中。在用户操作较快时会执行DOWN ,MOVE,UP,在操作较慢时最后会执行CANCEL。目前仍不清楚发生这种情况的原因是什么,如果知道的可以留言告知。
最终的解决方案如下:
@Override
public boolean onTouch(View v, MotionEvent event) {
float y = event.getY();
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
scrollView.setCanScrollable(false);
layoutAudioPrepare.setSelected(true);
txtAudioPrepare.setText("松开结束");
popWinAudio.setStatus(true);
popWinAudio.checkAndShowStartRecording(layoutUploadContentContainer,//在此方法中执行的录音操作
Gravity.CENTER, 0, 0);
initY = y;
//如果出现弹框,程序会卡在checkAndShowStartRecording方法
// 在弹框操作结束后记录时间
firstTime = System.currentTimeMillis();
break;
case MotionEvent.ACTION_MOVE:
layoutAudioPrepare.setSelected(true);
if (initY - y > 100) {
txtAudioPrepare.setText("松开删除");
popWinAudio.setStatus(false);
} else {
txtAudioPrepare.setText("松开结束");
popWinAudio.setStatus(true);
}
break;
case MotionEvent.ACTION_UP:
long lastTime = System.currentTimeMillis();
//当出现弹框时,三个方法几乎同时进行。时间差极小,一般在10ms内
if (lastTime - firstTime< 500) {//正常录音少于500ms时也不记录
scrollView.setCanScrollable(true);
layoutAudioPrepare.setSelected(false);
popWinAudio.setStatus(false);
txtAudioPrepare.setText("按住录音");
popWinAudio.finishRecording();
layoutAudioFinish.setVisibility(View.GONE);
layoutAudioPrepare.setVisibility(View.VISIBLE);
absoluteFilepath = null;
} else {
scrollView.setCanScrollable(true);
layoutAudioPrepare.setSelected(false);
txtAudioPrepare.setText("按住录音");
popWinAudio.finishRecording();
}
break;
case MotionEvent.ACTION_CANCEL://弹出权限框结束,进入此方法,结束录音。
scrollView.setCanScrollable(true);
layoutAudioPrepare.setSelected(false);
popWinAudio.setStatus(false);
txtAudioPrepare.setText("按住录音");
popWinAudio.finishRecording();
layoutAudioFinish.setVisibility(View.GONE);
layoutAudioPrepare.setVisibility(View.VISIBLE);
absoluteFilepath = null;
break;
}
return true;
}