gpt4 book ai didi

android - 何时将 subview 从 XML 添加到 Layout/ViewGroup

转载 作者:塔克拉玛干 更新时间:2023-11-01 21:37:54 24 4
gpt4 key购买 nike

我的问题是:
我想知道 xLayout(或一般的 ViewGroup)何时从 XML 添加 subview ?我所说的“何时”是指在代码的哪个点,在 UI 工具包的“遍历”的哪个“ channel ”中?
我应该覆盖 xLayout 或 ViewGroup 的哪个方法?

我做了功课:我看过"Writing Custom Views For Android"在上次 Google I/O 中提出(由 Adam Powell 和 Romain Guy),我已经阅读了 Adam Powell 对此 Google+ post 的评论.

最佳答案

Looking for the exact point in Android's source code where children are added.



大家可以看看 setContentView(R.layout.some_id)正在做幕后工作。
setContentView(int)来电 PhoneWindow#setContentView(int) - PhoneWindow LinkWindow 的具体实现:
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor();
} else {
mContentParent.removeAllViews();
}
mLayoutInflater.inflate(layoutResID, mContentParent);
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
}

方法 LayoutInflater#inflate(layoutResID, mContentParent)最终调用 ViewGroup#addView(View, LayoutParams)mContentParent .介于两者之间, subview

I want to know what happens exactly after I set content view to an XML file that contains a custom view. Afer the constructor there has to be a part in the code where the custom view "parse/read/inflate/convert" XML-declared child views to actual views ! (comment by JohnTube)



模糊性:从 JohnTube 的评论来看,他似乎更感兴趣的是了解自定义 View 是如何膨胀的。要知道这一点,我们必须看看 LayoutInflater 的工作原理。 Link .

所以, Which method of xLayout or ViewGroup should I override ? 的答案是 ViewGroup#addView(View, LayoutParams) .请注意,此时,所有常规/自定义 View 的膨胀已经发生。

自定义 View 的膨胀:
LayoutInflater中的以下方法是 addView(View, LayoutParams)在父/根上调用:

注:调用 mLayoutInflater.inflate(layoutResID, mContentParent);PhoneWindow#setContentView(int)链到这个。这里 mContentParentDecorView : 可通过 getWindow().getDecorView() 访问的 View .
// Inflate a new view hierarchy from the specified XML node.
public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot)

// Recursive method used to descend down the xml hierarchy and instantiate views,
// instantiate their children, and then call onFinishInflate().
void rInflate(XmlPullParser parser, View parent, final AttributeSet attrs,
boolean finishInflate) throws XmlPullParserException, IOException

此方法(以及递归 rInflate(XmlPullParser, View, AttributeSet, boolean) )中感兴趣的调用是:
temp = createViewFromTag(root, name, attrs);

让我们看看 createViewFromTag(...)是在做:
View createViewFromTag(View parent, String name, AttributeSet attrs) {
....
....
if (view == null) {
if (-1 == name.indexOf('.')) {
view = onCreateView(parent, name, attrs);
} else {
view = createView(name, null, attrs);
}
}
....
}

period(.)决定是否 onCreateView(...)createView(...)叫做。

为什么要做这个检查?因为一个 View定义于 android.view , android.widgetandroid.webkit包通过其类名访问。例如:
android.widget: Button, TextView etc.

android.view: ViewStub. SurfaceView, TextureView etc.

android.webkit: WebView

当遇到这些 View 时, onCreateView(parent, name, attrs)叫做。此方法实际上链接到 createView(...) :
protected View onCreateView(String name, AttributeSet attrs) throws ClassNotFoundException {
return createView(name, "android.view.", attrs);
}

这将处理 SurfaceView , TextureView以及在 android.view 中定义的其他 View 包裹。如果您有兴趣了解 TextView, Button etc.正在处理,看 PhoneLayoutInflater Link - 它扩展了 LayoutInflater并覆盖 onCreateView(...)检查是否 android.widgetandroid.webkit是预期的包名称。事实上,调用 getLayoutInflater()为您提供 PhoneLayoutInflater 的实例.这就是为什么如果你要继承 LayoutInflater , 你甚至不能为最简单的布局充气 - 因为 LayoutInflater只能处理来自 android.view 的 View 包裹。

无论如何,我离题了。这个额外的位发生在常规 View 中——没有 period(.)在他们的定义中。自定义 View 的名称中有句点 - com.my.package.CustomView .这就是 LayoutInflater区分两者。

因此,在常规 View (例如 Button)的情况下, prefixandroid.widget将作为第二个参数传递 - 对于自定义 View ,这将是 null . prefix然后与 name 一起使用获取该特定 View 类的构造函数。自定义 View 不需要这个,因为它们的 name已经完全合格了。我想这样做是为了方便。否则,您会以这种方式定义布局:
<android.widget.LinearLayout
...
... />

(虽然它是合法的......)

此外,这也是来自支持库(例如 )的 View 必须使用完全限定名称的原因。

顺便说一句,如果您确实想将布局编写为:
<MyCustomView ../>

你所要做的就是扩展 LayoutInflater 并添加你的包名 com.my.package.到膨胀期间检查的字符串列表。查看 PhoneLayoutInflater寻求帮助。

让我们看看自定义 View 和常规 View 在最后阶段会发生什么 - createView(...) :
public final View createView(String name, String prefix, AttributeSet attrs)
throws ClassNotFoundException, InflateException {

// Try looking for the constructor in cache
Constructor<? extends View> constructor = sConstructorMap.get(name);
Class<? extends View> clazz = null;

try {
if (constructor == null) {
// Class not found in the cache, see if it's real, and try to add it
clazz = mContext.getClassLoader().loadClass(
prefix != null ? (prefix + name) : name).asSubclass(View.class);
....
// Get constructor
constructor = clazz.getConstructor(mConstructorSignature);
sConstructorMap.put(name, constructor);
} else {
....
}

Object[] args = mConstructorArgs;
args[1] = attrs;

// Obtain an instance
final View view = constructor.newInstance(args);
....

// We finally have a view!
return view;
}
// A bunch of catch blocks:
- if the only constructor defined is `CustomView(Context)` - NoSuchMethodException
- if `com.my.package.CustomView` doesn't extend View - ClassCastException
- if `com.my.package.CustomView` is not found - ClassNotFoundException

// All these catch blocks throw the often seen `InflateException`.
}

... 一个 View出生。

关于android - 何时将 subview 从 XML 添加到 Layout/ViewGroup,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17334078/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com