java界面自定义布局_自定义布局执行流程之 画出自己定义的View

以下代码示例针对(Android 2.3)

你玩过植物大战僵尸吗?你玩过愤怒的小鸟吗?你是不是很疑惑精美的UI界面是如何作出来的呢?很明显andriod 自带的控件是不可能做到那样的效果的,这里就用到了对控件、布局的重写。

单从重写控件来看,你会感觉到很简单(只需要覆盖onMeasure()及onLayout()方法)就可以了,但是这两个方法的被谁调用?它的Framework层的布局流程究竟是怎样的,只有搞清楚这些我们才能很好的去重写布局,布上我们的View,从而实现我们想要的效果。

为了更直观地展示布局调用结构,我在这里简略绘制了布局Framework层的类图。

。。。。。。

请看此图

。。。。。。

View.java

// 注意final修饰,该方法永远不会被覆盖,整个布局结构 measure方法唯一

public final void measure(int widthMeasureSpec, int heightMeasureSpec) {

onMeasure(widthMeasureSpec, heightMeasureSpec);

}

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

//注意final修饰,该方法永远不会被覆盖,整个布局结构layout方法唯一

public final void layout(int l, int t, int r, int b) {

boolean changed = setFrame(l, t, r, b);

if (changed || (mPrivateFlags & LAYOUT_REQUIRED) == LAYOUT_REQUIRED) {

onLayout(changed, l, t, r, b);

}

}

protected void onLayout(boolean changed, int left, int top, int right, int bottom) { } //空方法

ViewGroup.java extends View.java

@Override

protected abstract void onLayout(boolean changed, int l, int t, int r, int b);

// 测量该ViewGroup所包含的所有布局

protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {}

protected void measureChild(View child, int parentWidthMeasureSpec,

int parentHeightMeasureSpec) {}

//我会单讲mChildren数组mChildren中的View是如何来的。

public View getChildAt(int index) { return mChildren[index]; }

public int getChildCount() { return mChildrenCount; }

RelativeLayout.java extends ViewGroup.java

//当继承RelativeLayout布局时,我们应当覆盖该方法,以实现测量该布局包含的View,//此处的实现并不能测量所有的View

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

protected void onLayout(boolean changed, int l, int t, int r, int b) {}

private void measureChild(View child, LayoutParams params, int myWidth, int myHeight) {}

//还包含一个重要的内部类,代表RelativeLayout所包含的每一个view大小及位置信息

public static class LayoutParams extends ViewGroup.MarginLayoutParams{

private int mLeft, mTop, mRight, mBottom;

}

下面我要自定义一个布局,定义布局的目的肯定是为了向其内部放置View

CustomGridLayout.java extends RelativeLayout.java

初学者会问,我们到底需要继承RelativeLayout类的哪个方法呢!!

抛去一切,我们自己想象,布局控件需要

第一:控件(View)的大小

第二:控件(View)的位置

第三:知道要放置多少个View

通过熟读文档,我们应该知道:

onMeasure方法负责测量将要放在CustomGridLayout内部的View的大小。

onLayout方法负责分配尺寸及位置给将要放在CustomGridLayout内部的View。

所以很明显,需要我们继承的方法是

1. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

功能:测量该布局所包含的所有View的大小(会在框架层循环取得每一个View,然后测量其大小),该方法会被View.java中的measure方法调用。而measure方法会被

2. protected void onLayout(boolean changed, int l, int t, int r, int b) {}

功能:在相应的位置放置相应的View,该方法会被View.java中的layout方法调用,而layout方法会被谁调用呢?

(1) 调用requestLayout()方法. 该方法内部会执行Object.layout(…)

(2)直接调用 Object.layout(…)

(3) 调用addView(View child, ...)时,

调用addView(...)之前一般需要先调用android.view.View.setLayoutParams(LayoutParams params)

最后我简略分析了一下布局调用的流程调用,如下:

77161a22d98bcc6a76240b6619252966.png

也许有人会问当调用addView时,会和框架层的layout,onLayout,measure, onMeasure等几个布局方法有什么关系,或者说后者几个方法是怎么被触发的,别着急,看见addView(...)的底层实现了吗

public void addView(View child, int index, LayoutParams params) {

// addViewInner() will call child.requestLayout() when setting the new LayoutParams

// therefore, we call requestLayout() on ourselves before, so that the child's

// request will be blocked at our level

requestLayout();

invalidate();

addViewInner(child, index, params, false);

}

很明显,addView在底层调用了requestLayout方法,该方法如时序图所示,会依次触发我们的onMeasure,onLayout方法。

看了这里,你是不是对布局layout,onLayout,measure, onMeasure,requestLayout,等方法的调用清晰多了呢?好了,就先写到这吧,有什么问题欢迎大家共同探讨.

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

当我们在画布上显示自己定义的View时,

canvas.save();

canvas.rotate(float 角度,float X,float Y);//画布以某个点为中心转动的角度

onDrawItem()方法;

canvas.restore();

protected void onDrawItem(Canvas c, int index, Point screenCoords) {

final BalloonItem focusedItem = balloonItems.get(index);

if (index == mTapIndex) {

LinearLayout viewIcon = null;

viewIcon=(LinearLayout)LayoutInflater.from(mContext).inflate(R.layout.balloon_layout, null);

viewIcon.setBackgroundDrawable(mContext.getResources().getDrawable(R.drawable.balloon_blue));

//得到自己定义的布局View之后,要先measure和layout,才能画出

viewIcon.measure(0, 0);

viewIcon.layout(0, 0, viewIcon.getMeasuredWidth(), viewIcon.getMeasuredHeight());

c.save();

c.translate(screenCoords.x - mDensity*10, screenCoords.y - mDensity*20);

viewIcon.draw(c);

c.restore();

Button playBtn = (Button) mT.findViewById(R.id.btn_start);

final TextView title = (TextView) mT.findViewById(R.id.poi_title);

title.setText(focusedItem.getAttractionsName());

mT.measure(0, 0);

mT.layout(0, 0, mT.getMeasuredWidth(), mT.getMeasuredHeight());

c.save();

c.translate(screenCoords.x - mDensity*10, screenCoords.y - playBtn.getMeasuredHeight() - playBtn.getTop() - mDensity*30);

mT.draw(c);

c.restore();

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要设置自定义控件的布局,可以在自定义控件的构造函数中调用 `setLayout()` 方法,将布局设置为您想要的布局。例如,如果您想要使用垂直布局,则可以使用以下代码: ```java public class MyCustomView extends View { public MyCustomView(Context context) { super(context); // 创建一个垂直布局 LinearLayout layout = new LinearLayout(context); layout.setOrientation(LinearLayout.VERTICAL); // 在布局中添加其他控件,如 TextView、Button 等 TextView textView = new TextView(context); textView.setText("Hello, world!"); layout.addView(textView); // 将布局设置为自定义控件的布局 setLayout(layout); } } ``` 请注意,`setLayout()` 方法是自定义控件类中的一个自定义方法,您需要根据自己的需求来实现它。在 `setLayout()` 方法中,您可以使用各种布局类(如 LinearLayout、RelativeLayout 等)来创建您想要的布局,并将其设置为自定义控件的布局。 ### 回答2: 自定义控件是在原有控件的基础上进行扩展或改造,使其具备特定的功能和样式。设置布局是指对自定义控件的布局进行定制,使其具备不同的排列方式和样式。 在自定义控件中设置布局可以通过以下步骤实现: 1. 在自定义控件的布局文件中定义布局容器,例如使用LinearLayout、RelativeLayout等。可以根据需要进行嵌套,构建复杂的布局结构。 2. 在自定义控件的构造方法中引入布局文件,通过LayoutInflater的inflate()方法将布局文件解析为View对象。 3. 通过findViewById()方法获取布局文件中定义的各个组件的引用,以便进行后续的操作。 4. 可以在自定义控件中添加自定义属性,通过在attrs.xml文件中定义属性,在布局文件中引用并在控件的构造方法中获取属性值。例如可以定义属性设置文本大小、颜色等。 5. 在代码中对布局进行动态操作,例如设置组件的大小、位置、背景色等。可以通过LayoutParams来设置布局参数,例如设置宽高、边距等。 6. 在自定义控件的代码中实现各种事件的监听和响应,例如点击事件、长按事件等。可以通过setXXXListener()方法来设置事件监听器,并在监听器中编写对应的逻辑代码。 通过以上步骤,我们可以灵活地设置自定义控件的布局,并且根据需要进行各种样式的定制。这样可以大大提高了控件的灵活性和可重用性,并且使得界面布局更加符合需求。这对于开发中复杂的UI界面和特定的功能需求都是非常有帮助的。 ### 回答3: 自定义控件是Android开发中常用的一种技术,通过自定义控件,我们可以实现更加丰富多样的布局效果。设置布局是自定义控件中的一个重要部分,可以通过设置不同的布局来达到不同的展示效果。 首先,要实现自定义控件的布局设置,我们需要在自定义控件的类中添加相应的方法和属性。常用的方法有onMeasure()和onLayout()。onMeasure()方法用于测量布局的宽高,通过setMeasuredDimension()方法设置测量结果;onLayout()方法用于定义布局中各个子控件的位置和大小。 在布局的设置过程中,我们可以使用常见的布局管理器,如LinearLayout、RelativeLayout等,对子控件进行排列和布局。在使用这些布局管理器时,可以通过LayoutParams属性来设置子控件的位置和大小,如设置权重、边距等。 在自定义控件中,我们还可以使用自定义布局管理器,通过继承ViewGroup类来实现。在自定义布局管理器中,我们可以根据实际需求来定义子控件的排列规则,例如可以实现一个流式布局、网格布局等。 除了传统的布局管理器,我们还可以通过自定义xml布局文件,使用LayoutInflater来加载布局。在加载布局时,可以通过findViewById()方法来获取布局中的子控件,然后进行相应的设置和操作。 总之,自定义控件的布局设置是一个灵活多样的过程,通过灵活运用布局管理器和自定义xml布局文件,结合自定义的测量和排列方法,我们可以实现各种各样独特的布局效果,提升用户体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值