【问题标题】:Dynamic buttons added to a GridLayout using RecyclerView are added on top of each other使用 RecyclerView 添加到 GridLayout 的动态按钮相互叠加
【发布时间】:2016-09-07 07:02:09
【问题描述】:

我想将动态按钮添加到网格布局中,如下所示。 每次添加一个按钮,布局必须自行更新以使按钮成行。 我的问题是,当添加一个新按钮时,它会显示在添加的最后一个按钮的顶部,并且没有以行格式放置。在下面的这张图片中,我展示了我想要的内容以及当前正在发生的事情,标有数字的方块是我的按钮。在我所拥有的图像中,我在布局中添加了 6 个按钮,但它们都在彼此之上。

我已经查看了如何执行此操作,建议我使用 RecyclerViewGridLayoutManager。我已将此添加到我的代码中,但就像我说的 问题是,当我添加一个按钮时,如果我添加另一个按钮,第二个按钮会添加到第一个按钮之上。如果我的按钮是预设的,则可以使用,但不适用于动态按钮。

这是我的代码:

主片段启动RecyclerView。我有另一个启动createButton() 方法并传递drawableString 的活动。这些可绘制对象和字符串根据用户的操作一次一个地传递给此方法,并一次一个地创建一个图像按钮。

public class MyFragment extends Fragment {

    private GridLayoutManager lLayout;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

    }

    // onCreateView
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.my_fragment, container, false);

        // Get screen size to have different layouts for phone and tablet
        int screenSize = getResources().getConfiguration().screenLayout &
                Configuration.SCREENLAYOUT_SIZE_MASK;

        String toastMsg;
        switch (screenSize) {
            case Configuration.SCREENLAYOUT_SIZE_LARGE:
                toastMsg = "Large screen";
                Log.d("tag_name", "Large screen");
                break;
            case Configuration.SCREENLAYOUT_SIZE_NORMAL:
                toastMsg = "Normal screen";
                Log.d("tag_name", "Normal screen");
                break;
            case Configuration.SCREENLAYOUT_SIZE_SMALL:
                toastMsg = "Small screen";
                Log.d("tag_name", "Small screen");
                break;
            default:
                toastMsg = "Screen size is neither large, normal or small";
                Log.d("tag_name", "Screen size is not large, normal, or small");
        }
        Toast.makeText(getActivity(), toastMsg, Toast.LENGTH_LONG).show();

        // Create an empty list to initialize the adapter (or else get nullPointerException error)
        List<ItemObject> myList = new ArrayList<ItemObject>();

        // 3 rows for tablet
        // 2 rows for phone

        if (screenSize == Configuration.SCREENLAYOUT_SIZE_LARGE
                || screenSize == Configuration.SCREENLAYOUT_SIZE_XLARGE) {
            lLayout = new GridLayoutManager(getActivity(), 3, GridLayoutManager.HORIZONTAL, false);
        } else
            lLayout = new GridLayoutManager(getActivity(), 2, GridLayoutManager.HORIZONTAL, false);

        RecyclerView rView = (RecyclerView) view.findViewById(R.id.recycler_view);

        rView.setHasFixedSize(true);
        rView.setLayoutManager(lLayout);

        RecyclerViewAdapter rcAdapter = new RecyclerViewAdapter(getActivity(), myList);
        rView.setAdapter(rcAdapter);

        return view;
    }

    private List<ItemObject> getAllItemList(String applicationName, Drawable app_drawable) {

        List<ItemObject> allItems = new ArrayList<ItemObject>();
        allItems.add(new ItemObject(applicationName, app_drawable));

        return allItems;
    }

    public void createButton(Drawable d, String appName) {

        List<ItemObject> rowListItem = getAllItemList(appName, d);
        lLayout = new GridLayoutManager(getActivity(), 2, GridLayoutManager.HORIZONTAL, false);


        RecyclerView rView = (RecyclerView) getView().findViewById(R.id.recycler_view);
        rView.setHasFixedSize(true);
        rView.setLayoutManager(lLayout);

        RecyclerViewAdapter rcAdapter = new RecyclerViewAdapter(getActivity(), rowListItem);
        rView.setAdapter(rcAdapter);

    }

}

我的布局(my_fragment):

<LinearLayout android:id="@+id/my_fragment"
          xmlns:android="http://schemas.android.com/apk/res/android"
          android:layout_width="match_parent"
          android:layout_height="match_parent"
          android:layout_marginTop="15dp"
          android:gravity="center_horizontal">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:scrollbars="horizontal"/>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <ImageButton
            android:id="@+id/new_app_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"/>

        <TextView
            android:id="@+id/new_app_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_below="@+id/new_app_button"
            android:gravity="center"/>

    </RelativeLayout>
</LinearLayout>

编辑

这是我为 MyFragment 更新的代码 现在,当添加第二个 ImageButton 时,我得到了一个 nullPointerException

public class MyFragment extends Fragment {

private GridLayoutManager lLayout;

private ButtonCreator bc;
RecyclerViewAdapter rcAdapter;


@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    List<ItemObject> myList = new ArrayList<ItemObject>();
    rcAdapter = new RecyclerViewAdapter(getActivity(),myList);

}

// onCreateView
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.home_fragment, container, false);


    // Get screen size so we can have different layouts for phone and tablet
    int screenSize = getResources().getConfiguration().screenLayout &
            Configuration.SCREENLAYOUT_SIZE_MASK;

    String toastMsg;
    switch(screenSize) {
        case Configuration.SCREENLAYOUT_SIZE_LARGE:
            toastMsg = "Large screen";
            Log.d("tag_name", "Large screen");
            break;
        case Configuration.SCREENLAYOUT_SIZE_NORMAL:
            toastMsg = "Normal screen";
            Log.d("tag_name", "Normal screen");
            break;
        case Configuration.SCREENLAYOUT_SIZE_SMALL:
            toastMsg = "Small screen";
            Log.d("tag_name", "Small screen");
            break;
        default:
            toastMsg = "Screen size is neither large, normal or small";
            Log.d("tag_name", "Screen size is not large, normal, or small");
    }
    Toast.makeText(getActivity(), toastMsg, Toast.LENGTH_LONG).show();


    // Create an empty list to initialize the adapter (or else get nullPointerException error)

    if (screenSize == Configuration.SCREENLAYOUT_SIZE_LARGE
    || screenSize == Configuration.SCREENLAYOUT_SIZE_XLARGE) {
        lLayout = new GridLayoutManager(getActivity(), 3, GridLayoutManager.HORIZONTAL, false);
    }

    else lLayout = new GridLayoutManager(getActivity(), 2, GridLayoutManager.HORIZONTAL, false);


    RecyclerView rView = (RecyclerView)view.findViewById(R.id.recycler_view);

    rView.setHasFixedSize(true);
    rView.setLayoutManager(lLayout);


    rView.setAdapter(rcAdapter);


    return view;
}

private List<ItemObject> getAllItemList(String applicationName, Drawable app_drawable){

    List<ItemObject> allItems = new ArrayList<ItemObject>();
    allItems.add(new ItemObject(applicationName, app_drawable));


    return allItems;
}


public void createButton (Drawable d, String appName){

    rcAdapter.addItem(new ItemObject(appName , d));

}



// In order to get the view of HomeFragment correctly

public interface ButtonCreator{
    void buttonCreator(Drawable d, String string);
}



@Override
public void onAttach(Activity activity) {
    super.onAttach(activity);
    bc = (ButtonCreator) activity;
}

}

我如何在 MyFragment 中调用 createButton 方法(在 runOnUiThread 方法中调用)

public class MainActivity extends FragmentActivity implements MyFragment.ButtonCreator {


private Thread repeatTaskThread;
private byte[] byteArray;

@Override
public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);

   FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
   ft.replace(R.id.container, new MyFragment()).commit();
}


public void buttonCreator(Drawable d,String a) {
    MyFragment homeFragment = (MyFragment)getSupportFragmentManager().getFragments().get(1);
    homeFragment.createButton(d, a);
}

public void RepeatTask(){

     repeatTaskThread = new Thread() {
        public void run() {
           while (true) {


              try {


                 System.out.println("TRY_1");

                 Socket socket = new Socket("192.168.0.26", 5050);

                 // Get data sent through socket
                 DataInputStream DIS = new DataInputStream(socket.getInputStream());

                 System.out.println("DataInputStream Started");

                 // read data that got sent
                 final String applicationName = DIS.readUTF();


                 // read array data for bitmap

                 int len = DIS.readInt();
                 byte[] data = new byte[len];

                 DIS.readFully(data, 0, data.length);


                 Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
                 final Drawable d = new BitmapDrawable(getResources(), bitmap);


                  runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                       // TODO Auto-generated method stub


                       Log.d("tag_name", "Try_2");

                        buttonCreator(d,applicationName);


                    }
                 });


                 socket.close();




              } catch (Exception e) {


                 System.out.println("Exception is " + e.toString());

              }


              try {
                 // Sleep for 5 seconds
                 Thread.sleep(5000);
              } catch (Exception e) {
                 e.printStackTrace();
              }
           }
        }

        ;
     };
     repeatTaskThread.start();


}

}

我的应用在创建第二个按钮时崩溃时的错误:

05-11 17:05:25.275 14257-14257/it.anddev.bradipao.janus E/RecyclerView: No adapter attached; skipping layout
05-11 17:05:25.275 14257-14257/it.anddev.bradipao.janus D/AndroidRuntime: Shutting down VM
05-11 17:05:25.275 14257-14257/it.anddev.bradipao.janus W/dalvikvm: threadid=1: thread exiting with uncaught exception (group=0x437c0160)
05-11 17:05:25.305 14257-14257/it.anddev.bradipao.janus E/AndroidRuntime: FATAL EXCEPTION: main
                                                                      Process: it.anddev.bradipao.janus, PID: 14257
                                                                      java.lang.NullPointerException
                                                                          at android.support.v7.widget.RecyclerView.computeHorizontalScrollRange(RecyclerView.java:1518)
                                                                          at android.view.View.onDrawScrollBars(View.java:12248)
                                                                          at android.view.View.draw(View.java:14794)
                                                                          at android.support.v7.widget.RecyclerView.draw(RecyclerView.java:3171)
                                                                          at android.view.View.getDisplayList(View.java:13655)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.View.draw(View.java:14505)
                                                                          at android.view.ViewGroup.drawChild(ViewGroup.java:3134)
                                                                          at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2971)
                                                                          at android.view.View.getDisplayList(View.java:13650)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.View.draw(View.java:14505)
                                                                          at android.view.ViewGroup.drawChild(ViewGroup.java:3134)
                                                                          at android.support.v7.widget.RecyclerView.drawChild(RecyclerView.java:3693)
                                                                          at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2971)
                                                                          at android.view.View.draw(View.java:14791)
                                                                          at android.support.v7.widget.RecyclerView.draw(RecyclerView.java:3171)
                                                                          at android.view.View.getDisplayList(View.java:13655)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.ViewGroup.dispatchGetDisplayList(ViewGroup.java:3108)
                                                                          at android.view.View.getDisplayList(View.java:13593)
                                                                          at android.view.View.getDisplayList(View.java:13697)
                                                                          at android.view.HardwareRenderer$GlRenderer.buildDisplayList(HardwareRenderer.java:1570)
                                                                          at android.view.HardwareRenderer$GlRenderer.draw(HardwareRenderer.java:1449)
                                                                          at android.view.ViewRootImpl.draw(ViewRootImpl.java:2476)
                                                                          at android.view.ViewRootImpl.performDraw(ViewRootImpl.java:2300)
                                                                          at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1929)
                                                                          at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1043)
                                                                          at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5885)
                                                                          at android.view.Choreographer$CallbackRecord.run(Choreographer.java:771)
                                                                          at android.view.Choreographer.doCallbacks(Choreographer.java:574)
                                                                          at android.view.Choreographer.doFrame(Choreographer.java:544)
                                                                          at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:757)
                                                                          at android.os.Handler.handleCallback(Handler.java:733)
                                                                          at android.os.Handler.dispatchMessage(Handler.java:95)
                                                                          at android.os.Looper.loop(Looper.java:149)
                                                                          at android.app.ActivityThread.main(ActivityThread.java:5257)
                                                                          at java.lang.reflect.Method.invokeNative(Native Method)
                                                                          at java.lang.reflect.Method.invoke(Method.java:515)
                                                                          at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:788)
                                                                          at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:604)
                                                                          at dalvik.system.NativeStart.main(Native Method)

【问题讨论】:

  • 我看到每次您需要创建一个按钮时,您都在重新初始化recyclerView,结果是使用该按钮创建了一个新的recyclerView。
  • @varunkr 我明白了!所以我想我每次都在 MyFragment 的 createButton 方法中初始化 recyclerView 对吗?那么你知道我怎样才能改变这段代码来更新,而不是每次都创建一个新的 recyclerView 吗?
  • 看看我的回答

标签: java android dynamic android-recyclerview grid-layout


【解决方案1】:

做这样的事情

public class MyFragment extends Fragment {

    private GridLayoutManager lLayout;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

    }

    // onCreateView
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.my_fragment, container, false);

        // Get screen size to have different layouts for phone and tablet
        int screenSize = getResources().getConfiguration().screenLayout &
                Configuration.SCREENLAYOUT_SIZE_MASK;

        String toastMsg;
        switch (screenSize) {
            case Configuration.SCREENLAYOUT_SIZE_LARGE:
                toastMsg = "Large screen";
                Log.d("tag_name", "Large screen");
                break;
            case Configuration.SCREENLAYOUT_SIZE_NORMAL:
                toastMsg = "Normal screen";
                Log.d("tag_name", "Normal screen");
                break;
            case Configuration.SCREENLAYOUT_SIZE_SMALL:
                toastMsg = "Small screen";
                Log.d("tag_name", "Small screen");
                break;
            default:
                toastMsg = "Screen size is neither large, normal or small";
                Log.d("tag_name", "Screen size is not large, normal, or small");
        }
        Toast.makeText(getActivity(), toastMsg, Toast.LENGTH_LONG).show();

        // Create an empty list to initialize the adapter (or else get nullPointerException error)

        // 3 rows for tablet
        // 2 rows for phone

        if (screenSize == Configuration.SCREENLAYOUT_SIZE_LARGE
                || screenSize == Configuration.SCREENLAYOUT_SIZE_XLARGE) {
            lLayout = new GridLayoutManager(getActivity(), 3, GridLayoutManager.HORIZONTAL, false);
        } else
            lLayout = new GridLayoutManager(getActivity(), 2, GridLayoutManager.HORIZONTAL, false);

        RecyclerView rView = (RecyclerView) view.findViewById(R.id.recycler_view);

        rView.setHasFixedSize(true);
        rView.setLayoutManager(lLayout);

        rView.setAdapter(rcAdapter);

        return view;
    }

    RecyclerViewAdapter rcAdapter;

    private List<ItemObject> getAllItemList(String applicationName, Drawable app_drawable) {

        List<ItemObject> allItems = new ArrayList<ItemObject>();
        allItems.add(new ItemObject(applicationName, app_drawable));

        return allItems;
    }

    public void createButton(Drawable d, String appName) {

        rcAdapter.addItem(new ItemObject(appName , d))

    }

}

并在适配器中添加此方法

public void addItem(int position) {
        list.add(position);
        notifyItemInserted(position);
        notifyItemRangeChanged(position, list.size());
    }

编辑

为防止出现警告,请尝试将适配器初始化移至 onCreate

@Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            List<ItemObject> myList = new ArrayList<ItemObject>();
            rcAdapter = new RecyclerViewAdapter(getActivity(), myList);
        }

【讨论】:

  • 谢谢!对于我的 addItem,我必须使用以下代码:“public void addItem(ItemObject item) { mList.add(item); notifyItemInserted(mList.size() - 1); }” 这样我就不会出错,并且可以建立我的毕业典礼。当我为图像按钮获取第一张图像时,我收到警告“未连接适配器:跳过布局”,但我的图像按钮创建得很好。但是当我尝试向布局中添加第二个图像按钮时,我得到了相同的警告,并且我的应用程序崩溃并给出了 nullPointerException。有什么明显的原因可以导致这种情况吗?
  • @Natalie 你能引用你得到的确切空指针异常吗
  • 如果我将适配器初始化添加到 onCreate,它并没有消除错误,但我确实更新了我的问题以显示错误。我将不得不尝试找出抛出“null”的原因,因为我的 stackTrace 现在并没有告诉我太多。
  • @Natalie 您究竟是如何从活动中调用此 createButton 方法的?您之前所做的与您现在所做的区别在于,现在您的创建按钮不会像之前那样初始化 recyclerView 和适配器。由于这个原因,您可能会遇到异常,因为您的适配器等可能第二次保持未初始化状态。这取决于您如何启动片段以及如何从活动中调用此 createButton。能发一下活动代码吗这可能会解决问题!
  • 我更新了我的问题的“编辑”部分,以显示我是如何开始我的片段的。我在这里调用它的方式与我在发布的另一个问题中的方式不同,因为这会在创建第一个图像按钮时导致 nullPointerExceptions,现在它发生在第二个图像按钮上,但也许这一切都源于我错误地初始化我的片段的同样问题?
【解决方案2】:

当你创建一个按钮时,你应该只需要做两件事:

  1. 更新支持RecyclerView 的数据结构 适配器
  2. 使用适配器的notify...()方法之一通知适配器数据发生变化(有several of them,使用最适用的一个)

您当前的代码似乎没有做这些事情。您也不需要创建或设置任何新的适配器或布局管理器。

在您的情况下,您将向适配器类保存的列表中添加一个元素,然后可能会在适当的位置调用notifyItemInserted()。您可以通过向适配器添加方法来同时完成这两项工作:

public void addItem(ItemObject item) {
    mList.add(item);
    notifyItemInserted(mList.size() - 1);
}

无关:在 Android 上,您可以使用资源解析系统来确定要显示的列数。

制作src/main/res/values/integers.xml 并添加:

<resources>
    <integer name="grid_column_count">2</integer>
</resources>

制作src/main/res/values-large/integers.xml 并添加:

<resources>
    <integer name="grid_column_count">3</integer>
</resources>

注意 -large,它将通知系统此尺寸及以上的屏幕应使用此备用资源值。现在在你的 Fragment 中你可以简单地写:

int columnCount = getResources().getInteger(R.integer.grid_column_count);
GridLayoutManager layout = new GridLayoutManager(getActivity(), columnCount, 
        GridLayoutManager.HORIZONTAL, false);

我提到这一点是因为屏幕大小存储桶(-large)可能不是确定列数的最佳方法。您可能更适合使用屏幕宽度限定符(例如-w600dp),如果设备旋转,它将更新(而在这种情况下大小桶不会改变)。您可以阅读有关资源限定符的更多信息here

【讨论】:

  • 谢谢!关于更新我的适配器:当我为我的图像按钮获取第一个图像时,我收到警告“没有附加适配器:跳过布局”,但我的图像按钮被创建得很好。但是当我尝试向布局中添加第二个图像按钮时,我得到了相同的警告,并且我的应用程序崩溃并给出了 nullPointerException。有什么明显的原因可以导致这种情况吗?
  • @Natalie 如果您不能先​​自己解决,最好发布一个单独的问题,包括来自 Logcat 的整个堆栈跟踪
  • 非常感谢您对如何获取我的屏幕尺寸的建议。您的解决方案更加优雅。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-09-06
  • 2023-03-14
  • 2017-09-04
  • 2021-08-06
  • 2012-02-06
  • 1970-01-01
相关资源
最近更新 更多