gpt4 book ai didi

Android fragment 相互重叠

转载 作者:行者123 更新时间:2023-11-29 19:15:05 26 4
gpt4 key购买 nike

在我的应用程序中,我有两个 fragment :FragmentAFragmentB,它们是在 的帮助下从我的 MainActivity 加载的滑动菜单

当显示 FragmentB 时,它会立即在后台启动一个 AsyncTask 以从服务器加载数据。加载过程由 Activity 的 SwipeRefreshLayout 反射(reflect)给用户。

加载完成后,FragmentB 的 UI 会在传递给 MyAsyncClass 的委托(delegate)的帮助下用加载的数据刷新。

我现在面临以下问题:一旦我的委托(delegate)中的 processFinish 被调用,我就会在 swipeRefreshLayout.setRefreshing(false) 的帮助下停止 SwipeRefreshLayout 的刷新。如果用户返回 FragmentA 而 background-Task 仍在进行中,它出现在大约 20% 的情况下, FragmentB 仍在 的后台>用户界面中的 FragmentA(请参见下面的屏幕截图)。

如您在图像中所见,显示了 FragmentA(文本“No tracks played yet”来自 FragmentA),但在背景中您可以看到仍然是 FragmentB

如前所述,这并非 100% 的情况,因此这不应该是背景/透明度问题。如果我注释掉 swipeRefreshLayout.setRefreshing(false),那么问题就不会重现,但我需要以某种方式停止 SwipeRefreshLayout,以防用户停留在 fragment B.

知道是什么导致了这种行为吗?

Screenshot of the problem

异步响应接口(interface):

public interface AsyncResponse {

void processFinish(String result);
}

fragment B 级:

    public class FragmentB extends MyFragment implements SwipeRefreshLayout.OnRefreshListener, AsyncResponse {

private SwipeRefreshLayout swipeRefreshLayout;
private View view;
private RecyclerView recyclerView;

public FragmentB () {
}


@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

view = inflater.inflate(R.layout.songs_list, container, false);

swipeRefreshLayout = (SwipeRefreshLayout) view.findViewById(R.id.swipe_refresh_layout);
swipeRefreshLayout.setOnRefreshListener(this);
swipeRefreshLayout.setColorSchemeColors(getResources().getColor(R.color.colorPrimary));

// Set the adapter
Context context = view.getContext();
recyclerView = (RecyclerView) view.findViewById(R.id.listinclude);
recyclerView.setLayoutManager(new LinearLayoutManager(context));

recyclerView.addItemDecoration(new DividerItemDecoration(getActivity(), R.drawable.divider));

List<Song> songsList = databaseHandler.getAllSongs();
setVisibilities(songsList);

this.recyclerViewAdapter = new RecyclerViewAdapter(songsList, mListener, false, false);

recyclerView.setAdapter(this.recyclerViewAdapter);

swipeRefreshLayout.setRefreshing(true);
receive();

return view;
}

private void setVisibilities(List<Song> songsList) {
ViewFlipper viewFlipper = (ViewFlipper) view.findViewById(R.id.viewFlipper);
if (songsList.isEmpty() && viewFlipper.getDisplayedChild() == 0) {
viewFlipper.setDisplayedChild(1);
} else if (!songsList.isEmpty() && viewFlipper.getDisplayedChild() == 1) {
viewFlipper.setDisplayedChild(0);
}
}


@Override
public void onRefresh() {
if (MyOnlineHelper.isOnline(getContext())) {
receive();
} else {
swipeRefreshLayout.setRefreshing(false);
}
}

private void receive() {
new MyAsyncClass(this, getContext()).execute("receiving");
}

@Override
public void processFinish(String output) {
// refreshUI
swipeRefreshLayout.setRefreshing(false);
}
}

FragmentA 类:

public class FragmentA extends MyFragment implements AsyncResponse {

private View view;
private RecyclerView recyclerView;

public FragmentA() {
}


@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
List<Song> songsList = myDatabaseHandler.getAllSongs();

view = inflater.inflate(R.layout.home_list, container, false);

// Set the adapter
Context context = view.getContext();
recyclerView = (RecyclerView) view.findViewById(R.id.listinclude);
recyclerView.setLayoutManager(new LinearLayoutManager(context));

recyclerView.addItemDecoration(new DividerItemDecoration(getActivity(), R.drawable.divider));

setVisibilities(songsList);

this.recyclerViewAdapter = new RecyclerViewAdapter(songsList, mListener, false, true);

recyclerView.setAdapter(this.recyclerViewAdapter);
return view;
}

@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
}


private void setVisibilities(List<Song> songsList) {
ViewFlipper viewFlipper = (ViewFlipper) view.findViewById(R.id.viewFlipper);
if (songsList.isEmpty() && viewFlipper.getDisplayedChild() == 0) {
viewFlipper.setDisplayedChild(1);
} else if (!songsList.isEmpty() && viewFlipper.getDisplayedChild() == 1) {
viewFlipper.setDisplayedChild(0);
}
}


@Override
public void processFinish(String output) {
// does something
}
}

MyAsyncClass:

public class MyAsyncClass extends AsyncTask<String, Integer, String> {

private AsyncResponse delegate;
private Context mContext;

private MyDatabaseHandler myDatabaseHandler;

public MyAsyncClass(AsyncResponse delegate, Context context) {
this.delegate = delegate;
this.mContext = context;
}


@Override
protected String doInBackground(String... params) {

// calling webservice and writing it to the database
}

@Override
protected void onPostExecute(String result) {
delegate.processFinish(result);
super.onPostExecute(result);
}
}

MainActivity 类:

public class MainActivity extends AppCompatActivity implements MyFragment.OnListFragmentInteractionListener, AsyncResponse {


private FragmentA fragmentA = new FragmentA();
private FragmentB fragmentB;

private NavigationView navigationView;
private DrawerLayout drawer;
private Toolbar toolbar;

// index to identify current nav menu item
private static int navItemIndex = 0;

public static String CURRENT_TAG = MyConstants.TAG_FRAGMENT_A;

// toolbar titles respected to selected nav menu item
private String[] activityTitles;

// flag to load home fragment when user presses back key
private Handler mHandler;

@Override
protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

fragmentA.setDatabaseHandler(this.myDatabaseHandler);

// Init UI
toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);

mHandler = new Handler();

drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
navigationView = (NavigationView) findViewById(R.id.nav_view);

fabSendButton = (FloatingActionButton) findViewById(R.id.fab);

// Navigation view header
navHeader = navigationView.getHeaderView(0);


// load toolbar titles from string resources
activityTitles = getResources().getStringArray(R.array.sliding_menu_item_activity_titles);


// initializing navigation menu
setUpNavigationView();

if (savedInstanceState == null) {
navItemIndex = 0;
CURRENT_TAG = MyConstants.TAG_FRAGMENT_A;
loadHomeFragment();
}
}

/***
* Returns respected fragment that user
* selected from navigation menu
*/
private void loadHomeFragment() {

// set toolbar title
setToolbarTitle();

// if user select the current navigation menu again, don't do anything
// just close the navigation drawer
if (getSupportFragmentManager().findFragmentByTag(CURRENT_TAG) != null) {
drawer.closeDrawers();
return;
}

// Sometimes, when fragment has huge data, screen seems hanging
// when switching between navigation menus
// So using runnable, the fragment is loaded with cross fade effect
// This effect can be seen in GMail app
Runnable mPendingRunnable = new Runnable() {
@Override
public void run() {
// update the activity_main_header_with_item content by replacing fragments
Fragment fragment = getFragment();
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.setCustomAnimations(android.R.anim.fade_in, android.R.anim.fade_out);
fragmentTransaction.replace(R.id.frame, fragment, CURRENT_TAG);
fragmentTransaction.commit();
}
};

// If mPendingRunnable is not null, then add to the message queue
if (mPendingRunnable != null) {
mHandler.post(mPendingRunnable);
}

//Closing drawer on item click
drawer.closeDrawers();

// refresh toolbar menu
invalidateOptionsMenu();
}

private Fragment getFragment() {
switch (navItemIndex) {
case 0:
return this.fragmentA;
case 1:
if (fragmentB == null) {
fragmentB = new FragmentB();
fragmentB.setDatabaseHandler(this.myDatabaseHandler);
}
return fragmentB;
default:
return this.fragmentA;
}
}

private void setToolbarTitle() {
getSupportActionBar().setTitle(activityTitles[navItemIndex]);
}

private void setUpNavigationView() {
//Setting Navigation View Item Selected Listener to handle the item click of the navigation menu
navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {

// This method will trigger on item Click of navigation menu
@Override
public boolean onNavigationItemSelected(MenuItem menuItem) {

//Check to see which item was being clicked and perform appropriate action
switch (menuItem.getItemId()) {
//Replacing the activity_main_header_with_item content with ContentFragment Which is our Inbox View;
case R.id.nav_A:
navItemIndex = 0;
CURRENT_TAG = MyConstants.TAG_FRAGMENT_A;
break;
case R.id.nav_B:
navItemIndex = 1;
CURRENT_TAG = MyConstants.TAG_FRAGMENT_B;
break;
default:
navItemIndex = 0;
}

loadHomeFragment();

return true;
}
});

ActionBarDrawerToggle actionBarDrawerToggle = new ActionBarDrawerToggle(this, drawer, toolbar, R.string.openDrawer, R.string.closeDrawer) {

@Override
public void onDrawerClosed(View drawerView) {
// Code here will be triggered once the drawer closes as we dont want anything to happen so we leave this blank
super.onDrawerClosed(drawerView);
}

@Override
public void onDrawerOpened(View drawerView) {
// Code here will be triggered once the drawer open as we dont want anything to happen so we leave this blank
super.onDrawerOpened(drawerView);
}
};

//Setting the actionbarToggle to drawer layout
drawer.setDrawerListener(actionBarDrawerToggle);

//calling sync state is necessary or else your hamburger icon wont show up
actionBarDrawerToggle.syncState();
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_main_header_with_item, menu);
return true;
}
}

最佳答案

感谢 Prsnth Gettin High 和这篇文章(When switch fragment with SwipeRefreshLayout during refreshing, fragment freezes but actually still work),将 SwipeRefreshLayout 包装在 FrameLayout 中会有所帮助,就像这样:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">

<android.support.v4.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/swipe_refresh_layout"
android:layout_width="match_parent"
android:layout_height="match_parent">

<ViewFlipper
android:id="@+id/viewFlipper"
android:layout_width="match_parent"
android:layout_height="match_parent">

<include
android:id="@+id/listinclude"
layout="@layout/list"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginLeft="8dp"
android:layout_marginRight="8dp"
android:visibility="visible"/>

<include
android:id="@+id/emptyinclude"
layout="@layout/fragment_empty"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:visibility="visible"/>
</ViewFlipper>

</android.support.v4.widget.SwipeRefreshLayout>
</FrameLayout>

关于Android fragment 相互重叠,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43823144/

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