Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Подключение кнопочной клавиатуры к AVR.
Форум разработчиков электроники ELECTRONIX.ru > Сайт и форум > В помощь начинающему > Схемотехника
Stolbov
Всем добрый вечер!
Вообщем, я делаю небольшую поделку на основе микроконтроллера AVR,
но так как мне нужно было 6 кнопок, а выводов осталось мало, я решил
попробовать "выкрутиться" из этой ситуации таким образом, как это показано
на рисунке.
Нажмите для просмотра прикрепленного файла
PA0 и PA1 - ножки микроконтроллера, на которых постоянно находится
высокий уровень. В первый момент времени я устанавливаю на PB0 низкий
уровень, и, получается, что когда мы нажимаем кнопку S1 или S4, на одном из
входов МК так же установится низкий уровень напряжения. Таким образом,
можно отследить, какая кнопка нажата в момент времени. Остальные кнопки
опрашиваются по этому же алгоритму (подаём 0 на PB1, а потом на PB2).
За сим вопрос: правильно ли я вообще делаю? Когда я проводил эксперимент
с одной кнопкой, всё работало, и я с лёгкостью мог отследить её нажатие, но
когда я собрал эту схему на макетке, всё пошло крахом: ни одна кнопка не отзывается.
Если кто-нибудь сможет разобраться, есть ли у этой схемы шанс заработать,
буду премного благодарен. Код опроса этой конструкции так же прилагаю
(может быть, его можно было бы сделать изящнее, но получилось вот так).
Код
#define STEP1 1 //Метки для атомата
#define STEP2 2
#define STEP3 3

//...

DDRA |= (0<<PORTA0)|(0<<PORTA1); //Настройка ножек порта А на вход
DDRB |= (1<<PORTB0)|(1<<PORTB1)|(1<<PORTB2); //Настройка ножек порта B на выход

//..

int button_flag; //Флаг для цикла
int button_state; //Для хренения состояния автомата

//..

void key_polling()
{
    button_state = STEP1; //Первое состояние
    button_flag = 1; //Устанавливаем флаг
    while(button_flag == 1)
    {
        switch(button_state)
        {
            case STEP1: //Опрашиваем кнопки 1 и 4
                PORTB |= (0<<PORTB0)|(1<<PORTB1)|(1<<PORTB2); //На ножке PB0 устанавливаем низкий уровень
                if((PINA & (1<<PORTA0) == 0)) //Если на PA0 - 0, то...
                {
                    //Некое действие
                    button_state = STEP2; //Переходим к опрос следующей двойки
                }
                else if((PINA & (1<<PORTA1) == 0)) //Если на PA1 - 0, то...
                {
                    //Что-то делаем
                    button_state = STEP2;
                }
                else //Если не получили низкий уровень ни на одной из ножек
                {
                    button_state = STEP2;
                }        
            break;
            
            case STEP1: //Опрашиваем кнопки 2 и 5
                PORTB |= (1<<PORTB0)|(0<<PORTB1)|(1<<PORTB2); //На PB1 устанавливаем низкий уровень
                //Код, аналогичный блоку STEP1
            break;
            
            case STEP3: //Опрашиваем кнопки 3 и 6
                PORTB |= (1<<PORTB0)|(1<<PORTB1)|(0<<PORTB2); //На PB2 устанавливаем низкий уровень
                //Код, аналогичный блоку STEP1
                button_flag = 0; //Так как все кнопки опросили, обнуляем флаг для выхода из цикла
            break;
        }
    }
}
@Ark
Ноги порта PB должны быть первоначально конфигурированы как входы.
При опросе, переключаете по одной на выход и устанавливаете низкий уровень.
По завершению опроса - снова переключаете на вход. И так - для всех ног порта PB.
Plain
Скорость работы неизвестна, и кнопки фантастические, без дребезга.
vladec
Цитата
(может быть, его можно было бы сделать изящнее, но получилось вот так).

Для шести кнопок в контроллере можно легко обойтись одним входом подключенным к АЦП контроллера. Для этого запитываете цепочку из семи последовательно включенных одинаковых резисторов (последний на землю), после каждого из остальных резисторов ставите кнопку на землю, с последнего резистора сигнал на АЦП контроллера. По измеренному напряжению определяете нажатую кнопку.
@Ark
Цитата(vladec @ Aug 15 2018, 09:25) *
Для шести кнопок в контроллере можно легко обойтись одним входом подключенным к АЦП контроллера. Для этого запитываете цепочку из семи последовательно включенных одинаковых резисторов (последний на землю), после каждого из остальных резисторов ставите кнопку на землю, с последнего резистора сигнал на АЦП контроллера. По измеренному напряжению определяете нажатую кнопку.

Как отследить одновременное нажатие нескольких кнопок?

jcxz
Цитата(Stolbov @ Aug 14 2018, 21:23) *
За сим вопрос: правильно ли я вообще делаю?

Неправильно если могут быть одновременно нажаты несколько кнопок. Необходимы диоды.
jeelman
если не требуется большая защищённость от помех и разрядов то можно сделать наподобие того как во вложении. подпрограмму опроса запускать в прерывании.
Arlleex
Цитата(jcxz @ Aug 15 2018, 10:53) *
Неправильно если могут быть одновременно нажаты несколько кнопок. Необходимы диоды.

Да можно и без диодов, в общем-то. Сканировать не жесткими Push-Pull-ными '0' и '1', а '0' и 'Hi-Z'. КЗ не случится.
haker_fox
QUOTE (Stolbov @ Aug 15 2018, 02:23) *
Если кто-нибудь сможет разобраться, есть ли у этой схемы шанс заработать,
буду премного благодарен.

Ваше решение - матричная клавиатура. http://cxem.net/mc/book45.php
Впрочем, вы почти её и реализовали. Но всё же лучше сделать, как на схеме по ссылке. Поставить подтягивающие резисторы и диоды.
Тогда, настроив подтянутые к VCC пины на вход, и подавая "бегущий" ноль на выходы (с диодами), вы можете определить нажатую клавишу. Ну или несколько клавиш (для этого диоды и нужны).
С программой, я думаю, справитесь. Там алгоритм не очень сложный. Но это зависит от того, что вы хотите получить. Например, вам нужна задержка перед автоповтором, задержка между автоповтором, но быстрая реакция на нажатие, если клавишу нажали и тут же отпустили? Как в компьютере?
Arlleex
Цитата(haker_fox @ Aug 15 2018, 13:56) *
Ваше решение - матричная клавиатура. http://cxem.net/mc/book45.php
Впрочем, вы почти её и реализовали. Но всё же лучше сделать, как на схеме по ссылке. Поставить подтягивающие резисторы и диоды.
Тогда, настроив подтянутые к VCC пины на вход, и подавая "бегущий" ноль на выходы (с диодами), вы можете определить нажатую клавишу. Ну или несколько клавиш (для этого диоды и нужны).
С программой, я думаю, справитесь. Там алгоритм не очень сложный. Но это зависит от того, что вы хотите получить. Например, вам нужна задержка перед автоповтором, задержка между автоповтором, но быстрая реакция на нажатие, если клавишу нажали и тут же отпустили? Как в компьютере?

Опять же, диоды можно не ставить rolleyes.gif
Stolbov
haker_fox, мне нужна быстрая реакция на нажатие. Нажал кнопку - произошло некоторое действие. Скорее всего, сделаю так, как Вы и сказали. Я сейчас пробовал делать по совету @Ark'а (он писал про то, что ножки, по которым у меня происходит опрос кнопок, должны быть настроены как входы), но у меня выходит что-то невразумительное, но тут уже, наверное, дело в моих навыках программиста.
haker_fox
QUOTE (Stolbov @ Aug 15 2018, 18:32) *
haker_fox, мне нужна быстрая реакция на нажатие.

Ну что значит "быстрая?". Вы ведь кнопки обрабатываете. И потом, какое отношение матричная клавиатура имеет к понятию "быстродействие"? Опять же, вам понадобится фильтрация дребезга, если эти кнопки не с волешбной планеты Нибиру (бездребезговые, от слова "совсем"). А подавление дребезга - это уже около 10 - 50 мс. Это быстро для вас или меделенно?

QUOTE (Arlleex @ Aug 15 2018, 18:05) *
Опять же, диоды можно не ставить rolleyes.gif

Ну кому как нравится)
jcxz
Цитата(Arlleex @ Aug 15 2018, 10:49) *
Да можно и без диодов, в общем-то. Сканировать не жесткими Push-Pull-ными '0' и '1', а '0' и 'Hi-Z'. КЗ не случится.

Без диодов невозможно корректно сканировать если возможны произвольные одновременные нажатия. От слова - никак. Никакие пуш-пуллы тут не помогут.

Цитата(haker_fox @ Aug 15 2018, 12:56) *
Ваше решение - матричная клавиатура. http://cxem.net/mc/book45.php
Впрочем, вы почти её и реализовали. Но всё же лучше сделать, как на схеме по ссылке. Поставить подтягивающие резисторы и диоды.
Тогда, настроив подтянутые к VCC пины на вход, и подавая "бегущий" ноль на выходы (с диодами), вы можете определить нажатую клавишу. Ну или несколько клавиш (для этого диоды и нужны).

То что на приведённой Вами схеме, нормально будет работать только с некоторыми одновременными нажатиями. Но не со всеми.
При нормальной реализации диоды ставят не так как там, а последовательно с кнопками.

Цитата(haker_fox @ Aug 15 2018, 16:02) *
Ну кому как нравится)

"Нравятся" - это девушки. А схема - или работает или нет. rolleyes.gif
Приведённая Вами не работает со многими тройными нажатиями. А если поставить диоды правильно, то будет работать с любыми комбинациями нажатий.

Цитата(haker_fox @ Aug 15 2018, 16:02) *
Это быстро для вас или меделенно?

Видимо автор полагает что он умеет нажимать на кнопки с частотой 1кГц. biggrin.gif
@Ark
Цитата(Stolbov @ Aug 15 2018, 13:32) *
Я сейчас пробовал делать по совету @Ark'а (он писал про то, что ножки, по которым у меня происходит опрос кнопок, должны быть настроены как входы), но у меня выходит что-то невразумительное, но тут уже, наверное, дело в моих навыках программиста.

Изначально рабочие порты PB должны быть заданы как входы.
Когда надо опросить, например, линию PB0, то переключаете ее временно на выход.
Записываете в этот порт "ноль", выдерживаете некоторую паузу, затем, смотрите - что у Вас на входах порта PA.
Далее снова переводите порт PB0 на вход. Повторяете так для всех рабочих линий порта PB.
P.S. Не нужны вам никакие диоды, эта схема годится.

Stolbov
haker_fox, я просто не верно выразился по поводу "быстродействия". Я же не собираюсь засекать время между нажатием кнопки и выполнением некоего действия. 10 -50 мс для подавления дребезга - это нормально. У меня проблема сейчас какая: я пока использую свою схему, всё же пытаюсь её "добить", но так как у меня между состояниями опроса кнопок нет никаких пауз + дребезг, моя клавиатура превратилась в генератор случайных чисел. Я пока пытаюсь привести всё в "человеческий" вид, но пока не выходит.
haker_fox
QUOTE (jcxz @ Aug 15 2018, 21:34) *
То что на приведённой Вами схеме, нормально будет работать только с некоторыми одновременными нажатиями. Но не со всеми.
При нормальной реализации диоды ставят не так как там, а последовательно с кнопками.

Почему не будет работать? Мы подаём ноль на конкретный столбец. И мы знаем, на какой. Следовательно, если нажать одновременно кнопки и в текущем столце (на который подан ноль), и в других, то они нам ни как не помешают. Но будут опрошены, когда лог. 0 дойдёт до них.
QUOTE (jcxz @ Aug 15 2018, 21:34) *
Приведённая Вами не работает со многими тройными нажатиями.

Приведите, пожалуйста, пример такой тройной комбинации.

QUOTE (Stolbov @ Aug 15 2018, 22:02) *
Я пока пытаюсь привести всё в "человеческий" вид, но пока не выходит.

Ну приведите в порядок железо (схему). Затем программу. Нарисуйте алгоритм обработки клавиатуры на бумажке, смоделируйте его вручную. Многое прояснится.
Plain
Цитата(Stolbov @ Aug 15 2018, 17:02) *
пытаюсь привести всё в "человеческий" вид

... т.е. когда все действия производятся с привязкой к реальному времени, для чего Вам сперва надо было освоить настройку таймеров и прерываний по ним.
jcxz
Цитата(haker_fox @ Aug 15 2018, 17:51) *
Приведите, пожалуйста, пример такой тройной комбинации.

По Вашей ссылке: предположим что нажаты SB1 и SB5. И теперь нажимаем ещё SB4 или SB8 (не отпуская SB1, SB5).
Теперь как Вы определите, что именно нажато - SB4 или SB8 или одновременно SB4 && SB8? smile3046.gif
Аналогично для всех прочих пар SB2 && SB6, SB3 && SB7. И для других столбцов.

Цитата(haker_fox @ Aug 15 2018, 17:51) *
Почему не будет работать? Мы подаём ноль на конкретный столбец. И мы знаем, на какой.

...и он тут же оказывается на другом столбце, как только нажимаются 2 кнопки. После этого уже ничего не знаем. laughing.gif
Без диодов последовательно кнопкам - никак.
Arlleex
Ну да. Такие комбинации я не учел laughing.gif
Без диодов действительно работать не будет.
haker_fox
QUOTE (jcxz @ Aug 15 2018, 23:04) *
Без диодов последовательно кнопкам - никак.

jcxz, искренне благодарен за объяснение! rolleyes.gif
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2024 Invision Power Services, Inc.