- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我一直在使用 Android v4 支持库中的 FragmentTabHost 一段时间,遇到了一个我无法解决的严重问题。我的目标要求如下。
1) Fragment
将包括也从 Fragments
构建的选项卡,其中还包括嵌套的 Fragments
,具有如下层次结构:
android.support.v4.app.Fragment
(HostFragment)
android.support.v4.app.FragmentTabHost
(TabHost)
android.support.v4.app.Fragment
(Tab1Fragment)
android.support.v4.app.Fragment
(Tab1Fragment1)android.support.v4.app.Fragment
(Tab1Fragment2)android.support.v4.app.Fragment
(Tab1Fragment3) android.support.v4.app.Fragment
(Tab2Fragment)
android.support.v4.app.Fragment
(Tab2Fragment1)android.support.v4.app.Fragment
(Tab2Fragment2) HostFragment + TabHost
设置是根据文档中的说明完成的 here .
2) 在屏幕旋转时保留 HostFragment
的状态,因为重新创建此设置是一项耗费资源的操作,除了布局顺序之外,此屏幕上没有任何变化,因此不需要额外的工作来显示屏幕.只需在 Tab1
和 Tab2
的 onCreateView
回调中返回不同的 layout
并将现有 fragment 重新附加到相同的 id。
要实现这一点,人们会认为只需设置 HostFragment.setRetainInstance(true)
即可完成工作,并且部分地确实如此。没有重新创建任何内容,选项卡按应有的方式保留,一切正常。现在让我们进入正题。
问题
大问题稍后出现。 Tab1Fragment1
(以及所有其他 TabXFragmentY
)似乎没有附加到旋转时创建的新 Activity
。它们仅在第一次运行 Activity
时附加,然后当用户旋转屏幕时什么都不做。
后果
这给我带来了两个大问题(可能还有更多):
Activity
被泄露,因为 TabHost
中的所有 Fragments
都无缘无故地附加到它。TabXFragmentY
上调用 getActivity()
时,您将获得旧 Activity
和可怕的“在 onSavedInstance 之后无法执行操作”异常。当您想要从父 Activity 中获取某些内容时,这会导致严重的问题。此外,在设置 HostFragment.setRetainInstance(false)
时不会发生这种情况,所有内容都会重新创建并且工作正常。
问题
这是我只看到的问题吗?我找不到关于该主题的任何内容,而且我已经检查了我的代码一百次。
我还尝试从 v4 示例中更改 com.example.android.supportv4.app.FragmentTabsFragmentSupport
,我确实将其设置为保留它的实例并在其嵌套 fragment 之一中添加了日志信息在每个 onAttach
调用上进行验证,我可以看到仍然存在相同的问题。我很困惑。帮助。
最佳答案
最近我一直在使用 Fragment
开发一个 TabHost
并发现了同样的问题。基本上,您需要控制附加/分离哪些 Fragment
,在我的例子中,我在 onTabChanged()
事件中执行此操作。
我有一个 TabInfo
类,我在其中存储每个 Tab
的以下信息
Fragment
的名称(用于识别目的)。 fragment
。TabHost.TabSpec
规范,就像您要删除某些选项卡一样,这里有重新创建其余选项卡的信息,因为 TabHost
对于删除选项卡有点棘手.Bundle
(基本上是配置更改之前保存的实例)。此外,我需要跟踪 lastTab
和 newTab
打开,因为 TabHost
没有本地方式知道哪个TabHost
刚刚关闭,所以我在类范围内声明一个变量。这就是我现在的处理方式,我会尝试添加尽可能多的评论:
@Override
public void onTabChanged(final String tag) {
// I get the info for the Tab just triggered using its tag
final TabInfo newTab = mTabInfo.get(tag);
// If there's actually been a tab change...
if (lastTab != newTab) {
// You'll have to make a transaction for replacing the Fragment
final FragmentTransaction ft = this.getSupportFragmentManager().beginTransaction();
// If the last tab actually has a Fragment associated to it
if ((lastTab != null) && (lastTab.getFragment() != null)) {
// In my case I've an additional level of complexion, as I have a nested Fragment
// inside my content Fragment. So I have to remove it first prior to detaching
// the parent Fragment. This is not needed if you have just one Fragment as content.
final Fragment loginFrag = (Fragment) lastTab.getFragment().getActivity().getSupportFragmentManager().findFragmentById(lastTab.getLoginFragId());
ft.remove(loginFrag);
// And this is what does the trick: I initially was calling detach() instead of remove()
// but seems that with some versions there's a problem that makes not apply it,
// calling remove will actually remove this Fragment
ft.remove(lastTab.getFragment());
}
// You've detached the old Fragment, you have now to attach the new one
if (newTab != null) {
if (newTab.getFragment() == null) {
// Inflate the new content if it's the first time the tab has been fired
final TabFragmentInflater tabInf = new TabFragmentInflater();
newTab.setFragment(Fragment.instantiate(this, tabInf.getClass().getName(), newTab.getArgs()));
ft.add(R.id.realtabcontent, newTab.getFragment(), newTab.getTag());
}
else {
// if not, just attach its fragment
ft.attach(newTab.getFragment());
}
}
ft.commit();
this.getSupportFragmentManager().executePendingTransactions();
lastTab = newTab;
}
}
----编辑----
回答您的问题:
我确实在 TabHost
上调用了 newTabSpec()
,它在一个单独的方法中,我没有包含因为我只是包含了 onTabChanged( )
回调。您必须正常创建选项卡,此onTabChanged()
方法只会在您单击选项卡时触发。对于创作,我做了这样的事情:
private void initTabHost(final Bundle args) {
final TabHost th = (TabHost) findViewById(android.R.id.tabhost);
th.setup();
// I have a HashMap called fragMap where as the key I define the tab's name
// And as the value, I have an Integer which is a unique identifier
// to know what to inflate when I call the TabFragmentInflater (I will
// add the code below). You can perfectly add it as an id or a tag also.
for (final String tablabel : fragMap.keySet()) {
final TabHost.TabSpec tabSpec = th.get().newTabSpec(tabname).setIndicator(tabname);
// Here I initialize a TabInfo object for this tab, which will include additional
// Handling info: Name of tab, Tab Spec, The unique ID I explained above,
// the forth argument is irrelevant in your example, and args (saved instance)
final TabInfo tabInfo = new TabInfo(tabname, tabSpec, fragMap.get(tablabel), R.id.someLayout, args);
// This is not actually the TabHost's `addTab()` method, I'll call it inside
// this method (see below)
MyClass.addTab(this, th, tabSpec, tabInfo);
// I have to be able to keep tracking of that info
mTabInfo.put(tabInfo.getTag(), tabInfo);
}
// For the tab creation, I force it to start on the first tab
this.onTabChanged(firstTabNameTag);
th.setOnTabChangedListener(this);
}
addTab()
方法非常简单,它会膨胀对象并在需要时将其分离。
private static void addTab(final MyClass activity, final TabHost tabHost, final TabHost.TabSpec tabSpec, final TabInfo tabInfo) {
tabSpec.setContent(new TabFactory(activity)); // This is just an empty View
final String tag = tabSpec.getTag();
// Here I check if there's already a Fragment for that tab, probably in a
// previously saved state. If this happens, we deactivate it, because our
// former state for that tab is "not shown".
tabInfo.setFragment(activity.getSupportFragmentManager().findFragmentByTag(tag));
if ((tabInfo.getFragment() != null) && (!tabInfo.getFragment().isDetached())) {
final FragmentTransaction ft = activity.getSupportFragmentManager().beginTransaction();
ft.detach(tabInfo.getFragment());
ft.commit();
activity.getSupportFragmentManager().executePendingTransactions();
}
// Actually, there's where I call the "official" `addTab()` from `TabHost`
tabHost.addTab(tabSpec);
}
所以只剩下 TabFragmentInflater。它只是一个 Fragment
,它根据我上面提到的唯一 ID 来膨胀相应的布局,所以它是这样的:
@Override
public View onCreateView(final LayoutInflater inflater, final ViewGroup container, final Bundle savedInstanceState) {
fragId = getArguments().getInt("fragid");
if (view != null) {
ViewGroup parent = (ViewGroup) view.getParent();
if (parent != null)
parent.removeView(view);
}
try {
switch (fragId) {
case 1: // My first tab...
final LinearLayout fragLayout = (LinearLayout) inflater.inflate(R.layout.myfirsttab_fragment_layout, container, false);
...
return fragLayout;
case 2: // My second tab
fragLayout = (LinearLayout) inflater.inflate(R.layout.mysecondtab_fragment_layout, container, false);
...
return fragLayout;
...
}
}
catch (final InflateException e) { return view; }
return null;
}
关于android - 嵌套保留的 FragmentTabHost 不会将选项卡附加到新 Activity,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22219414/
我关注了 tutorial on creating a popup for an add-on在 Firefox 中,效果很好。 我现在遇到的问题是它创建的弹出窗口不会更改大小以适应我添加到其中的内容
我有一些视频,我需要连接一个标题并添加一些覆盖,问题是我需要先做覆盖,否则时间不正确,然后才将标题连接到视频的开头 ffmpeg -i talk.mp4 -i start_pancarte.png
我正在尝试附加一个 CSV 文件。这是我正在使用的线路。不幸的是,我找不到 export-csv 的附加选项。任何想法都有助于使其发挥作用。 Get-ADGroupMember "Domain Adm
我正在努力理解 Attach API (com.sun.tools.attach.*) 的用途。它的典型用途是什么?它是为了“模拟”JVM,以便您可以在不部署/启动代码的情况下测试您的代码吗?它是一个
我不明白为什么这不起作用。 soup_main = BeautifulSoup('FooBar') soup_append = BeautifulSoup('Meh') soup_main.body.
我有以下代码来返回我想要的字符串 $sql = " SELECT `description` FROM `auctions` WHERE `description` REGEX
我正在尝试从数组中附加具有多个值的元素,但我做错了。这是我的代码: for(var i=0; i ` + pricesArray[i].start_date ` ` + pricesArray[i
我正在尝试将图像链接添加到此 javascript 附加表中。使图像位于按钮上方 这是代码 $("#1").append(""+section+""+no+""+price+""+button+""
我有一个问题,我已经解决了,但它太烦人了。 我有一个 js 代码,当使用“追加”按下按钮时,它会放下一些 html 代码,并且通过该代码,我为 x 按钮提供了一个 id,并为容器元素提供了一个 id。
我想逐行读取文件,并且每一行可能都有很多字符。 这个版本的readline效果很好 func readLine(r *bufio.Reader) ([]byte, error) { var (
我有一个网站,每次用户登录或注销时,我都会将其保存到文本文件中。 如果不存在,我的代码在附加数据或创建文本文件时不起作用。这是示例代码 $myfile = fopen("logs.txt", "wr"
我正在尝试使用 typescript 和 Formik 创建一个自定义输入字段。我可以就完成以下代码的最佳方式获得一些帮助吗?我需要添加额外的 Prop 标签和名称......我已经坚持了一段时间,希
我有一个字符串 big_html,我想将它添加到某个 div 中。我观察到以下方面的性能差异: $('#some-div').append( big_html ); // takes about 10
如何使用 FormData 创建以下结果 ------WebKitFormBoundaryOmz20xyMCkE27rN7 Content-Disposition: form-data; name="
有没有办法附加 jQuery 事件处理程序,以便在任何先前附加的事件处理程序之前触发该处理程序?我遇到了this article ,但代码不起作用,因为事件处理程序不再存储在数组中,而这正是他的代码所
我正在开发一个需要网络登录的 iPhone 应用程序。像往常一样我打电话 [[UIApplication sharedApplication] openURL:loginURL]; 这将关闭应用程序并
我想开发一个仅针对特定域激活的扩展。 我不希望它在不浏览此特定域时出现在浏览器菜单中。 有可能这样做吗? 最佳答案 可能:对于菜单,您可以添加一个弹出窗口侦听器,用于检查当前加载的URL(docs f
这段 JavaScript 代码 function writeCookie(CookieName, CookieValue, CookieDuration) { var expiration
我正在使用 Handlebars 来渲染使用ajax从本地服务器获得的信息。我的 HTML 看起来像: {{#each Tabs}}
我尝试了以下代码,但当输入框中没有数据时它不会通知。当我直接添加此内容(不附加)时,它会起作用。我在这里做错了什么 var output = "\n"+ "\n"+
我是一名优秀的程序员,十分优秀!