Re: TDA7439 + энкодер + ИК пульт + STANDBY + анализатор спектра
#define IR_1 0x1FEC03F // меню кнопка вверх
#define IR_2 0x1FE40BF // меню кнопка вниз
#define IR_3 0x1FE609F // кнопка >
#define IR_4 0x1FEA05F // кнопка <
#define IR_5 0x1FED827 // BASS++
#define IR_6 0x1FE50AF // BASS--
#define IR_7 0x1FEB04F // TREBLE++
#define IR_8 0x1FE30CF // TREBLE--
#define IR_9 0x1FEF00F // MIDDLE++
#define IR_10 0x1FE00FF // MIDDLE--
#define IR_11 0x1FE58A7 // IN
#define IR_12 0x1FE48B7 // STANDBY
//// VU metr ////////////////////////////
#define AUTO_GAIN 1 // автонастройка по громкости
#define VOL_THR 35 // порог тишины (ниже него отображения на матрице не будет)
#define LOW_PASS 30 // нижний порог чувствительности шумов (нет скачков при отсутствии звука)
#define MAX_COEF 0.5
#define DEF_GAIN 80 // максимальный порог по умолчанию
#define FHT_N 128 // ширина спектра х2
#define LOG_OUT 1
#define _LCD_TYPE 2
/////////////////////////////////////////
#include <FHT.h>
#include <Wire.h>
#include <TDA7439.h>
//#include <LiquidCrystal.h>
#include <font_LCD_1602_RUS.h>
#include <LCD_1602_RUS_ALL.h>
#include <EEPROM.h>
#include <Encoder.h>
#include <MsTimer2.h>
#include <boarddefs.h>
#include <IRremote.h>
IRrecv irrecv(10); // указываем вывод модуля IR приемника
TDA7439 tda;
Encoder myEnc(9, 8);//CLK, DT
decode_results ir;
LCD_1602_RUS lcd(7, 6, 2, 3, 4, 5);// RS,E,D4,D5,D6,D7
#define printByte(args) write(args);
#define cbi(sfr, bit) (_SFR_BYTE(sfr) &= ~_BV(bit))
#define sbi(sfr, bit) (_SFR_BYTE(sfr) |= _BV(bit))
byte posOffset[16] = {2, 3, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30}; // вч выше
int vol,vol_old,vol_d,treb,treb_d,balans,in,standby=1,gain0,gain1,gain2,gain3,gain4;
byte w,w1,in_d,gr1,gr2,www;
int z,z0,z1,bass,bass_d,mids,mids_d,menu,w2;
unsigned long time,time1, oldPosition = -999,newPosition;
unsigned long gainTimer;
byte maxValue, maxValue_f,spek;
float k = 0.1;
byte gain_sp = DEF_GAIN;
int i1,yyy,bbb=1;
byte posLevel_old[16];
void setup() { Serial.begin(9600); Serial.println("TDA7439");
lcd.begin(16, 2);
sbi(ADCSRA, ADPS2);
cbi(ADCSRA, ADPS1);
sbi(ADCSRA, ADPS0);
analogReference(INTERNAL);
irrecv.enableIRIn(); // запускаем модуль IR
pinMode(13,OUTPUT);// STANDBY
digitalWrite(13,HIGH);
lcd.setCursor(0,0);lcd.print("АУДИОПРОЦ 7439");delay(1000);lcd.clear();
pinMode(12,INPUT);// меню кнопка энкодера SW
pinMode(11,INPUT);// in 1-4
pinMode(A2,INPUT);// POWER
pinMode(A0,INPUT);// in R
//pinMode(A1,INPUT);// in L
pinMode(17,OUTPUT);// IN1 (выход A3)
MsTimer2::set(3, to_Timer);MsTimer2::start();
vol = EEPROM.read(0);// vol eeprom
bass = EEPROM.read(1)-7;// bass eeprom
mids = EEPROM.read(2)-7;// mids eeprom
treb = EEPROM.read(3)-7;// treb eeprom
balans = EEPROM.read(4)-4;// balans eeprom
in = EEPROM.read(5);// in eeprom
gain1 = EEPROM.read(6);gain2 = EEPROM.read(7);gain3 = EEPROM.read(8);gain4 = EEPROM.read(9);
switch(in){
case 0: gain0 = gain1;break;
case 1: gain0 = gain2;break;
case 2: gain0 = gain3;break;
case 3: gain0 = gain4;break;
}
if(standby==1){vol_old=vol;vol=0;}
delay(500);audio();
}
void loop() {
if (newPosition != oldPosition) {if(spek==1){cl();menu=0;}}
if(spek==0){
byte a1[8] = {0b00000,0b10101,0b10101,0b10101,0b10101,0b10101,0b10101,0b00000};
byte a2[8] = {0b00000,0b10100,0b10100,0b10100,0b10100,0b10100,0b10100,0b00000};
byte a3[8] = {0b00000,0b10000,0b10000,0b10000,0b10000,0b10000,0b10000,0b00000};
lcd.createChar(0,a1);lcd.createChar(1,a2);lcd.createChar(2,a3);
}else{
byte v1[8] = {0, 0, 0, 0, 0, 0, 0, 21};
byte v2[8] = {0, 0, 0, 0, 0, 0, 21, 21};
byte v3[8] = {0, 0, 0, 0, 0, 21, 21, 21};
byte v4[8] = {0, 0, 0, 0, 21, 21, 21, 21};
byte v5[8] = {0, 0, 0, 21, 21, 21, 21, 21};
byte v6[8] = {0, 0, 21, 21, 21, 21, 21, 21};
byte v7[8] = {0, 21, 21, 21, 21, 21, 21, 21};
byte v8[8] = {21, 21, 21, 21, 21, 21, 21, 21};
lcd.createChar(0, v1);lcd.createChar(1, v2);lcd.createChar(2, v3);lcd.createChar(3, v4);lcd.createChar(4, v5);lcd.createChar(5, v6);lcd.createChar(6, v7);lcd.createChar(7, v8);
}
///////////////////// управление //////////////////////////////////////////
if ( irrecv.decode( &ir )) {Serial.print("0x");Serial.println( ir.value,HEX);time1=millis();time=millis();w=1;w1=1;if(spek==1){menu=0;cl();}irrecv.resume();}// IR приемник
if(digitalRead(13)==LOW&&standby==0){menu++;cl();myEnc.write(0);time=millis();w=1;w2=1;if(menu>4){menu=0;}}// меню
if((digitalRead(11)==HIGH || ir.value==IR_11)&&standby==0){if(menu==5){in++;};cl();myEnc.write(0);time=millis();w=1;w2=1;www=1;menu=5;if(in>3){in=0;};audio();}// in
if(standby==1){menu=100;spek=0;w=0;lcd.setCursor(0,0);lcd.print(" НЕ РАbОТАЕТ ");}
if((analogRead(A2)>1000 || ir.value==IR_12)&&standby==1){standby=0;spek=0;cl();time=millis();w=1;w2=1;lcd.clear();lcd.setCursor(0,0);lcd.print(" ВКЛЮЧЕНО ");
digitalWrite(12,HIGH);vol=vol_old;audio();delay(1000);menu=0;w2=1;}
if((analogRead(A2)>1000 || ir.value==IR_12)&&standby==0){standby=1;spek=0;cl();time=millis();w=1;w2=1;lcd.clear();digitalWrite(12,LOW);vol_old=vol;vol=0;audio();delay(500);menu=100;}
if(ir.value==IR_1&&standby==0){menu++;gr1=0;gr2=0;cl();time=millis();w=1;w2=1;if(menu>4){menu=0;}}// меню кнопка вверх
if(ir.value==IR_2&&standby==0){menu--;gr1=0;gr2=0;cl();time=millis();w=1;w2=1;if(menu<0){menu=4;}}// меню кнопка вниз
if(ir.value==IR_5&&menu!=1&&standby==0){bass++;menu=1;gr1=1;gr2=0;cl();;w=1;w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}
if(ir.value==IR_6&&menu!=1&&standby==0){bass--;menu=1;gr1=0;gr2=1;cl();;w=1;w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}
if(ir.value==IR_7&&menu!=2&&standby==0){mids++;menu=2;gr1=1;gr2=0;cl();;w=1;w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}
if(ir.value==IR_8&&menu!=2&&standby==0){mids--;menu=2;gr1=0;gr2=1;cl();;w=1;w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}
if(ir.value==IR_9&&menu!=3&&standby==0){treb++;menu=3;gr1=1;gr2=0;cl();;w=1;w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}
if(ir.value==IR_10&&menu!=3&&standby==0){treb--;menu=3;gr1=0;gr2=1;cl();w=1;w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}
// индикация громкости + управление //////////////
if(menu==0){
if(ir.value==IR_3){vol++;gr1=1;gr2=0;cl2();w2=1;if(vol<0){vol=0;}if(vol>48){vol=48;}audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){vol++;gr2=0;cl2();w2=1;if(vol<0){vol=0;}if(vol>48){vol=48;}audio();}// кнопка >>>>>>
if(ir.value==IR_4){vol--;gr1=0;gr2=1;cl2();w2=1;if(vol<0){vol=0;}if(vol>48){vol=48;}audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){vol--;gr1=0;cl2();w2=1;if(vol<0){vol=0;}if(vol>48){vol=48;}audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
vol=vol+newPosition;myEnc.write(0);newPosition=0;time=millis();w=1;w2=1;if(vol<0){vol=0;}if(vol>48){vol=48;}audio();}
lcd.setCursor(0,0);lcd.print("Громкость");
lcd.print(" ");lcd.print(String(-48+vol));lcd.print(" ");lcd.setCursor(13,0);lcd.print("dB");vol_d=vol;
if(w2==1){
for(z=0,z0=0,z1=0;z<=vol_d;z++,z1++){if(z1>2){z1=0;z0++;}
if(z1==1){lcd.setCursor(z0,1);lcd.write((uint8_t)0);lcd.setCursor(z0+1,1);lcd.print(" ");}}
if(z1==3){lcd.setCursor(z0,1);lcd.write((uint8_t)1);}
if(z1==2){lcd.setCursor(z0,1);lcd.write((uint8_t)2);}w2=0;}}
// индикация тембр НЧ //////////////
if(menu==1){
if(ir.value==IR_3||ir.value==IR_5){bass++;gr1=1;gr2=0;cl1();w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){bass++;gr2=0;cl1();w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}// кнопка >>>>>>
if(ir.value==IR_4||ir.value==IR_6){bass--;gr1=0;gr2=1;cl1();w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){bass--;gr1=0;cl1();w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
bass=bass+newPosition;myEnc.write(0);newPosition=0;time=millis();w=1;w2=1;if(bass>7){bass=7;}if(bass<-7){bass=-7;}audio();}
lcd.setCursor(0,0);lcd.print("bасс ");
lcd.print(" ");lcd.print(String(bass*2));lcd.print(" ");lcd.setCursor(13,0);lcd.print("dB");bass_d=bass*2+15;
if(w2==1){
for(z=0,z0=0,z1=0;z<=bass_d;z++,z1++){if(z1>2){z1=0;z0++;}
if(z1==1){lcd.setCursor(z0+3,1);lcd.write((uint8_t)0);lcd.setCursor(z0+1+3,1);lcd.print(" ");}}
if(z1==3){lcd.setCursor(z0+3,1);lcd.write((uint8_t)1);}
if(z1==2){lcd.setCursor(z0+3,1);lcd.write((uint8_t)2);}w2=0;}}
// индикация тембр CЧ //////////////
if(menu==2){
if(ir.value==IR_3||ir.value==IR_7){mids++;gr1=1;gr2=0;cl1();w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){mids++;gr2=0;cl1();w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}// кнопка >>>>>>
if(ir.value==IR_4||ir.value==IR_8){mids--;gr1=0;gr2=1;cl1();w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){mids--;gr1=0;cl1();w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
mids=mids+newPosition;myEnc.write(0);newPosition=0;time=millis();w=1;w2=1;if(mids>7){mids=7;}if(mids<-7){mids=-7;}audio();}
lcd.setCursor(0,0);lcd.print("Средние ");
lcd.print(" ");lcd.print(String(mids*2));lcd.print(" ");lcd.setCursor(13,0);lcd.print("dB");mids_d=mids*2+15;
if(w2==1){
for(z=0,z0=0,z1=0;z<=mids_d;z++,z1++){if(z1>2){z1=0;z0++;}
if(z1==1){lcd.setCursor(z0+3,1);lcd.write((uint8_t)0);lcd.setCursor(z0+1+3,1);lcd.print(" ");}}
if(z1==3){lcd.setCursor(z0+3,1);lcd.write((uint8_t)1);}
if(z1==2){lcd.setCursor(z0+3,1);lcd.write((uint8_t)2);}w2=0;}}
// индикация тембр BЧ //////////////
if(menu==3){
if(ir.value==IR_3||ir.value==IR_9){treb++;gr1=1;gr2=0;cl1();w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){treb++;gr2=0;cl1();w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}// кнопка >>>>>>
if(ir.value==IR_4||ir.value==IR_10){treb--;gr1=0;gr2=1;cl1();w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){treb--;gr1=0;cl1();w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
treb=treb+newPosition;myEnc.write(0);newPosition=0;time=millis();w=1;w2=1;if(treb>7){treb=7;}if(treb<-7){treb=-7;}audio();}
lcd.setCursor(0,0);lcd.print("Низкие ");
lcd.print(" ");lcd.print(String(treb*2));lcd.print(" ");lcd.setCursor(13,0);lcd.print("dB");treb_d=treb*2+15;
if(w2==1){
for(z=0,z0=0,z1=0;z<=treb_d;z++,z1++){if(z1>2){z1=0;z0++;}
if(z1==1){lcd.setCursor(z0+3,1);lcd.write((uint8_t)0);lcd.setCursor(z0+1+3,1);lcd.print(" ");}}
if(z1==3){lcd.setCursor(z0+3,1);lcd.write((uint8_t)1);}
if(z1==2){lcd.setCursor(z0+3,1);lcd.write((uint8_t)2);}w2=0;}}
///////////////////////////////////////////////////////////
// индикация баланс + управление кнопками +\- 4 дБ ///////
if(menu==4){
if(ir.value==IR_3){balans++;gr1=1;gr2=0;cl();if(balans>4){balans=4;}if(balans<-4){balans=-4;}audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){balans++;gr2=0;cl();if(balans>4){balans=4;}if(balans<-4){balans=-4;}audio();}// кнопка >>>>>>
if(ir.value==IR_4){balans--;gr1=0;gr2=1;cl();if(balans>4){balans=4;}if(balans<-4){balans=-4;}audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){balans--;gr1=0;cl();if(balans>4){balans=4;}if(balans<-4){balans=-4;}audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
balans=balans+newPosition;myEnc.write(0);newPosition=0;lcd.clear();time=millis();w=1;if(balans>4){balans=4;}if(balans<-4){balans=-4;}audio();}
lcd.setCursor(0,0);
if(balans>=0){lcd.print("-");}else{lcd.print("+");}
lcd.print(String abs(balans));lcd.print(" dB ");
lcd.print(" <> ");
if(balans>=0){lcd.print("+");}else{lcd.print("-");}
lcd.print(String abs(balans));lcd.print(" dB ");
lcd.setCursor(0,1);lcd.print("L");
lcd.setCursor(15,1);lcd.print("R");
if(balans<0){lcd.setCursor(balans+7,1);lcd.write((uint8_t)0);}
if(balans>0){lcd.setCursor(balans+8,1);lcd.write((uint8_t)0);}
if(balans==0){lcd.setCursor(7,1);lcd.write((uint8_t)0);lcd.setCursor(8,1);lcd.write((uint8_t)0);}
}
///////////////////// input + gain
if(in==1&&standby==0){digitalWrite(17,HIGH);}else{digitalWrite(17,LOW);}
if(menu==5){
switch(in){
case 0: gain0 = gain1;break;
case 1: gain0 = gain2;break;
case 2: gain0 = gain3;break;
case 3: gain0 = gain4;break;
}
if(ir.value==IR_3){gain0++;gr1=1;gr2=0;cl1();lcd.clear();w2=1;gain_func();audio();}// кнопка >
if(ir.value==0xFFFFFFFF and gr1==1){gain0++;gr2=0;cl1();lcd.clear();w2=1;gain_func();audio();}// кнопка >>>>>>
if(ir.value==IR_4){gain0--;gr1=0;gr2=1;cl1();lcd.clear();w2=1;gain_func();audio();}// кнопка <
if(ir.value==0xFFFFFFFF and gr2==1){gain0--;gr1=0;cl1();lcd.clear();w2=1;gain_func();audio();}// кнопка <<<<<<
if (newPosition != oldPosition) {
oldPosition = newPosition;
gain0=gain0+newPosition;myEnc.write(0);newPosition=0;lcd.clear();time=millis();w=1;www=1;gain_func();audio();}
switch(in){
case 0: gain1 = gain0;break;
case 1: gain2 = gain0;break;
case 2: gain3 = gain0;break;
case 3: gain4 = gain0;break;
}
if(www==1){audio();www=0;}
lcd.setCursor(0,0);
switch(in){
case 0: lcd.print(" USB ");break;
case 1: lcd.print(" AUX ");break;
case 2: lcd.print(" RADIO ");break;
case 3: lcd.print(" BLUETOOTH ");break;
}
lcd.setCursor(1,1);lcd.print("GAIN");lcd.print(String(gain0));lcd.print(" dB");
}
/////////////////////////////////////////////////////////////
if(millis()-time>10000 && w==1){// запись всех настроек в EEPROM
EEPROM.update(0,vol);
EEPROM.update(4,balans+4);
EEPROM.update(1,bass+7);
EEPROM.update(2,mids+7);
EEPROM.update(3,treb+7);
EEPROM.update(5,in);
EEPROM.update(6,gain1);EEPROM.update(7,gain2);EEPROM.update(8,gain3);EEPROM.update(9,gain4);
lcd.clear();w=0,w2=1;menu=100;spek=1;
}
if(spek==1){
analyzeAudio(); // функция FHT, забивает массив fht_log_out[] величинами по спектру
for (int pos = 0; pos < 16; pos++) { // для окошек дисплея с 0 по 15
// найти максимум из пачки тонов
if (fht_log_out[posOffset[pos]] > maxValue) maxValue = fht_log_out[posOffset[pos]];
lcd.setCursor(pos, 0);
// преобразовать значение величины спектра в диапазон 0..15 с учётом настроек
int posLevel = map(fht_log_out[posOffset[pos]], LOW_PASS, gain_sp, 0, 15);
posLevel = constrain(posLevel, 0, 15);
if (posLevel > 7) { // если значение больше 7 (значит нижний квадратик будет полный)
lcd.printByte(posLevel - 8); // верхний квадратик залить тем что осталось
lcd.setCursor(pos, 1); // перейти на нижний квадратик
lcd.printByte(7); // залить его полностью
} else { // если значение меньше 8
lcd.print(" "); // верхний квадратик пустой
lcd.setCursor(pos, 1); // нижний квадратик
lcd.printByte(posLevel); // залить полосками
}
}
// if (AUTO_GAIN) {
// maxValue_f = maxValue * k + maxValue_f * (1 - k);
// if (millis() - gainTimer > 1500) { // каждые 1500 мс
// // если максимальное значение больше порога, взять его как максимум для отображения
// if (maxValue_f > VOL_THR) gain_sp = maxValue_f;
// если нет, то взять порог побольше, чтобы шумы вообще не проходили
// else gain_sp = 100;
// gainTimer = millis();
// }
// }else{gain_sp = DEF_GAIN;}
// }
//}//void
if (AUTO_GAIN) {
if (millis() - gainTimer > 10) { // каждые 10 мс
maxValue_f = maxValue * k + maxValue_f * (1 - k);
// если максимальное значение больше порога, взять его как максимум для отображения
if (maxValue_f > LOW_PASS) gain_sp = (float) MAX_COEF * maxValue_f;
// если нет, то взять порог побольше, чтобы шумы вообще не проходили
else gain_sp = 100;
gainTimer = millis();
}
}
}
}
void gain_func(){if(gain0<0){gain0=0;}if(gain0>15){gain0=15;}}
void to_Timer(){newPosition = myEnc.read()/4;}
void cl(){ir.value=0;delay(300);lcd.clear();spek=0;}
void cl2(){ir.value=0;delay(50);spek=0;}
void cl1(){ir.value=0;delay(200);spek=0;}
void audio(){
tda.setInput(in+1); // 1 to 4 // номер канала
tda.inputGain(gain0); // 0 to 15 // усиление
tda.setVolume(vol); // 0 to 48 ( 0 is mute) // громкость
tda.setSnd(bass, 1); //-7 to +7 , 1 - Bass | 2 - Mids | 3 - Trebble
tda.setSnd(mids, 2);
tda.setSnd(treb, 3);
tda.spkAtt(4-balans, 4+balans); // баланс R L от 0 to 79 (db)
}
void analyzeAudio() {
cli();
for (int i = 0 ; i < FHT_N ; i++) {
int sample = analogRead(A0);
fht_input[ i] = sample; // put real data into bins
}
fht_window(); // window the data for better frequency response
fht_reorder(); // reorder the data before doing the fht
fht_run(); // process the data in the fht
fht_mag_log(); // take the output of the fht
sei();