Все технические форумы на одном сайте Удобный поиск информации с популярных форумов в одном месте
Вопрос: Ошибка в контекстном меню AppCompatActivity

Здравствуйте!
Я только начинаю изучать разработку под Android.
Использую IDE Android Studio 2.2.2, встроенный эмулятор, Android SDK Platform 24.
Не знаю, по адресу ли обратился, но самостоятельно найти решение у меня не получается, надеюсь на помощь опытных форумчан.

Суть проблемы - решил поэкспериментировать с контекстным меню:

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
public class MainActivity extends Activity {
 
    TextView myTv;
    final int MENU_ID_A =1;
    final int MENU_ID_B =2;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
 
        myTv = (TextView) findViewById(R.id.myTv);
        registerForContextMenu(myTv);
    }
 
    @Override
    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {
        super.onCreateContextMenu(menu, v, menuInfo);
        menu.add(0, MENU_ID_A, 0, "A");
        menu.add(0, MENU_ID_B, 0, "B");
    }
 
    @Override
    public boolean onContextItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case MENU_ID_A:
                Toast.makeText(this, "A", Toast.LENGTH_SHORT).show();
                break;
            case MENU_ID_B:
                Toast.makeText(this, "B", Toast.LENGTH_SHORT).show();
                break;
        }
        return super.onContextItemSelected(item);
    }
}
Здесь меня все устраивает, все работает нормально, но когда я создаю новый модуль с опцией Backwards Compability с тем же самым кодом, за исключением того что класс MainActivity расширяется уже от AppCompatActivity,

Java
1
public class MainActivity extends AppCompatActivity
то происходит следующая ситуация:
При первом вызове, контекстное меню открывается, я выбираю какой-либо пункт, появляется соответствующее сообщение Toast, все хорошо. Но при следующей попытке вызвать меню, эмулятор вылетает с ошибкой.
Первый раз столкнулся с этим когда развлекался с анимацией и пробовал закрыть меню тычком по свободному месту экрана, меню закрывалось, но повторно его уже было не вызвать - тот же самый вылет эмулятора. Однако, если я тыкал именно пункты меню, а не свободное место, то никаких вылетов не было, меню вызывалось сколько угодно раз. И это меня еще больше запутывает. Вот код из того примера:

Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
    @Override
    public boolean onContextItemSelected(MenuItem item) {
        Animation anim = null;
        switch (item.getItemId()) {
            case MENU_ALPHA_ID:
                anim = AnimationUtils.loadAnimation(this, R.anim.myalpha);
                break;
            case MENU_SCALE_ID:
                anim = AnimationUtils.loadAnimation(this, R.anim.myscale);
                break;
            case MENU_ROTATE_ID:
                anim = AnimationUtils.loadAnimation(this, R.anim.myrotate);
                break;
            case MENU_COMBO_ID:
                anim = AnimationUtils.loadAnimation(this, R.anim.mycombo);
                break;
        }
        tv.startAnimation(anim);
        return super.onContextItemSelected(item);
    }
В общем, с Activity все хорошо, с AppCompatActivity все плохо.
Пожалуйста, подскажите, в чем может быть дело? В эмуляторе? Или в версии этой штуки: com.android.support:appcompat-v7:24.2.1 из скрипта Gradle? Или я не правильно работаю с обработчиками событий?
Ответ: Я сам не особо разбираюсь, но те кто разбираются обычно требуют показать код ошибки. И неохотно смотрят если его нет
Вопрос: Как сделать, чтобы кнопка добавлялась не в каждую строку ListView, а только в самый низ?

Здравствуйте, форумчане. Как сделать чтобы кнопка не добавлялась в каждую строку ListView. Кнопка нужна только под ListView.
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
36
37
38
39
40
41
<RelativeLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content">
 
    <Button
           android:id="@+id/button_start"
           android:layout_width="wrap_content"
           android:layout_height="wrap_content"
           android:layout_alignParentBottom="true"
           android:onClick="onClick"
           android:text="Start" 
    />
    
   <ListView
        android:id="@+id/item_list_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentTop="true" 
        android:layout_above="@id/button_start"
        android:layout_marginBottom="30dip">
       
   </ListView>
  
    <ImageView
        android:id="@+id/img_track"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:padding="5dp"
    />
    <TextView
        android:id="@+id/tv_name_track"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/img_track"
        android:padding="5dp"
    />
    
</RelativeLayout>

Ответ: del
Вопрос: Как сделать чтобы кнопка правильно располагалась

Как сделать чтобы кнопка располагалась поверх RecyclerView и в тоже время внизу экрана(привязана к нижней его границе)?
Вот разметка

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
36
37
38
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/tab_fragment_first"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:weightSum="100"
    android:background="@android:color/white">
    <!--android:padding="4dp"-->
 
    <android.support.v7.widget.RecyclerView
        android:id="@+id/photoList"
        android:layout_width="fill_parent"
        android:layout_height="match_parent"
        android:divider="@null"
        android:dividerHeight="0dp"
        android:scrollbars="vertical"
        android:layout_margin="2dp"
        android:horizontalSpacing="10dp"
        android:verticalSpacing="10dp">
    </android.support.v7.widget.RecyclerView>
 
    <LinearLayout
        android:id="@+id/linearLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
 
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="NEXT"
            android:layout_marginLeft="-5dp"/>
 
    </LinearLayout>
 
</RelativeLayout>
Ответ: Ясно
Вопрос: Как сделать кнопку, в которой будет помещена картинка?

Добрый день! Я - новичок, возник такой вопрос: как сделать такую кнопку, чтобы в ней была помещена картинка и текст? Пример во вложении.
Ответ: То что у Вас на картинке это РесайклерВью или на худой конец ЛистВью.
а каждая строка в нем это айтем, айтем делается отдельным хмл. в данном случае испульзуют линеар леаут,горизонтальный, в нем лежит имейдж вью, текст вью а в текст вью справа еще и дровобл добавлен (стрелка вправо)
если вас интересует как сделать такую кнопку, не для ресайклера, вам нужно создать класс, который наследует от линеар лейаут и в его хмл добавить элементы.
Вопрос: Контекстное меню не работает

Доброго времени суток.
Я отбираю записи из базы и формирую список в ListView, насаживая на каждый элемент событие onClick, чтобы открывать запись.
Фрагмент кода ниже
Кликните здесь для просмотра всего текста

Java
1
2
3
4
5
6
7
8
9
10
@Override
    protected void onCreate(Bundle savedInstanceState) {
//...
ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, R.layout.custom_spinner_layout, listRecord);
        adapter.setDropDownViewResource(android.R.layout.simple_list_item_1);
        Singleton.getInstance().setAdapter(adapter); // (!!!)
        ListView listView = (ListView) findViewById(R.id.list_view);
        listView.setAdapter(adapter);
        listView.setOnItemClickListener(mMessageClickedHandler);
}


Следуя примерам из интернета написал следующее:
Кликните здесь для просмотра всего текста

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
public static final int CONTEXT_MENU_EDIT_ITEM = 101;
    public static final int CONTEXT_MENU_DELETE_ITEM = 102;
 
    @Override
    public void onCreateContextMenu(ContextMenu contextMenu, View v, ContextMenu.ContextMenuInfo menuInfo) {
        AdapterView.AdapterContextMenuInfo info = (AdapterView.AdapterContextMenuInfo) menuInfo;
        String selectedWord = ((TextView) info.targetView).getText().toString();
        long selectedWordId = info.id;
 
        contextMenu.setHeaderTitle(selectedWord);
        contextMenu.add(0, CONTEXT_MENU_EDIT_ITEM, 0, "Редактировать");
        contextMenu.add(0, CONTEXT_MENU_DELETE_ITEM, 1, "Удалить");
    }
 
    @Override
    public boolean onContextItemSelected(MenuItem item)
    {
        CharSequence message;
        switch (item.getItemId())
        {
            case CONTEXT_MENU_EDIT_ITEM:
                message = "Выбран пункт Редактировать";
                break;
            case CONTEXT_MENU_DELETE_ITEM:
                message = "Выбран пункт Удалить";
                break;
            default:
                return super.onContextItemSelected(item);
        }
        Toast toast = Toast.makeText(this, message, Toast.LENGTH_SHORT);
        toast.setGravity(Gravity.CENTER, 0, 0);
        toast.show();
        return true;
    }


Затея была в том, чтобы помимо onCLick нацепить на эти элементы еще и OnItemLongClick с диалогом для удаления, но подход через onCreateContextMenu показался мне более простым и идеологически правильным.
Т.е. в конечном итоге я хочу добиться, чтобы при OnItemLongClick всплывал диалог со списком допустимых действий.
Ответ:

Не по теме:


Сообщение от fraley
Во-вторых, по поводу контекстного меню - лень писать, вот первый же пример:

Единственное, в registerForContextMenu прописать ListView.
Спасибо. Забыл зарегистрировать ListView

Вопрос: Обработка нажатия элементов контекстного меню

Здравствуйте!
Выполняю ДЗ по урокам андройд и вот с чем столкнулся.
Java(TM) 2 Platform Standard Edition 5.0
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
public class MainActivity extends AppCompatActivity {
    TextView tvColor, tvSize;
    final int MENU_COLOR_RED=1;
    final int MENU_COLOR_GREEN=2;
    final int MENU_COLOR_BLUE=3;
    final int MENU_SIZE_22=4;
    final int MENU_SIZE_26=5;
    final int MENU_SIZE_30=6;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
       Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        tvColor=(TextView)findViewById(R.id.tvColor);
        tvSize=(TextView)findViewById(R.id.tvSize);
 
        registerForContextMenu(tvColor);
        registerForContextMenu(tvSize);
 
    }
    @Override
    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {
 
 
    switch (v.getId()){
            case R.id.tvColor:
 
//                menu.add(0,MENU_COLOR_RED,0,"RED");
//                menu.add(0,MENU_COLOR_GREEN,0,"GREEN");
//                menu.add(0,MENU_COLOR_BLUE,0,"BLUE");
                getMenuInflater().inflate(R.menu.menu_color, menu);
                  break;
 
            case R.id.tvSize:
//                menu.add(0,MENU_SIZE_22,0,"22"); //
//                menu.add(0,MENU_SIZE_26,0,"26");
//                menu.add(0,MENU_SIZE_30,0,"30");
                getMenuInflater().inflate(R.menu.menu_size, menu);
                  break;
 
        }
      super.onCreateContextMenu(menu, v, menuInfo);
   }
 
    @Override
    public boolean onContextItemSelected(MenuItem item) {
        switch (item.getItemId()){
            case MENU_COLOR_RED:
                tvColor.setTextColor(Color.RED);
                tvColor.setText("Text color red");
                break;
            case MENU_COLOR_GREEN:
                tvColor.setTextColor(Color.GREEN);
                tvColor.setText("Text color green");
                break;
            case MENU_COLOR_BLUE:
                tvColor.setTextColor(Color.BLUE);
                tvColor.setText("Text color blue");
                break;
            case MENU_SIZE_22:
                tvSize.setTextSize(22);
                tvSize.setText("text size 22");
                break;
            case MENU_SIZE_26:
                tvSize.setTextSize(26);
                tvSize.setText("text size 26");
                break;
            case MENU_SIZE_30:
                tvSize.setTextSize(30);
                tvSize.setText("text size 30");
                break;
        }
        return super.onContextItemSelected(item);
    }
Код рабочий,но как, меню выводится,однако ничего ен происходит по нажатию ан элементы.
Понимаю,что с обработчиком что-то нужно делать,А что?
Ответ:
Сообщение от correcеt
switch (item.getItemId()){
* * * * * * case MENU_COLOR_RED:
Вы должны использовать не эти константы, а из R.menu....
Вопрос: Android Studio полностью круглая кнопка

Здравствуйте.
Я хочу сделать круглую кнопку где края не мешают кликать на другие объекты.
У меня есть один activity с кодом:
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
36
37
38
39
40
41
42
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.ualg.a52045.testcirclebuttons.MainActivity">
 
 
    <FrameLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">
 
        <Button
            android:text="Button"
            android:layout_alignParentTop="true"
            android:layout_alignParentLeft="true"
            android:layout_alignParentStart="true"
            android:id="@+id/button2"
            android:background="@drawable/round_button_2"
            android:layout_marginLeft="-130dp"
            android:layout_marginTop="-100dp"
            android:layout_width="480dp"
            android:layout_height="480dp"
            android:cursorVisible="false" />
 
        <Button
            android:text="Button"
            android:layout_alignParentTop="true"
            android:layout_alignParentLeft="true"
            android:layout_alignParentStart="true"
            android:id="@+id/button"
            android:background="@drawable/round_button"
            android:layout_marginLeft="-50dp"
            android:layout_marginTop="-50dp"
            android:layout_width="350dp"
            android:layout_height="350dp" />
 
    </FrameLayout>
 
</RelativeLayout>
и два dravable с кодом:
XML
1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="utf-8"?>
<ripple xmlns:android="http://schemas.android.com/apk/res/android"
    android:color="#ff3500">
    <item>
        <shape android:shape="oval">
            <solid android:color="#fa9e09"/>
        </shape>
    </item>
</ripple>
результат на картинке прикрепленной к теме.

зоны выделены синим принадлежат первой кнопке а я хочу чтобы они принадлежали второй.
Как я ето могу сделать.
Ответ: а как ее сделать? если можно с примером кода.
Вопрос: Listview, adapter получение информации из нажатого элемента списка

Есть список, который заполняется из базы данных. Мне нужно через контекстное меню удалить этот элемент из списка и исходя из информации, которая содержится в списке изменить данные в БД. Для этого мне нужно из элемента списка получить некоторые значения, которые он отображает. Я понял, что это делается через id элемента списка, но запутался в том, как с помощью него получить необходимые данные, которые отображаются в этом элементе... Подскажите пожалуйста.
Вообще я пытался сделать по примеру... Но у меня есть отличия:

1. Мне не нужен массив элементов, т.к. каждый элемент списка представляет собой один билет, который содержит номер рейса и место, который забронировал пользователь. Через контекстное меню надо отменить эту бронь, соотвественно нужно получить номер рейса и номер места, для того, что бы это место отметить в БД как доступное для бронирования.

2. Более важный момент, я не могу ни внутри адаптера ни в классе, которое работает с этим списком воспользоваться переопределенным методом получения id.

В итоге я запутался как это сделать. В идеале оно должно работать так:
1.Создается список и заполняется данными из БД.
2.По нажатию на определенный элемент появляется контекстное меню, через которое я нажимаю на пункт "Отменить бронирование".
3.Программа извлекает из этого поля все данные, для того чтобы в БД правильно отменить бронь. После чего список пересоздается, но уже без этого элемента.
Ответ: Нет ничего хуже чем разбирать чужой код, тем более обрывки.
objects = prof_tick;
надо заменить на
object=new ArrayList<prof_tick>;
for (prof_tick pf_tk : prof_tick) {
object.add(pf_tk);
}
Класс prof_tick должен быть приблизительно такого вида.
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class DataForFolder {// этот класс хранилище данных для описания и отображения наподобии структуры в Си
        File file;//путь к файлу 
        //String nameFolderOrFile;//имя отображаемой папки или файла
        int level;// уровень вложения папки или файла
        boolean open;//если папка открыта то труе, если папака закрыта то фальсе
        int idRecordIntoDB;//_id записи в БД
        public DataForFolder(File file,int level, boolean open,int idRecordIntoDB) {
            // TODO Auto-generated constructor stub
            this.file=file;
            //this.nameFolderOrFile=nameFolderOrFile;
            this.level=level;
            this.open=open;
            this.idRecordIntoDB=idRecordIntoDB;
        }
}
Про меню и контекстное меню пока забудьте. Повесьте на ваш ListView OnItemClickListener. И попробуйте выдернуть вашу информацию в методе onItemClick
через объект вашего prof_adapter prof_adapter.getItem(int position)
П.С. В коде могут быть ошибки. Они там обязательно есть.

Добавлено через 2 часа 2 минуты
Сообщение от Eugen167
Вот код кнопки контекстного меню:
Немного не понятно, зачем на кнопку контекстное меню вешать. Если ваша кнопка входит в состав view. То и делайте обработку при нажатии на кнопку. Код такой.
В адаптере
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        if(convertView == null){
            convertView = layoutInflater.inflate(R.layout.my_list_item_btn,parent,false);//загрузили view
        }
        TextView txtView=(TextView) convertView.findViewById(R.id.tvOut);//нашли  TextView txtView
        txtView.setText(object.get(position));//вывели в txtView текст 
        Button btnDel = (Button) convertView.findViewById(R.id.btnDel);//нашли Button btnDel
        btnDel.setTag(position);//присвоили тег, чтобы знать вкакой позиции нажали кноку
        btnDel.setFocusable(false);//убрали фокус иначе у ListView в MainActivity не будет работать метод onItemClick
        btnDel.setOnClickListener(listener);
        return convertView;
    }
В МаинАктивити обработка нажатия кнопки
Java
1
2
3
4
5
6
7
public void onClick(View v) { //обработка нажатия кнопок
        // TODO Auto-generated method stub
case R.id.btnDel://нажали  Х ,кнопочку удаления записи 
            int position=Byte.valueOf(v.getTag().toString());// вычислили позицию в таблице на которой нажали кнопочку
            по int position можно вытащить из адаптера Object getItem(int position) а в
Object должно ыть прописано все что вам нужно для редактирования БД.
            break;
Не уверен но по такому же принципу можно и контекстное меню сделать.
Вопрос: Как программно удалить кнопку которую программно создал но только если она уже существует

Есть ScrollView в котором много компонентов,чтоб не выводить сразу все решил выводить по 10 штук и снизу пихать кнопку "еще",эту самую кнопку я создаю программно ПОСЛЕ того как код выведет в ScrollView те самые 10 элементов,например я нажимаю на кнопку "еще" и у меня грузятся еще 10 элементов,теперь мне надо удалить старую кнопку и создать ее заново в конце.Собственно как это сделать? (к слову элементов может быть и 5 штук и кнопка "еще" может и не существовать)
Или может есть другие идеи как сделать подобную кнопку?
Ответ:
Вопрос: FrameLayout - скрыть кнопки

Здравствуйте.
Что хочу сделать?
1. У меня есть набор кнопок, и в зависимости от выбранного варианта работы, часть этих кнопок может быть недоступна.
2. Хочу красиво убрать неиспользуемые в данный момент времени кнопки. Можно просто скрывать, но тогда это похоже на выбитые зубы, или разломанную кирпичную стену... Не айс, в общем.
3. Я добавил, 4 FrameLayout и на них для каждого варианта работы разместил нужные кнопки, без пробелов, в общем как надо.
4. И тут проблема, визуально все работает, наборы кнопок красиво сменяют друг друга. НО, при нажатии на оные ничего не происходит.
5. Собственно, почему так происходит понятно - id другие. Получаться мне придется создавать, еще обработчики, привязывать новые кнопки к новым ресурсам, и тоже самое делать с действиями. Это очень много дублирующей работы.
6 Нельзя ли как то использовать только один набор кнопок для всех случаев? Я пробовал просто не создавать каждый раз id а использовать существующий.
Например: первый лайот - @+id/button, а на втором пробовал не создавать ресурс (он уже есть) - 2 лайот - @id/button - ошибок в ресурсах нет, но и эффекта нет.
7. Есть какие нибудь варианты?
Ответ: Спасибо всем за совет, буду пробовать, разбираться.