Форум программистов, компьютерный форум, киберфорум
Наши страницы

Программирование Android

Войти
Регистрация
Восстановить пароль
 
New man
51 / 42 / 10
Регистрация: 23.05.2011
Сообщений: 344
#1

Список с разными и сложными элементами - Программирование Android

17.02.2016, 02:49. Просмотров 688. Ответов 8
Метки нет (Все метки)

Здравствуйте, дорогие форумчане.
Задумал я тут создать мобильный клиент для VK и у меня появились проблемы.
Мне нужно отобразить список новостей. Так как их много, логично выводить их как-то в виде списка.
Однако новости могут быть разными, на них должно быть несколько разных кнопок.
В общем, как мне сделать список, у которого каждый элемент будет содержать несколько других компонент?
Думал попробовать сделать fragment и выводить их, но как-то не получилось, да и по результатам гугления понял, что это скорее неправильный подход.
Вот на toster есть предложение создать adapter свой, но не хотелось бы переписывать стандартный класс, так как мне кажется, что кто-то эту проблему уже решал (создатели официального приложения, например).

Что-то ссылка не прикрепляется :\

Заранее спасибо за помощь.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
17.02.2016, 02:49
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Список с разными и сложными элементами (Программирование Android):

xsl две колонки с разными элементами - XML/XSL
Пишу xsl-шаблон для быстрого создания excel-файла из xml. Вывести на экран элементы одного типа легко <xsl:for-each...

Список: Как добавлять элемент в список, не в начало и не в конец, а между 1 и 2 элементами списка? - C++
Как добавлять элемент в список, не в начало и не в конец, а между 1 и 2 элементами списка? Вот сам список: #include <conio.h> ...

Вывести весь список и список с исключёнными нечётными элементами - F#
Написать на приложении Visual Studio F# программу, выводящую список, а после - список, с исключёнными нечётными элементами. Есть...

Разработать программу поиска номера максимального элемента в числовой последовательности с разными элементами - Delphi
В визуальной среде программирование: 1) Разработать программу поиска номера элемента с заданным значением в не упорядочной числовой...

Список с разными маркерами - HTML, CSS
Как правильно сделать список чтобы у каждого элемента списка была своя картинка-маркер? создал Вот такой список: <div...

Список с разными типами данных, унаследованным от одного - C++
Доброго времени суток! У меня имеются унаследованные классы от одного , так же имеется vector<общий_тип*> arr. В программе создаются в...

8
Pablito
2525 / 2004 / 624
Регистрация: 12.05.2014
Сообщений: 7,030
Завершенные тесты: 1
17.02.2016, 11:41 #2
HelloWorld с парой кнопок уже сделал?
чем отличается активити от фрагмента понимаешь?
про RecyclerView читал?
0
New man
51 / 42 / 10
Регистрация: 23.05.2011
Сообщений: 344
17.02.2016, 20:44  [ТС] #3
Спасибо за recyclerView, но мне нужно написать приложение, которое будет работать с API выше 4.0.3, а RecyclerView появился только в Android 5.

Hello World писал. Различия между fragment и activity вроде понимаю.
0
Pablito
2525 / 2004 / 624
Регистрация: 12.05.2014
Сообщений: 7,030
Завершенные тесты: 1
17.02.2016, 20:54 #4
Цитата Сообщение от New man Посмотреть сообщение
RecyclerView появился только в Android 5.
нэнэнэ, неверная инфа

Добавлено через 7 минут
на базе RecyclerView написана уже куча велосипедов на любой вкус - и анимации и бесконечная подгрузка
я бы для начала поискал на гитхабе по словам "android recyclerview", может там уже что-то подходящее есть
1
New man
51 / 42 / 10
Регистрация: 23.05.2011
Сообщений: 344
17.02.2016, 22:58  [ТС] #5
Я вот попробовал использовать стандартный ListView и написать свой Adapter.

Вот Adapter.
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
public class BoxAdapter extends BaseAdapter {
    Context ctx;
    LayoutInflater layoutInflater;
    List<AbstractItemClass> objects;
 
    BoxAdapter(Context context, List<AbstractItemClass> items) {
        ctx = context;
        objects = items;
        layoutInflater = (LayoutInflater) ctx.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }
 
    @Override
    public int getCount() {
        return objects.size();
    }
 
    @Override
    public Object getItem(int position) {
        return objects.get(position);
    }
 
    @Override
    public long getItemId(int position) {
        return position;
    }
 
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        View view = convertView;
        AbstractItemClass item = objects.get(position);
        if (view == null) {
            final int id = Item0.class.isInstance(item) ? R.layout.item0 : R.layout.item1;
            view = layoutInflater.inflate(id, parent, false);
        }
        //Инициализируем содержимое по разному в зависимости от того, какого класса наш объект
        if (Item0.class.isInstance(item))
            ((TextView) view.findViewById(R.id.textView)).setText(((Item0) item).text);
        else {
            Item1 i = (Item1) item;
            ((TextView) view.findViewById(R.id.textView2)).setText(i.text1);
            ((TextView) view.findViewById(R.id.textView3)).setText(i.text2);
        }
        ((Button) view.findViewById(R.id.btn)).setOnClickListener(onClickListener);
        return view;
    }
 
    View.OnClickListener onClickListener = new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            //Do something
        }
    };
}

Это классы данных
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class Item0 extends AbstractItemClass {
    public String text = "";
 
    public Item0(String arg) {
        text = arg;
    }
 
    @Override
    public String toString() {
        return text;
    }
}
 
public class Item1 extends AbstractItemClass {
    public String text1;
    public String text2;
 
    public Item1(String arg1, String arg2) {
        text1 = arg1;
        text2 = arg2;
    }
 
    @Override
    public String toString() {
        final StringBuffer sb = new StringBuffer();
        sb.append(text1).append("\t").append(text2);
        return sb.toString();
    }
}
Вот это MainActivity.
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public class MainActivity extends AppCompatActivity {
 
    final Random rand = new Random();
    List<AbstractItemClass> items = new LinkedList<>();
    BoxAdapter boxAdapter;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        fillData();
        boxAdapter = new BoxAdapter(this, items);
        ((ListView) findViewById(R.id.lvMain)).setAdapter(boxAdapter);
    }
 
    private void fillData() {
        for (int i = 0; i < 50; i++) {
            String str = "" + i;
            AbstractItemClass item = rand.nextBoolean() ? new Item0(str) : new Item1(str,str);
            items.add(item);
        }
    }
}


В результаты разные элементы списка выводились, как на скрине, хотя по макету в версии ZeroType сначала должен был быть текст, а под ним кнопка. Но, как видно на скрине, текст вообще затёрся.
Плюс ко всему, всё это падает при прокрутке вниз с сигналом
02-17 19:52:45.276 27398-27398/? I/Process: Sending signal. PID: 27398 SIG: 9
Как бы это исправить?
Прикрепил скрин того, что получилось.
P.S. А про Github я не подумал, надо поискать, спасибо.
0
Миниатюры
Список с разными и сложными элементами  
AlexCarter
1 / 0 / 0
Регистрация: 26.12.2015
Сообщений: 2
17.02.2016, 23:12 #6
Возможно для решения вашей задачи подойдет стандартный класс SimpleAdapter. Применяя его к списку можно задать для всех строк произвольный layout.
Прикрепил скриншот
0
Миниатюры
Список с разными и сложными элементами  
Spelcrawler
527 / 497 / 111
Регистрация: 12.03.2014
Сообщений: 1,654
Завершенные тесты: 1
18.02.2016, 10:45 #7
Цитата Сообщение от New man Посмотреть сообщение
Я вот попробовал использовать стандартный ListView и написать свой Adapter.
Я бы не советовал сейчас его изучать т.к. RecyclerView удобнее во всем и возможностей куда больше. И он очень даже работает на 4 андроиде.
1
New man
51 / 42 / 10
Регистрация: 23.05.2011
Сообщений: 344
19.02.2016, 14:51  [ТС] #8
Спасибо вам всем за советы.
Разобрался.
Вот годные ссылки на stackoverflow и developer.android. com
Ссылки нерабочие, так как форум рабочие ссылки зачем-то портит
stackoverflow. com/questions/28110033/recyclerview-with-different-cardlayouts
developer.android. com/intl/ru/training/material/lists-cards.html

В общем, смог реализовать список со сложными элементами и разной логикой. Привожу код, так как это может оказаться кому-нибудь полезным.

Так как RecyclerView и CardView - нестандартные, надо в dependencies в файле build.gradle (Module app) добавить
Код
    compile 'com.android.support:cardview-v7:23.0.+'
    compile 'com.android.support:recyclerview-v7:23.0.+'
Вот код элементов, которые нужно представить:
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public abstract class Item {
    public abstract String toString();
}
 
public class Item0 extends Item {
    private String value;
    public Item0(String v){value=v;}
    @Override
    public String toString() {return value;}
}
 
public class Item1 extends Item {
    public boolean checked = false;
    public String caption;
    public Item1(String v){caption = v;}
    @Override
    public String toString() {return caption+": "+checked;}
}


Вот код xml файлов:
Кликните здесь для просмотра всего текста

Код activity
XML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.example.user.tmp.MainActivity">
    <android.support.v7.widget.RecyclerView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/mainRecyclerView"
        android:scrollbars="vertical"
        ></android.support.v7.widget.RecyclerView>
</RelativeLayout>
Код item0.xml:
XML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:card_view="http://schemas.android.com/apk/res-auto"
    android:layout_height="wrap_content"
    android:layout_margin="@dimen/activity_vertical_margin"
    android:layout_width="match_parent">
    <!-- A CardView that contains a TextView -->
    <android.support.v7.widget.CardView xmlns:card_view="http://schemas.android.com/apk/res-auto"
        android:id="@+id/card_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        card_view:cardCornerRadius="4dp">
 
        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >
 
            <TextView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:text="New Text"
                android:id="@+id/textView" />
        </RelativeLayout>
    </android.support.v7.widget.CardView>
 
</LinearLayout>
Код item1.xml:
XML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:card_view="http://schemas.android.com/apk/res-auto"
    android:layout_height="wrap_content"
    android:layout_margin="@dimen/activity_vertical_margin"
    android:layout_width="match_parent">
    <!-- A CardView that contains a TextView -->
    <android.support.v7.widget.CardView xmlns:card_view="http://schemas.android.com/apk/res-auto"
        android:id="@+id/card_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        card_view:cardCornerRadius="4dp">
 
        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content" >
 
            <TextView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_alignParentTop="true"
                android:layout_centerHorizontal="true"
                android:text="New Text"
                android:id="@+id/textView" />
 
            <CheckBox
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@id/textView"
                android:text="New CheckBox"
                android:id="@+id/checkBox" />
        </RelativeLayout>
    </android.support.v7.widget.CardView>
 
</LinearLayout>


Мой переопределенный Adapter. Именно здесь определяется, какой именно View будет в качестве элемента + связываются обработчики нажатий и всё такое.
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
/**
 * Created by Angelicos Phosphohoros on 19.02.2016.
 */
public class MyAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolder> {
 
    private List<Item> data;
 
    public MyAdapter(List<Item> strs){
        data = strs;
    }
 
    @Override
    public int getItemCount() {
        return data.size();
    }
 
    //Переопределяем метод, так как именно результат этого метода отправляется в onCreateViewHolder
    @Override
    public int getItemViewType(int position) throws IllegalArgumentException{
        Item item = data.get(position);
        if (Item0.class.isInstance(item)) return 0;
        if (Item1.class.isInstance(item)) return 1;
        throw  new IllegalArgumentException("Unknown element");
    }
 
    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view;
        RecyclerView.ViewHolder vh;
        //Чтобы понять, откуда мы знаем viewType, см. getItemViewType выше
        switch (viewType){
            case 0: view= LayoutInflater.from(parent.getContext()).inflate(R.layout.item0,parent,false);
                vh = new ViewHolderZero(view);
                break;
            case 1: view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item1,parent,false);
                vh = new ViewHolderOne(view);
                break;
            default: vh = null;
        }
        return vh;
    }
 
    //Заполняем всё данными
    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        if (Item0.class.isInstance(data.get(position))){
            Item0 item = (Item0)data.get(position);
            ((ViewHolderZero)holder).tv.setText(item.toString());
        }
        else{
            Item1 item = (Item1)data.get(position);
            ViewHolderOne vh1 = ((ViewHolderOne) holder);
            vh1.cb.setChecked(item.checked);
            vh1.tv.setText(item.caption);
            //В тэги закидываем наш объект из-за того, что в обработчике надо получить этот объект из кнопки
            vh1.cb.setTag(item);
            vh1.cb.setOnCheckedChangeListener(onCheckedChangeListener);
        }
    }
 
    //Два класса ViewHolder
 
    public class ViewHolderZero extends RecyclerView.ViewHolder{
        TextView tv;
        public ViewHolderZero(View v){
            super(v);
            tv = (TextView)v.findViewById(R.id.textView);
        }
    }
 
    public class ViewHolderOne extends RecyclerView.ViewHolder{
        TextView tv;
        CheckBox cb;
        public ViewHolderOne(View v){
            super(v);
            tv =(TextView)v.findViewById(R.id.textView);
            cb = (CheckBox)v.findViewById(R.id.checkBox);
        }
    }
 
    //Обработчик клика на чекбокс
    CompoundButton.OnCheckedChangeListener onCheckedChangeListener = new CompoundButton.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
            ((Item1)buttonView.getTag()).checked = buttonView.isChecked();
        }
    };
}


Вот описание MainActivity.
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public class MainActivity extends AppCompatActivity {
    private Random random = new Random();
    private RecyclerView recyclerView;
    private RecyclerView.Adapter adapter;
    private RecyclerView.LayoutManager layoutManager;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        recyclerView = (RecyclerView)findViewById(R.id.mainRecyclerView);
        layoutManager = new LinearLayoutManager(this);
        recyclerView.setLayoutManager(layoutManager);
        adapter = new MyAdapter(createData(50));
        recyclerView.setAdapter(adapter);
    }
 
    private List<Item> createData(int length){
        LinkedList<Item> ll = new LinkedList<>();
        for (int i = 0;i<length;i++){
            StringBuilder str = new StringBuilder(i);
            for (int j = 0;j<=i;j+=10)
                str.append('\n').append(j);
            if (random.nextBoolean()){
                ll.add(new Item0("ZeroType\n"+str.toString()));
            }
            else {
                ll.add(new Item1("FirstType\n"+str.toString()));
            }
        }
        return ll;
    }
}
1
Миниатюры
Список с разными и сложными элементами  
Pablito
19.02.2016, 20:45     Список с разными и сложными элементами
  #9

Не по теме:

это было самое очевидное решение )
вся магия в адаптере

0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
19.02.2016, 20:45
Привет! Вот еще темы с ответами:

Дан список с произвольными элементами. Заменить каждый элемент на Т ,если элемент - список - Lisp
Дан список с произвольными элементами. Заменить каждый элемент на Т ,если элемент - список. И на NIL в противном случае.

выпадающий список с разными формами ввода при выборе элемента списка - HTML, CSS
вкратце: на данный момент имеем: &lt;script type=\'text/javascript\'&gt; &lt;!-- function viewdiv(id) { var...

Выпадающий список с элементами из БД - WordPress
На созданной странице, нужно поставить выпадающий список с названиями городов. Список городов расположен в таблице БД вордпреса. Как...

Создать список, элементами которого являются 2^n - Prolog
Создать список, элементами которого являются 2^n,n!, члены последовательности Фибоначчи


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
9
19.02.2016, 20:45
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru