android控件拖动,移动、解决父布局重绘时控件回到原点

时间:2023-11-11 08:14:50

这是主要代码: 保证其params发生改变,相对于父布局的位置就能达到位置移动到原来的位置

// 每次移动都要设置其layout,不然由于父布局可能嵌套listview,当父布局发生改变冲毁(如下拉刷新时)则移动的view会回到原来的位置
RelativeLayout.LayoutParams lpFeedback = new RelativeLayout.LayoutParams(
RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
lpFeedback.leftMargin = v.getLeft();
lpFeedback.topMargin = v.getTop();
lpFeedback.setMargins(v.getLeft(), v.getTop(), 0, 0);
v.setLayoutParams(lpFeedback);

思路是:当view的位置发生改变时,也要相应改变其layoutparams,否则父布局重绘时,由于view的layoutparams没发生改变导致重绘的时候会回到原点,所以只要改变其params就好了

  1. /**
  2. *
  3. * @description 设置意见反馈,用以灰度发布
  4. * @author zhongwr
  5. * @params
  6. * @update 2016年1月12日 下午5:36:07
  7. */
  8. private ImageView getFeedBackView() {
  9. ImageView ivFeedback = new ImageView(BaseActivity.this);
  10. ivFeedback.setImageResource(R.drawable.fuli_feedback);
  11. RelativeLayout.LayoutParams lpFeedback = new RelativeLayout.LayoutParams(
  12. RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
  13. lpFeedback.setMargins(0, 0, 20, 218);
  14. lpFeedback.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM);
  15. lpFeedback.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);
  16. ivFeedback.setLayoutParams(lpFeedback);
  17. ivFeedback.setOnClickListener(new OnClickListener() {
  18. @Override
  19. public void onClick(View v) {// 意见反馈
  20. FeedBackActivity.startInstance(BaseActivity.this);
  21. }
  22. });
  23. return ivFeedback;
  24. }
  25. private boolean isIntercept = false;
  26. /** 按下时的位置控件相对屏幕左上角的位置X */
  27. private int startDownX;
  28. /** 按下时的位置控件距离屏幕左上角的位置Y */
  29. private int startDownY;
  30. /** 控件相对屏幕左上角移动的位置X */
  31. private int lastMoveX;
  32. /** 控件相对屏幕左上角移动的位置Y */
  33. private int lastMoveY;
  34. /**
  35. *
  36. * @description 拖动意见反馈位置
  37. * @author zhongwr
  38. * @params
  39. * @update 2016年1月14日 下午5:08:26
  40. */
  41. private void setFeedBackViewDragTouch(ImageView ivFeedBack) {
  42. ivFeedBack.setOnTouchListener(new OnTouchListener() {
  43. @Override
  44. public boolean onTouch(View v, MotionEvent event) {
  45. int action = event.getAction();
  46. switch (action) {
  47. case MotionEvent.ACTION_DOWN:
  48. startDownX = lastMoveX = (int) event.getRawX();
  49. startDownY = lastMoveY = (int) event.getRawY();
  50. break;
  51. case MotionEvent.ACTION_MOVE:
  52. int dx = (int) event.getRawX() - lastMoveX;
  53. int dy = (int) event.getRawY() - lastMoveY;
  54. int left = v.getLeft() + dx;
  55. int top = v.getTop() + dy;
  56. int right = v.getRight() + dx;
  57. int bottom = v.getBottom() + dy;
  58. if (left < 0) {
  59. left = 0;
  60. right = left + v.getWidth();
  61. }
  62. if (right > mScreenWidth) {
  63. right = mScreenWidth;
  64. left = right - v.getWidth();
  65. }
  66. if (top < 0) {
  67. top = 0;
  68. bottom = top + v.getHeight();
  69. }
  70. if (bottom > mScreenHeight) {
  71. bottom = mScreenHeight;
  72. top = bottom - v.getHeight();
  73. }
  74. v.layout(left, top, right, bottom);
  75. lastMoveX = (int) event.getRawX();
  76. lastMoveY = (int) event.getRawY();
  77. break;
  78. case MotionEvent.ACTION_UP:
  79. int lastMoveDx = Math.abs((int) event.getRawX() - startDownX);
  80. int lastMoveDy = Math.abs((int) event.getRawY() - startDownY);
  81. if (0 != lastMoveDx || 0 != lastMoveDy) {
  82. isIntercept = true;
  83. } else {
  84. isIntercept = false;
  85. }
  86. // 每次移动都要设置其layout,不然由于父布局可能嵌套listview,当父布局发生改变冲毁(如下拉刷新时)则移动的view会回到原来的位置
  87. RelativeLayout.LayoutParams lpFeedback = new RelativeLayout.LayoutParams(
  88. RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
  89. lpFeedback.leftMargin = v.getLeft();
  90. lpFeedback.topMargin = v.getTop();
  91. lpFeedback.setMargins(v.getLeft(), v.getTop(), 0, 0);
  92. v.setLayoutParams(lpFeedback);
  93. break;
  94. }
  95. return isIntercept;
  96. }
  97. });
  98. }

demo:demo