SSaluti a tutti, mi chiamo Elia. Mi diletto anch'io di elettronica ed affini. Pensando di fare cosa gradita ho pensato di fare uno schemino dei collegamenti sul progetto di Windom che allego. In attesa dei nuovi suggerimenti proposti da Windom, ed ulteriori sviluppi, attendo scalpitando. segnalate eventuali errori. Saluti.
Nell' attesa cher venga pubblicato l'articolo, anticipo lo schema di come ho collegato il tutto al fine di controllare il puntamento via bus I2C.
I motori funzionano a 12 volt e non possono essere controllati direttamente da Arduino che verrebbe bruciato immediatamente,occorre interporre un driver per motori C.C. come indicato nello schema.
Siccome serve a poco lo schema senza il software,se qualcuno mi spiega come fare, pubblico anche questo. Resta il problema che il tutto è fatto in funzione della meccanica di rotazione modificata con l'aggiunta di un potenziometro per ottenera la posizione della meccanica stessa, come verrà spiegato dettagliatamente alla pubblicazione dell' articolo.
Grazie windom46 per lo schema. Adesso comincio a smanettare, qualcosa verrà fuori. Se non ti crea molto disturbo sarebbe molto gradito il software. Puoi allegarlo utilizzando il pulsante a sinistra "incolla da word". chiaramente deve essere in formato word. Grazie anticipatamente, saluti e a buon rendere.
Come suggeritomi da chimera, allego il software per Arduino (IDE1.05) che sto utilizzando attualmente in giro per la Sardegna da oltre un mese senza problemi.
Mentre ci sono, anticipo anche il link del video pubblicato su Youtube sulle ultime modifiche apportate e sul nuovo sistema di controllo via bluetooth con il Galaxy S3.
software per Arduino:
/* CONTROLLO BRANDEGGIO (Motore303) per Antenna SELFSAT, con ARDUINO-UNO, bussola * CMPS10 e Display 16x2 in modalità I2C (SatWindom_I2C_rev.2) * *********************************************************************************/ #include <Wire.h> // include librerie #include <LiquidCrystal_I2C.h> #include <EEPROM.h> #include <avr/wdt.h> #define Reset_AVR() wdt_enable(WDTO_30MS) // softReset sistema #define ADDRESS 0x60 // definisce indirizzo CMPS10 #define inSF A1 // pin in SatFinder #define inCom A2 // pin in Pulsanti #define inPot A3 // pin in Potenziometro #define ledMA 12 // pin led man/auto #define inMot 13 // pin motore acceso #define motDa 9 // pin motore direzione(1) #define motDb 10 // pin motore direzione(2) #define motEa 3 // pin motore elevazione(1) #define motEb 11 // pin motore elevazione(2) LiquidCrystal_I2C lcd(0x20,16,2); // setta LCD (address, colonne, linee)
int levelSF; // livello segnale del SatFinder char level = -1; // barra segnale SatFinder byte oldState = LOW; // imposta stato bistabile int dirOut; // lettura bussola in gradi int dirHB = EEPROM.read(0); // legge direziona dalla cella 0 int elevHB = EEPROM.read(1); // legge elevazione dalla cella 1 byte pass = 0; // num.loop per limitare scrittura display const byte sicRot = 60; // sicurezza rotazione const byte centro = 183; // rotazione a 180° int posPot; // posizione rotazione antenna const byte range = 1; // tolleranza su angolazioni byte dirLock = 0; // direzione OK byte elevLock = 0; // elevazione OK const byte velRid = 100; // riduzione velocità a 100/255 const byte prox = 5; // gradi mancanti alla posizione byte centroRot = 0; // centro rotazione meccanica int manButton; // ingresso comandi manuali char elevOut, skew; // valore elevazione e skew(da CMPS10)
void setup(){ lcd.init(); // inizializza lcd pinMode(inSF, INPUT); // ingresso livello SatFinder pinMode(inPot, INPUT); // cursore potenziometro rotazione pinMode(inCom, INPUT); // ingresso pulsanti comandi manuali pinMode(ledMA, OUTPUT); // LED manuale/auto pinMode(inMot, INPUT); // sensore motore acceso pinMode(motDa, OUTPUT); // motore orientamento 9/10 pinMode(motDb, OUTPUT); pinMode(motEa, OUTPUT); // motore elevazione 3/11 pinMode(motEb, OUTPUT); // pin digitali 0-1-2-4-5-6-7-8 e analogico A0, liberi. ferma(); // ferma i motori if (inMot == LOW){lcd.backlight();} // accende display // visualizza su display i valori memorizzati nella EEPROM lcd.setCursor(0,0); lcd.print("**** HOTBIRD ***"); lcd.setCursor(0,1); lcd.print("Dir."); lcd.print(dirHB, DEC); lcd.setCursor(9,1); lcd.print("Elev."); lcd.print(elevHB, DEC); lcd.print(" "); lcd.print(" "); delay(3000); // attesa 3 secondi lcd.clear(); }
void loop() { byte highByte, lowByte; // valori di highByte e lowByte per dirOut Wire.beginTransmission(ADDRESS); // inizio comunicazione con CMPS10 Wire.write(2); // invia registro per la lettura Wire.endTransmission(); Wire.requestFrom(ADDRESS, 4); // richiesta di 4 bytes da CMPS10 while(Wire.available() < 4); // attesa disponibilità bites highByte = Wire.read(); // lettura azimut(direzione) lowByte = Wire.read(); // lettura decimali azimut(direzione) elevOut = Wire.read()+57; // lettura elevazione+compensazione skew = Wire.read(); // lettura angolazione dirOut = ((highByte<<8)+lowByte)/10; // calcolo direzione manButton = analogRead(inCom); // valori comandi manuali immessi su(A2) posPot = map(analogRead(inPot),102,924,0,359); // posizione rotazione rimappata levelSF = analogRead(inSF); // livello segnale SatFinder // Partitore resistivo per inviare 6 valori (100-200-300-400-500-600) // sul pin(A2) e ottenere un convertitore A/D per le varie funzioni. // 7,8K verso 5v.+ 6 da 2,2k per il partitore + 1 da 100k tra A2 e massa
// ********* BISTABILE COMMUTAZIONE MAN./AUTO ****************** if (manButton > 50 && manButton < 150) { // manButton=100 (Manuale/Auto) if (oldState == LOW) { oldState = HIGH; // attiva automatismi e digitalWrite(ledMA, HIGH); // accende LED elevLock = 0; // imposta elevLock e dirLock a 0 dirLock = 0; }else{ oldState = LOW; // attiva comandi manuali e digitalWrite(ledMA, LOW); // spegne LED } delay(500); // attende 1/2 secondo } if (oldState == LOW) { // comandi manuali attivato if (manButton < 50) ferma(); // arresta motori if (manButton > 150 && manButton < 250) salva(); // 200;salva valori Elev.Dir. nella EPROM if (manButton > 250 && manButton < 350) alza(); // 300;alza antenna if (manButton > 350 && manButton < 450) abbassa(); // 400;abbassa antenna if (manButton > 450 && manButton < 550 && elevOut < sicRot) orario(); // 500;hiude antenna if (manButton > 550 && manButton < 650 && elevOut < sicRot) antiorario();// 600;apre antenna }else{orienta();} if (digitalRead(inMot) == HIGH) chiude(); // chiude ant.a motore acceso // ***** Scrive su LCD 16x2 con livello segnale ***** pass = pass+1; // limitazione scrittura display if (pass == 1) { if (elevOut > 85) { // controllo livello camper lcd.setCursor(0,0); lcd.print("LIVELLO___CAMPER"); lcd.setCursor(0,1); if (((88-elevOut)==0) && ((skew-3)== 0)) { // usa gli assi pitch,roll(da compensare) lcd.print("*** IN PIANO ***"); // per controllare il livello del camper }else{ lcd.print("A/P "); lcd.print(88-elevOut, DEC); lcd.print(" "); lcd.setCursor(9,1); lcd.print("Lat."); lcd.print(skew-3, DEC); lcd.print(" "); } }else{ // controllo posizionamento antenna lcd.backlight(); lcd.setCursor(0,0); lcd.print("Elev."); lcd.print(elevOut, DEC); lcd.print(" "); lcd.setCursor(9,0); lcd.print("Dir."); lcd.print(dirOut, DEC); lcd.print(" "); lcd.setCursor(0,1); if (levelSF > 30) { // formazione barra livello segnale lcd.print("Level."); for (int lev=0; lev<=levelSF/8; lev++) { lcd.print(level); } lcd.print(" "); }else{ lcd.print("Segn."); lcd.print(levelSF); lcd.print(" "); lcd.setCursor(9,1); lcd.print("Pos."); lcd.print(posPot, DEC); lcd.print(" "); } } } if (pass > 50) {pass = 0;} // scrive su display ogni 50 loop delay(10); } /*********** FUNZIONI RIPETITIVE *****************/ void alza() { digitalWrite(motEa, HIGH); if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut < 30){ analogWrite(motEb, velRid); }else{ digitalWrite(motEb, LOW); } } /***************************************************/ void abbassa() { digitalWrite(motEb, HIGH); if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut > 80){ analogWrite(motEa, velRid); }else{ digitalWrite(motEa, LOW); } } /***************************************************/ void orario() { // gira in senso orario digitalWrite(motDb, HIGH); if (((dirOut > dirHB-prox) && (dirOut < dirHB+prox)) || posPot > 345){ analogWrite(motDa, velRid); }else{ digitalWrite(motDa, LOW); } } /***************************************************/ void antiorario() { // gira in senso antiorario digitalWrite(motDa, HIGH); if (((dirOut > dirHB-prox) && (dirOut < dirHB+prox)) || posPot < 15){ analogWrite(motDb, velRid); }else{ digitalWrite(motDb, LOW); } } /***************************************************/ void ferma() { // ferma rotazione motori digitalWrite(motDa, HIGH); digitalWrite(motDb, HIGH); digitalWrite(motEb, HIGH); digitalWrite(motEa, HIGH); oldState = LOW; // disattiva automatismi e digitalWrite(ledMA, LOW); // spegne LED } /***************************************************/ void salva() { EEPROM.write(0, dirOut); // memorizza direzione nella cella 0 EEPROM.write(1, elevOut); // memorizza elevazione nella cella 1 delay(500); Reset_AVR(); // resetta sistema } /***************************************************/ void chiude() { // chiude antenna (MOTORE acceso) oldState = LOW; // disattiva automatismi e digitalWrite(ledMA, LOW); // spegne LED if (centroRot == 0){ if (elevOut > sicRot) {alza(); // porta elevazione in sicurezza }else{ if (posPot > centro + range) antiorario(); if (posPot < centro - range) orario(); // porta rotazione a 180° } } if (posPot == centro) { centroRot = 1; digitalWrite(motDa, HIGH); // ferma rotazione antenna digitalWrite(motDb, HIGH); } if (centroRot == 1) abbassa(); } /***************************************************/ void orienta() { // mantiene l'antenna entro i valore impostati if (elevOut > elevHB + range) {alza(); // alza antenna }else{ // orienta antenna se elevOut < elevHB if (dirOut > dirHB + range) antiorario(); if (dirOut < dirHB - range) orario(); if ((dirOut > dirHB - range) && (dirOut < dirHB + range)) { digitalWrite(motDa, HIGH); // ferma rotazione antenna digitalWrite(motDb, HIGH); dirLock = 1; // direzione OK } if (elevOut > elevHB + range) alza(); // alza antenna if (elevOut < elevHB - range) abbassa(); // abbassa antenna if ((elevOut < elevHB + range) && elevOut > (elevHB - range)) { digitalWrite(motEa, HIGH); // ferma elevazione antenna digitalWrite(motEb, HIGH); elevLock = 1; // elevazione OK } } if (((elevLock == 1) && (dirLock == 1)) || (levelSF > 80)) centra(); } /*** Centraggio perfetto satellite. Programma ancora da sviluppare, in attesa di perfezionare il movimento della meccanica. Sostituendo almeno il motore della rotazione con uno Stepper, si avrebbe la massima precisione e la possibilità di bloccare la meccanica, con un minimo di tensione PWM su una fase. Resta da verificare l'influenza dei maggiori campi magnetici creati sul funzionamento della bussola. Per il momento la ricerca si ferma se durante la scansione, in prossimità delle posizioni salvate, il livello segnale supera il 30% ********************/ void centra(){ if (levelSF > 30) { // se il segnale supera il 30%, si ferma. ferma(); oldState = LOW; // attiva comandi manuali e digitalWrite(ledMA, LOW); // spegne LED } } /******************************* FINE PROGRAMMA ********************************************/
Mi sono accorto di un erore del software rispetto allo schema elettrico, dovuto al fatto che a me non funzionava più il pin analogico A0 e quindi avevo usato i pin A1,A2 e A3.
Si prega pertanto di correggere le linee:
#define inSF A1 // pin in SatFinder #define inCom A2 // pin in Pulsanti #define inPot A3 // pin in Potenziometro
nel seguente modo:
#define inSF A0 // pin in SatFinder #define inPot A1 // pin in Potenziometro #define inCom A2 // pin in Pulsanti
Va detto che si possono usare i pin analogici che si desidera, basta che ci sia corrispondenza tra schema elettrico e software.
aiuto! sono in possesso di un brandeggio come quello di windom46 sprovvisto di motorini qualcuno mi può dare una dritta a quale modello mi devo orientare e possibilmente dove poterli reperire. Chiaramente devo pilotarli con arduino utilizzando il progetto di windom46. Grazie
Dovessi cambiare i motori adesso, però, sull' elevazione, essendo quello originale un po' lento, metterei il RH15812630, sempre a pag.125, che fa 9 RPM.
Questo è l'interno dei motoriduttori originali, ingranaggio che sforza meno in nylon e gli altri metallici, niente di eccezionale. Quelli che dici tu, almeno dalla struttura esterna, sono senz'altro meglio, vedo pure che ci sono con diverso rapporto di riduzione. Non resta che provare (altezza 63 mm....ci stanno?). Come fai ad accoppiarli alla meccanica senza ingranaggi? Fai tutto con pulegge e cinghie dentate?
Io, come vedi all'inizio del video, l'ho fatto sulla rotazione, riducendo a zero i giochi di accoppiamento.
La cosa comporta diverse modifiche alla meccanica, che saranno spiegate dettagliatamente non appena in redazione troveranno il tempo di pubblicare l'articolo.
i vecchi motorini funzionavano a 24v ed erano accoppiati tramite un ingranaggio in metallo che sono riuscito a conservare. Ho notato che i vecchi motori hanno lo stesso diametro dell'asse di rotazione, quindi posso utilizzare l'ingranaggio che viene blocccato con una coppiglia che funge da fermo. Comunque sto valutando l'idea di poter utilizzare qualche motore di alzacristallo di qualche auto, ma solo se sarà possibile utilizzare l'ingranaggio originale. Domani proverò a fare una foto all'interno del brandeggio.
i vecchi motorini funzionavano a 24v ed erano accoppiati tramite un ingranaggio in metallo che sono riuscito a conservare. Ho notato che i vecchi motori hanno lo stesso diametro dell'asse di rotazione, quindi posso utilizzare l'ingranaggio che viene blocccato con una coppiglia che funge da fermo. Comunque sto valutando l'idea di poter utilizzare qualche motore di alzacristallo di qualche auto, ma solo se sarà possibile utilizzare l'ingranaggio originale. Domani proverò a fare una foto all'interno del brandeggio.
Ciao tedesto, se ti riferisci al controllo via bluetooth con cellulare Android del video, l’app l’ho fatta con MitAppInventor(beta). Con la versione 2, trovo che il BlocksEditor non vada avanti e poi sia di difficile gestione, ma forse dipende dal mio PC un po’ vecchiotto o dal collegamento Internet.
Per inviare i dati al cellulare, ho creato una stringa unica di 30 caratteri con una “X” finale come riferimento. I dati inviati sono:
direzione, elevazione, skew, livello segnale, posizione potenziometro rotazione e stato bistabile manuale/automatico, per avere la certezza dello stato del controllo.
Occorrerà poi splittare la stringa in ricezione per ricavare tutte le variabili. Io ho provato anche questo metodo ma il primo sistema secondo me è più veloce e introduce meno latenza nell’esecuzione dei comandi.
Il tempo di mettere insieme tutto il materiale e vedro’ se qualcuno avrà voglia di pubblicarlo.
Allego il link dell’ultima interfaccia in quanto sono riuscito, dopo aver corretto anche una formula per il calcolo dell’orientamento parabola trovata su Wikipedia ad ottenere i dati di puntamento usando latitudine/longitudine del GPS del cellulare o anche solo dalla triangolazione delle cellule telefoniche.
Vorrei ancora riuscire a disattivare lo standby del display, tipo quando si usa il navigatore che il display rimane sempre acceso, ma questa la vedo più dura in quanto non ho trovato istruzioni in merito.
ciao chimera, può darsi che i primi brandeggi usassero motori a 24v, i miei invece, sono a 12v. Sui motori dei tergicristalli, a parte il fatto che bisogna riuscire a farceli stare, avevo sentito altre discussioni dove dicevano che i campi magnetici creati, rendevano impossibile l'utilizzo della bussola. Probabilmente o non sono sufficiente schermati contro i disturbi o assorbono troppo, creando maggior flusso magnetico. Penso che la prima idea che avevi avuto (eBay), non sia da scartare.
grazie per la tua risposta , si mi riferivo proprio a quello .
Grazie per la dritta segnali analogici , e da un po chesto giocando pure io con app invetor.
Windom guardando il tuo schema controllo quanto fili arrivano al brandeggio? 10 + Antenna?
4x motordriver
12v ,5v,GND,Poti,SDA,SCL(CMPS10)
Grazie
Antonio
[quote=windom46]
Ciao tedesto, se ti riferisci al controllo via bluetooth con cellulare Android del video, l’app l’ho fatta con MitAppInventor(beta). Con la versione 2, trovo che il BlocksEditor non vada avanti e poi sia di difficile gestione, ma forse dipende dal mio PC un po’ vecchiotto o dal collegamento Internet.
Per inviare i dati al cellulare, ho creato una stringa unica di 30 caratteri con una “X” finale come riferimento. I dati inviati sono:
direzione, elevazione, skew, livello segnale, posizione potenziometro rotazione e stato bistabile manuale/automatico, per avere la certezza dello stato del controllo.
Occorrerà poi splittare la stringa in ricezione per ricavare tutte le variabili. Io ho provato anche questo metodo ma il primo sistema secondo me è più veloce e introduce meno latenza nell’esecuzione dei comandi.
Il tempo di mettere insieme tutto il materiale e vedro’ se qualcuno avrà voglia di pubblicarlo.
Allego il link dell’ultima interfaccia in quanto sono riuscito, dopo aver corretto anche una formula per il calcolo dell’orientamento parabola trovata su Wikipedia ad ottenere i dati di puntamento usando latitudine/longitudine del GPS del cellulare o anche solo dalla triangolazione delle cellule telefoniche.
Vorrei ancora riuscire a disattivare lo standby del display, tipo quando si usa il navigatore che il display rimane sempre acceso, ma questa la vedo più dura in quanto non ho trovato istruzioni in merito.
Si, effettivamente più o meno è così, ma faccio prima a farti lo schema del cablaggio che vale più di 1000 spiegazioni. Il cavo dell'antenna che va al satfinder e poi al decoder è una cosa a parte.
Come alimentazione, al brandeggio, arriva solo, oltre alla massa, il +5 volt, i 12 volt arrivano ai motori dal driver. I 3 fili (SDA,SCL e potenziom.) vengono collegati nella scatola stagna a quelli che arrivano dal controllo, praticamente, sono solo di passaggio.
nel brandeggio hai lasciato i motori originali? l´elettronica con i relais interni non servono piu´ o si devono lasciare? i motori si collegano direttamente con il motordriver(emertee) ?
Io il mio l'ho collegato così ( il cavo di destra va al brandeggio). Come vedi oltre ai 4 fili dei motori e ai 3 passanti (bianco,grigio,marron) ci sono solo il nero di massa e il rosso grande dei 5 volt. Forse hanno cambiato qualcosa. Comunque se i motori girano fai solo attenzione che si fermino contro i fine corsa altrimenti bisogna invertire i fili, se mi ricordo bene, rosso e nero sui motoriduttori.
Ciao tedesco, nel dubbio sono andato a cercarmi le intruzioni dei collegamenti del mio brandeggio ed effettivamente, come puoi vedere dall'immagine scannerizzata, hanno cambiato il comune dal negativo al positivo, cosa che sinceramente non capisco in quanto normalmente il comune è la massa.
A questo punto, bisognerà fare arrivare anche il +12volt al brandeggio e modificare il software in modo che per attuare il comando non arrivi il positivo ma il negativo.
... e così anche per tutte le altre istruzioni che controllano i motori (solo quelle).
Certo non ci voleva, mi spiace solo di non avere un altro brandeggio come il tuo da fare le prove in quanto le polarità vanno rispettate per via dei condensatori elettrolitici che ricordo esserci sulle basette dei ralais, che scoppierebbero.
Adattare il software alle tue esigenze non comporta comunque rischi, al massimo, i motori non girano.
Eventualmente prova a cambiarne una funzione alla volta, iniziando da quella che ferma i motori.
Ho altri problemini dove sicuramente mi puoi aiutare , dove nel tuo sketch si trovano i dati del satellite Hotbird? con due pusanti vorrei implementare anche Astra 19,2 Pin Digitali 2 e 4?
Ancora non ho risolto con il Poti .....
come hai fatto con il cmps10 dopo l´hai montato ? potresi postare qualche foto? si deve calibrare come documentazione?
Calibration the CMPS10 I would recommend evaluating the CMPS10 performance first before implementing this function. Its purpose is to remove offsets caused by constant magnetic sources around the CMPS10. First of all you need to determine North and align the CMPS10 with it, then write a sequence of 3 commands in the correct order with a small delay between bytes, 100ms will be more than adequate. The sequence to enter calibration mode is 0x31,0x45,0x5A, then calibrate the first point by sending 0x5E to the command register, this should also light the LED. The Compass should then be rotated 90° and 0x5E sent to the command register again, repeat for two further 90° rotations and the calibration completes and the LED turns off. Please make sure that the CMPS10 is not located nearto ferrous objects as this will distort the magnetic field and induce errors in the reading.
partiamo dai dati di puntamento che sono memorizzati nella eeprom(cella 0 e 1). La prima volta bisogna cercare il satellite con la ricerca manuale sui valori ricavati da tabelle che si trovano in Internet o da programmi tipo Satfinder (Android), valori che dipendono dalla tua posizione che in genere sono intorno a 38 e 173 gradi. Successivamente, memorizzare questi valori con "Salva" per poi ritrovarseli in fase di ricerca automatica. Per quanto riguarda Astra, io non l'ho previsto in quanto uso il decoder TivuSat con scheda che usa esclusivamente Hotbird, ho però notato che senza cambiare elevazione, basta ruotare l'antenna di circa 5 gradi e si prende, anche più forte di Hotbird e puoi sempre memorizzare questi valori per la ricerca successiva.
La cmps10 non l'ho dovuta calibrare in quanto andava benissimo, mentre per la cmps03 che usavo prima e che si era scalibrata, ho usato questo sketch in modalità I2C che penso dovrebbe andare bene anche per la cmps10:
// CALIBRAZIONE CMPS03
#include <Wire.h> #define sensorAddress 0x60 // indirizzo bussola: #define resultRegister 0x02 // indirizzo di memoria per il risultato #define calibrationRegister 22 #define calibrationValue 0xF5 int delayTime = 20000; void setup() { // avvia il bus I2C: Wire.begin(); // avvia Seriale: Serial.begin(9600); }
void loop() { setRegisterRX(sensorAddress, resultRegister); int bearing = readData(sensorAddress, 2); // legge il risultato: Serial.print("posizione: "); // stampa il risultato: Serial.print(bearing/10); Serial.println(" gradi"); delay(10000); // attesa 10 secondi per prossima lettura
// calibrazione: Serial.println("puntare a NORD"); delay (delayTime); setRegisterTX(sensorAddress, calibrationRegister, calibrationValue); Serial.println("NORD scritto"); Serial.println("puntare a EST"); delay (delayTime); setRegisterTX(sensorAddress, calibrationRegister, calibrationValue); Serial.println("EST scritto"); Serial.println("puntare a SUD"); delay (delayTime); setRegisterTX(sensorAddress, calibrationRegister, calibrationValue); Serial.println("SUD scritto"); Serial.println("puntare a OVEST"); delay (delayTime); setRegisterTX(sensorAddress, calibrationRegister, calibrationValue); Serial.println("CALIBRAZIONE FINITA"); while(true) {}; } void setRegisterRX(int address, int thisRegister) {// start I2C transmission: Wire.beginTransmission(address); // write address to read from: Wire.write(thisRegister); // end I2C transmission: Wire.endTransmission(); }
void setRegisterTX(int address, int thisRegister, int thisValue) { // start I2C transmission: Wire.beginTransmission(address); // write address to write to: Wire.write(thisRegister); // write value to write to: Wire.write(thisValue); // end I2C transmission: Wire.endTransmission(); } int readData(int address, int numBytes) { int result = 0; // the result is two bytes long Wire.requestFrom(address, numBytes); // write I2C request for data: while (Wire.available() < 2 ) { // wait for two bytes to return: } // wait for result result = Wire.read() * 256; // read the two bytes, and combine them into one int: result = result + Wire.read(); return result; // return the result: }
Per quanto riguarda il montaggio della bussola, ho solo una foto a montaggio ultimato dell'antenna. Essendo ora integrati bussola e inclinometro, io l'ho sistemata nello scatolotto al posto dell' inclinometro, cercando di tenerla in piano in corrispondenza dei 38 gradi di elevazione e con il nord in avanti (usare solo viti in ottone o acciaio inox per via dei campi magnetici).
Commenti
sempre a far casino sta redazione
CHE GUEVARA
Grazie dell'interessamento Windom, e scusa se lo faccio solo oggi ma sono stato in giro qualche giorno.
Aspetterò con pazienza.
Ciao.
SSaluti a tutti, mi chiamo Elia. Mi diletto anch'io di elettronica ed affini. Pensando di fare cosa gradita ho pensato di fare uno schemino dei collegamenti sul progetto di Windom che allego. In attesa dei nuovi suggerimenti proposti da Windom, ed ulteriori sviluppi, attendo scalpitando. segnalate eventuali errori. Saluti.
Nell' attesa cher venga pubblicato l'articolo, anticipo lo schema di come ho collegato il tutto al fine di controllare il puntamento via bus I2C.
I motori funzionano a 12 volt e non possono essere controllati direttamente da Arduino che verrebbe bruciato immediatamente,occorre interporre un driver per motori C.C. come indicato nello schema.
Siccome serve a poco lo schema senza il software,se qualcuno mi spiega come fare, pubblico anche questo. Resta il problema che il tutto è fatto in funzione della meccanica di rotazione modificata con l'aggiunta di un potenziometro per ottenera la posizione della meccanica stessa, come verrà spiegato dettagliatamente alla pubblicazione dell' articolo.
http://i60.tinypic.com/14l0w1k.jpg
Grazie windom46 per lo schema. Adesso comincio a smanettare, qualcosa verrà fuori. Se non ti crea molto disturbo sarebbe molto gradito il software. Puoi allegarlo utilizzando il pulsante a sinistra "incolla da word". chiaramente deve essere in formato word. Grazie anticipatamente, saluti e a buon rendere.
Come suggeritomi da chimera, allego il software per Arduino (IDE1.05) che sto utilizzando attualmente in giro per la Sardegna da oltre un mese senza problemi.
Mentre ci sono, anticipo anche il link del video pubblicato su Youtube sulle ultime modifiche apportate e sul nuovo sistema di controllo via bluetooth con il Galaxy S3.
software per Arduino:
/* CONTROLLO BRANDEGGIO (Motore303) per Antenna SELFSAT, con ARDUINO-UNO, bussola
* CMPS10 e Display 16x2 in modalità I2C (SatWindom_I2C_rev.2) *
*********************************************************************************/
#include <Wire.h> // include librerie
#include <LiquidCrystal_I2C.h>
#include <EEPROM.h>
#include <avr/wdt.h>
#define Reset_AVR() wdt_enable(WDTO_30MS) // softReset sistema
#define ADDRESS 0x60 // definisce indirizzo CMPS10
#define inSF A1 // pin in SatFinder
#define inCom A2 // pin in Pulsanti
#define inPot A3 // pin in Potenziometro
#define ledMA 12 // pin led man/auto
#define inMot 13 // pin motore acceso
#define motDa 9 // pin motore direzione(1)
#define motDb 10 // pin motore direzione(2)
#define motEa 3 // pin motore elevazione(1)
#define motEb 11 // pin motore elevazione(2)
LiquidCrystal_I2C lcd(0x20,16,2); // setta LCD (address, colonne, linee)
int levelSF; // livello segnale del SatFinder
char level = -1; // barra segnale SatFinder
byte oldState = LOW; // imposta stato bistabile
int dirOut; // lettura bussola in gradi
int dirHB = EEPROM.read(0); // legge direziona dalla cella 0
int elevHB = EEPROM.read(1); // legge elevazione dalla cella 1
byte pass = 0; // num.loop per limitare scrittura display
const byte sicRot = 60; // sicurezza rotazione
const byte centro = 183; // rotazione a 180°
int posPot; // posizione rotazione antenna
const byte range = 1; // tolleranza su angolazioni
byte dirLock = 0; // direzione OK
byte elevLock = 0; // elevazione OK
const byte velRid = 100; // riduzione velocità a 100/255
const byte prox = 5; // gradi mancanti alla posizione
byte centroRot = 0; // centro rotazione meccanica
int manButton; // ingresso comandi manuali
char elevOut, skew; // valore elevazione e skew(da CMPS10)
void setup(){
lcd.init(); // inizializza lcd
pinMode(inSF, INPUT); // ingresso livello SatFinder
pinMode(inPot, INPUT); // cursore potenziometro rotazione
pinMode(inCom, INPUT); // ingresso pulsanti comandi manuali
pinMode(ledMA, OUTPUT); // LED manuale/auto
pinMode(inMot, INPUT); // sensore motore acceso
pinMode(motDa, OUTPUT); // motore orientamento 9/10
pinMode(motDb, OUTPUT);
pinMode(motEa, OUTPUT); // motore elevazione 3/11
pinMode(motEb, OUTPUT);
// pin digitali 0-1-2-4-5-6-7-8 e analogico A0, liberi.
ferma(); // ferma i motori
if (inMot == LOW){lcd.backlight();} // accende display
// visualizza su display i valori memorizzati nella EEPROM
lcd.setCursor(0,0);
lcd.print("**** HOTBIRD ***");
lcd.setCursor(0,1);
lcd.print("Dir.");
lcd.print(dirHB, DEC);
lcd.setCursor(9,1);
lcd.print("Elev.");
lcd.print(elevHB, DEC);
lcd.print(" ");
lcd.print(" ");
delay(3000); // attesa 3 secondi
lcd.clear();
}
void loop() {
byte highByte, lowByte; // valori di highByte e lowByte per dirOut
Wire.beginTransmission(ADDRESS); // inizio comunicazione con CMPS10
Wire.write(2); // invia registro per la lettura
Wire.endTransmission();
Wire.requestFrom(ADDRESS, 4); // richiesta di 4 bytes da CMPS10
while(Wire.available() < 4); // attesa disponibilità bites
highByte = Wire.read(); // lettura azimut(direzione)
lowByte = Wire.read(); // lettura decimali azimut(direzione)
elevOut = Wire.read()+57; // lettura elevazione+compensazione
skew = Wire.read(); // lettura angolazione
dirOut = ((highByte<<8)+lowByte)/10; // calcolo direzione
manButton = analogRead(inCom); // valori comandi manuali immessi su(A2)
posPot = map(analogRead(inPot),102,924,0,359); // posizione rotazione rimappata
levelSF = analogRead(inSF); // livello segnale SatFinder
// Partitore resistivo per inviare 6 valori (100-200-300-400-500-600)
// sul pin(A2) e ottenere un convertitore A/D per le varie funzioni.
// 7,8K verso 5v.+ 6 da 2,2k per il partitore + 1 da 100k tra A2 e massa
// ********* BISTABILE COMMUTAZIONE MAN./AUTO ******************
if (manButton > 50 && manButton < 150) { // manButton=100 (Manuale/Auto)
if (oldState == LOW) {
oldState = HIGH; // attiva automatismi e
digitalWrite(ledMA, HIGH); // accende LED
elevLock = 0; // imposta elevLock e dirLock a 0
dirLock = 0;
}else{
oldState = LOW; // attiva comandi manuali e
digitalWrite(ledMA, LOW); // spegne LED
}
delay(500); // attende 1/2 secondo
}
if (oldState == LOW) { // comandi manuali attivato
if (manButton < 50) ferma(); // arresta motori
if (manButton > 150 && manButton < 250) salva(); // 200;salva valori Elev.Dir. nella EPROM
if (manButton > 250 && manButton < 350) alza(); // 300;alza antenna
if (manButton > 350 && manButton < 450) abbassa(); // 400;abbassa antenna
if (manButton > 450 && manButton < 550 && elevOut < sicRot) orario(); // 500;hiude antenna
if (manButton > 550 && manButton < 650 && elevOut < sicRot) antiorario();// 600;apre antenna
}else{orienta();}
if (digitalRead(inMot) == HIGH) chiude(); // chiude ant.a motore acceso
// ***** Scrive su LCD 16x2 con livello segnale *****
pass = pass+1; // limitazione scrittura display
if (pass == 1) {
if (elevOut > 85) { // controllo livello camper
lcd.setCursor(0,0);
lcd.print("LIVELLO___CAMPER");
lcd.setCursor(0,1);
if (((88-elevOut)==0) && ((skew-3)== 0)) { // usa gli assi pitch,roll(da compensare)
lcd.print("*** IN PIANO ***"); // per controllare il livello del camper
}else{
lcd.print("A/P ");
lcd.print(88-elevOut, DEC);
lcd.print(" ");
lcd.setCursor(9,1);
lcd.print("Lat.");
lcd.print(skew-3, DEC);
lcd.print(" ");
}
}else{ // controllo posizionamento antenna
lcd.backlight();
lcd.setCursor(0,0);
lcd.print("Elev.");
lcd.print(elevOut, DEC);
lcd.print(" ");
lcd.setCursor(9,0);
lcd.print("Dir.");
lcd.print(dirOut, DEC);
lcd.print(" ");
lcd.setCursor(0,1);
if (levelSF > 30) { // formazione barra livello segnale
lcd.print("Level.");
for (int lev=0; lev<=levelSF/8; lev++) {
lcd.print(level);
}
lcd.print(" ");
}else{
lcd.print("Segn.");
lcd.print(levelSF);
lcd.print(" ");
lcd.setCursor(9,1);
lcd.print("Pos.");
lcd.print(posPot, DEC);
lcd.print(" ");
}
}
}
if (pass > 50) {pass = 0;} // scrive su display ogni 50 loop
delay(10);
}
/*********** FUNZIONI RIPETITIVE *****************/
void alza() {
digitalWrite(motEa, HIGH);
if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut < 30){
analogWrite(motEb, velRid);
}else{
digitalWrite(motEb, LOW);
}
}
/***************************************************/
void abbassa() {
digitalWrite(motEb, HIGH);
if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut > 80){
analogWrite(motEa, velRid);
}else{
digitalWrite(motEa, LOW);
}
}
/***************************************************/
void orario() { // gira in senso orario
digitalWrite(motDb, HIGH);
if (((dirOut > dirHB-prox) && (dirOut < dirHB+prox)) || posPot > 345){
analogWrite(motDa, velRid);
}else{
digitalWrite(motDa, LOW);
}
}
/***************************************************/
void antiorario() { // gira in senso antiorario
digitalWrite(motDa, HIGH);
if (((dirOut > dirHB-prox) && (dirOut < dirHB+prox)) || posPot < 15){
analogWrite(motDb, velRid);
}else{
digitalWrite(motDb, LOW);
}
}
/***************************************************/
void ferma() { // ferma rotazione motori
digitalWrite(motDa, HIGH);
digitalWrite(motDb, HIGH);
digitalWrite(motEb, HIGH);
digitalWrite(motEa, HIGH);
oldState = LOW; // disattiva automatismi e
digitalWrite(ledMA, LOW); // spegne LED
}
/***************************************************/
void salva() {
EEPROM.write(0, dirOut); // memorizza direzione nella cella 0
EEPROM.write(1, elevOut); // memorizza elevazione nella cella 1
delay(500);
Reset_AVR(); // resetta sistema
}
/***************************************************/
void chiude() { // chiude antenna (MOTORE acceso)
oldState = LOW; // disattiva automatismi e
digitalWrite(ledMA, LOW); // spegne LED
if (centroRot == 0){
if (elevOut > sicRot) {alza(); // porta elevazione in sicurezza
}else{
if (posPot > centro + range) antiorario();
if (posPot < centro - range) orario(); // porta rotazione a 180°
}
}
if (posPot == centro) {
centroRot = 1;
digitalWrite(motDa, HIGH); // ferma rotazione antenna
digitalWrite(motDb, HIGH);
}
if (centroRot == 1) abbassa();
}
/***************************************************/
void orienta() { // mantiene l'antenna entro i valore impostati
if (elevOut > elevHB + range) {alza(); // alza antenna
}else{ // orienta antenna se elevOut < elevHB
if (dirOut > dirHB + range) antiorario();
if (dirOut < dirHB - range) orario();
if ((dirOut > dirHB - range) && (dirOut < dirHB + range)) {
digitalWrite(motDa, HIGH); // ferma rotazione antenna
digitalWrite(motDb, HIGH);
dirLock = 1; // direzione OK
}
if (elevOut > elevHB + range) alza(); // alza antenna
if (elevOut < elevHB - range) abbassa(); // abbassa antenna
if ((elevOut < elevHB + range) && elevOut > (elevHB - range)) {
digitalWrite(motEa, HIGH); // ferma elevazione antenna
digitalWrite(motEb, HIGH);
elevLock = 1; // elevazione OK
}
}
if (((elevLock == 1) && (dirLock == 1)) || (levelSF > 80)) centra();
}
/*** Centraggio perfetto satellite. Programma ancora da sviluppare, in attesa di perfezionare
il movimento della meccanica. Sostituendo almeno il motore della rotazione con uno Stepper,
si avrebbe la massima precisione e la possibilità di bloccare la meccanica, con un minimo di
tensione PWM su una fase. Resta da verificare l'influenza dei maggiori campi magnetici creati
sul funzionamento della bussola. Per il momento la ricerca si ferma se durante la scansione,
in prossimità delle posizioni salvate, il livello segnale supera il 30% ********************/
void centra(){
if (levelSF > 30) { // se il segnale supera il 30%, si ferma.
ferma();
oldState = LOW; // attiva comandi manuali e
digitalWrite(ledMA, LOW); // spegne LED
}
}
/******************************* FINE PROGRAMMA ********************************************/
grazie, grazie, grazie *clapping* *biggrin*
Grazie Windom .... Domani comincerò a lavorarci.
Ciao.
Mi sono accorto di un erore del software rispetto allo schema elettrico, dovuto al fatto che a me non funzionava più il pin analogico A0 e quindi avevo usato i pin A1,A2 e A3.
Si prega pertanto di correggere le linee:
#define inSF A1 // pin in SatFinder
#define inCom A2 // pin in Pulsanti
#define inPot A3 // pin in Potenziometro
nel seguente modo:
#define inSF A0 // pin in SatFinder
#define inPot A1 // pin in Potenziometro
#define inCom A2 // pin in Pulsanti
Va detto che si possono usare i pin analogici che si desidera, basta che ci sia corrispondenza tra schema elettrico e software.
Ciao.
aiuto! sono in possesso di un brandeggio come quello di windom46 sprovvisto di motorini qualcuno mi può dare una dritta a quale modello mi devo orientare e possibilmente dove poterli reperire. Chiaramente devo pilotarli con arduino utilizzando il progetto di windom46. Grazie
ciao chimera, i motoriduttori originali, fanno circa 4,5 giri al minuto.
Io quando avevo pensato di sostituire i miei, avevo visto questo:
http://www.futuraelettronica.net/CATGEN2013_SEZIONI/Robotica_nuove_tecnologie.pdf
pag.125 art. RH15912630. Se ci stanno come ingombro, oltre ad essere economici, dovrebbere essere perfetti.
Ciao, a presto.
Dovessi cambiare i motori adesso, però, sull' elevazione, essendo quello originale un po' lento, metterei il RH15812630, sempre a pag.125, che fa 9 RPM.
ciao, windom46 cosa ne pensi di questo?
http://www.ebay.com/itm/12V-DC-5RPM-High-Torque-Gear-Box-Electric-Motor-/221036286413?pt=LH_DefaultDomain_0&hash=item3376ca11cd
costa molto meno.
Questo è l'interno dei motoriduttori originali, ingranaggio che sforza meno in nylon e gli altri metallici, niente di eccezionale. Quelli che dici tu, almeno dalla struttura esterna, sono senz'altro meglio, vedo pure che ci sono con diverso rapporto di riduzione. Non resta che provare (altezza 63 mm....ci stanno?). Come fai ad accoppiarli alla meccanica senza ingranaggi? Fai tutto con pulegge e cinghie dentate?
Io, come vedi all'inizio del video, l'ho fatto sulla rotazione, riducendo a zero i giochi di accoppiamento.
La cosa comporta diverse modifiche alla meccanica, che saranno spiegate dettagliatamente non appena in redazione troveranno il tempo di pubblicare l'articolo.
A preso, ciao.
<img alt="" src="
Non mi ha caricato l'immagine, ecco il link.
http://i57.tinypic.com/p1teq.jpg
i vecchi motorini funzionavano a 24v ed erano accoppiati tramite un ingranaggio in metallo che sono riuscito a conservare. Ho notato che i vecchi motori hanno lo stesso diametro dell'asse di rotazione, quindi posso utilizzare l'ingranaggio che viene blocccato con una coppiglia che funge da fermo. Comunque sto valutando l'idea di poter utilizzare qualche motore di alzacristallo di qualche auto, ma solo se sarà possibile utilizzare l'ingranaggio originale. Domani proverò a fare una foto all'interno del brandeggio.
i vecchi motorini funzionavano a 24v ed erano accoppiati tramite un ingranaggio in metallo che sono riuscito a conservare. Ho notato che i vecchi motori hanno lo stesso diametro dell'asse di rotazione, quindi posso utilizzare l'ingranaggio che viene blocccato con una coppiglia che funge da fermo. Comunque sto valutando l'idea di poter utilizzare qualche motore di alzacristallo di qualche auto, ma solo se sarà possibile utilizzare l'ingranaggio originale. Domani proverò a fare una foto all'interno del brandeggio.
Super Windom ,
dimmi per l´ app Android usi App inventor? come fai a visualizzare piu di un dato analogico ?
i dati li mandi in seriale -->serial.print? nello stesso metodo come li mandi sul lcd i2c?
// visualizza su display i valori memorizzati nella EEPROM
lcd.setCursor(0,0);
lcd.print("**** HOTBIRD ***");
serial.print ("**** HOTBIRD ***");
lcd.setCursor(0,1);
lcd.print("Dir.");
lcd.print(dirHB, DEC);
lcd.setCursor(9,1);
lcd.print("Elev.");
lcd.print(elevHB, DEC);
lcd.print(" ");
lcd.print(" ");
delay(3000); // attesa 3 secondi
lcd.clear();
}
Ciao tedesto, se ti riferisci al controllo via bluetooth con cellulare Android del video, l’app l’ho fatta con MitAppInventor(beta). Con la versione 2, trovo che il BlocksEditor non vada avanti e poi sia di difficile gestione, ma forse dipende dal mio PC un po’ vecchiotto o dal collegamento Internet.
Per inviare i dati al cellulare, ho creato una stringa unica di 30 caratteri con una “X” finale come riferimento. I dati inviati sono:
direzione, elevazione, skew, livello segnale, posizione potenziometro rotazione e stato bistabile manuale/automatico, per avere la certezza dello stato del controllo.
String stringBT, strDir, strElev, strSkew, strLev, strPot, strOS; // variabili trasmissione dati
SoftwareSerial mySerial(pinRX, pinTX); // impostare i pin 2=RX 4=TX
/**************************** SETUP *********************************/
void setup(){
mySerial.begin(38400); // settaggio comunicazione BT, IMPORTANTE! settare
// anche il modulo BT a 38400 con istruzioni AT
ecc.ecc…….
// stringa trasmissione dati
strDir = String(dirOut) + " "; strDir = strDir.substring(0,5); // 5 spazi tra virgolette
strElev = String(int(elevOut)) + " "; strElev = strElev.substring(0,5);
strSkew = String(int(skewOut)) + " "; strSkew = strSkew.substring(0,5);
strLev = String(int(levelSF)) + " "; strLev = strLev.substring(0,5);
strPot = String(int(posPot)) + " "; strPot = strPot.substring(0,5);
strOS = String(int(oldState)) + " "; strOS = strOS.substring(0,4)+"X";
stringBT = strDir + strElev + strSkew + strLev + strPot + strOS ; // lunghezza stringa 30 bytes
mySerial.print(stringBT); // invia dati alla Seriale BT
Nella app.apk le variabili vengono ricavate a blocchi di 5 caratteri, usando come riferimento la X.
Va detto che il programma sarebbe più pulito trasmettendo la stringa di variabili separate da “;” e “:” ottenendo l’accoppiata variabile/valore, così:
stringBT = dir:strDir;elev:strElev;skew:strSkew ;level:strLev; ecc……. // lunghezza stringa 30 bytes
Occorrerà poi splittare la stringa in ricezione per ricavare tutte le variabili. Io ho provato anche questo metodo ma il primo sistema secondo me è più veloce e introduce meno latenza nell’esecuzione dei comandi.
Il tempo di mettere insieme tutto il materiale e vedro’ se qualcuno avrà voglia di pubblicarlo.
Allego il link dell’ultima interfaccia in quanto sono riuscito, dopo aver corretto anche una formula per il calcolo dell’orientamento parabola trovata su Wikipedia ad ottenere i dati di puntamento usando latitudine/longitudine del GPS del cellulare o anche solo dalla triangolazione delle cellule telefoniche.
http://i59.tinypic.com/14udziu.png
Vorrei ancora riuscire a disattivare lo standby del display, tipo quando si usa il navigatore che il display rimane sempre acceso, ma questa la vedo più dura in quanto non ho trovato istruzioni in merito.
ciao chimera, può darsi che i primi brandeggi usassero motori a 24v, i miei invece, sono a 12v. Sui motori dei tergicristalli, a parte il fatto che bisogna riuscire a farceli stare, avevo sentito altre discussioni dove dicevano che i campi magnetici creati, rendevano impossibile l'utilizzo della bussola. Probabilmente o non sono sufficiente schermati contro i disturbi o assorbono troppo, creando maggior flusso magnetico. Penso che la prima idea che avevi avuto (eBay), non sia da scartare.
Ciao windom46,
grazie per la tua risposta , si mi riferivo proprio a quello .
Grazie per la dritta segnali analogici , e da un po chesto giocando pure io con app invetor.
Windom guardando il tuo schema controllo quanto fili arrivano al brandeggio? 10 + Antenna?
4x motordriver
12v ,5v,GND,Poti,SDA,SCL(CMPS10)
Grazie
Antonio
[quote=windom46]
Ciao tedesto, se ti riferisci al controllo via bluetooth con cellulare Android del video, l’app l’ho fatta con MitAppInventor(beta). Con la versione 2, trovo che il BlocksEditor non vada avanti e poi sia di difficile gestione, ma forse dipende dal mio PC un po’ vecchiotto o dal collegamento Internet.
Per inviare i dati al cellulare, ho creato una stringa unica di 30 caratteri con una “X” finale come riferimento. I dati inviati sono:
direzione, elevazione, skew, livello segnale, posizione potenziometro rotazione e stato bistabile manuale/automatico, per avere la certezza dello stato del controllo.
String stringBT, strDir, strElev, strSkew, strLev, strPot, strOS; // variabili trasmissione dati
SoftwareSerial mySerial(pinRX, pinTX); // impostare i pin 2=RX 4=TX
/**************************** SETUP *********************************/
void setup(){
mySerial.begin(38400); // settaggio comunicazione BT, IMPORTANTE! settare
// anche il modulo BT a 38400 con istruzioni AT
ecc.ecc…….
// stringa trasmissione dati
strDir = String(dirOut) + " "; strDir = strDir.substring(0,5); // 5 spazi tra virgolette
strElev = String(int(elevOut)) + " "; strElev = strElev.substring(0,5);
strSkew = String(int(skewOut)) + " "; strSkew = strSkew.substring(0,5);
strLev = String(int(levelSF)) + " "; strLev = strLev.substring(0,5);
strPot = String(int(posPot)) + " "; strPot = strPot.substring(0,5);
strOS = String(int(oldState)) + " "; strOS = strOS.substring(0,4)+"X";
stringBT = strDir + strElev + strSkew + strLev + strPot + strOS ; // lunghezza stringa 30 bytes
mySerial.print(stringBT); // invia dati alla Seriale BT
Nella app.apk le variabili vengono ricavate a blocchi di 5 caratteri, usando come riferimento la X.
Va detto che il programma sarebbe più pulito trasmettendo la stringa di variabili separate da “;” e “:” ottenendo l’accoppiata variabile/valore, così:
stringBT = dir:strDir;elev:strElev;skew:strSkew ;level:strLev; ecc……. // lunghezza stringa 30 bytes
Occorrerà poi splittare la stringa in ricezione per ricavare tutte le variabili. Io ho provato anche questo metodo ma il primo sistema secondo me è più veloce e introduce meno latenza nell’esecuzione dei comandi.
Il tempo di mettere insieme tutto il materiale e vedro’ se qualcuno avrà voglia di pubblicarlo.
Allego il link dell’ultima interfaccia in quanto sono riuscito, dopo aver corretto anche una formula per il calcolo dell’orientamento parabola trovata su Wikipedia ad ottenere i dati di puntamento usando latitudine/longitudine del GPS del cellulare o anche solo dalla triangolazione delle cellule telefoniche.
http://i59.tinypic.com/14udziu.png
Vorrei ancora riuscire a disattivare lo standby del display, tipo quando si usa il navigatore che il display rimane sempre acceso, ma questa la vedo più dura in quanto non ho trovato istruzioni in merito.
[/quote]
Si, effettivamente più o meno è così, ma faccio prima a farti lo schema del cablaggio che vale più di 1000 spiegazioni. Il cavo dell'antenna che va al satfinder e poi al decoder è una cosa a parte.
Come alimentazione, al brandeggio, arriva solo, oltre alla massa, il +5 volt, i 12 volt arrivano ai motori dal driver. I 3 fili (SDA,SCL e potenziom.) vengono collegati nella scatola stagna a quelli che arrivano dal controllo, praticamente, sono solo di passaggio.
Grazie Windom
Ciao Windom46,
nel brandeggio hai lasciato i motori originali? l´elettronica con i relais interni non servono piu´ o si devono lasciare? i motori si collegano direttamente con il motordriver(emertee) ?
grazie per la tua pazienza
Antonio
Nel brandeggio ho lasciato tutto com'era, collegandomi esclusivamente ai fili in uscita, 4 per i movimenti che vanno al driver + massa.
il mio brandeggio viene collegato in comune con il +
[url=http://www.fotos-hochladen.net][img]http://img5.fotos-hochladen.net/uploads/20140619213752uzamhotrs1.jpg[/img][/url]
io l´ collegato cosi´
----------------sopra - ------------------> Mot 1
---------------sotto - ------------------->Mot 1
---------------destra - -------------------->Mot2
---------------sinistra - -------------------->Mot2
--------------- + -------------------->COM +
[url=http://www.fotos-hochladen.net][img]http://img5.fotos-hochladen.net/uploads/20140619221251l280kw7d56.jpg[/img][/url]
sul motodriver ho collegato Power con 12V
Io il mio l'ho collegato così ( il cavo di destra va al brandeggio). Come vedi oltre ai 4 fili dei motori e ai 3 passanti (bianco,grigio,marron) ci sono solo il nero di massa e il rosso grande dei 5 volt. Forse hanno cambiato qualcosa. Comunque se i motori girano fai solo attenzione che si fermino contro i fine corsa altrimenti bisogna invertire i fili, se mi ricordo bene, rosso e nero sui motoriduttori.
Ciao tedesco, nel dubbio sono andato a cercarmi le intruzioni dei collegamenti del mio brandeggio ed effettivamente, come puoi vedere dall'immagine scannerizzata, hanno cambiato il comune dal negativo al positivo, cosa che sinceramente non capisco in quanto normalmente il comune è la massa.
A questo punto, bisognerà fare arrivare anche il +12volt al brandeggio e modificare il software in modo che per attuare il comando non arrivi il positivo ma il negativo.
Esempio:
void alza() {
digitalWrite(motEa, HIGH);
if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut < 30){
analogWrite(motEb, velRid);
}else{
digitalWrite(motEb, LOW);
}
}
cambiare così:
void alza() {
digitalWrite(motEa, LOW);
if (((elevOut > elevHB-prox) && (elevOut < elevHB+prox)) || elevOut < 30){
analogWrite(motEb, velRid);
}else{
digitalWrite(motEb, HIGH);
}
}
... e così anche per tutte le altre istruzioni che controllano i motori (solo quelle).
Certo non ci voleva, mi spiace solo di non avere un altro brandeggio come il tuo da fare le prove in quanto le polarità vanno rispettate per via dei condensatori elettrolitici che ricordo esserci sulle basette dei ralais, che scoppierebbero.
Adattare il software alle tue esigenze non comporta comunque rischi, al massimo, i motori non girano.
Eventualmente prova a cambiarne una funzione alla volta, iniziando da quella che ferma i motori.
Ciao Windom ,
grazie *pleasantry* ma questo l´avevo gia fatto .
Ho altri problemini dove sicuramente mi puoi aiutare , dove nel tuo sketch si trovano i dati del satellite Hotbird? con due pusanti vorrei implementare anche Astra 19,2 Pin Digitali 2 e 4?
Ancora non ho risolto con il Poti .....
come hai fatto con il cmps10 dopo l´hai montato ? potresi postare qualche foto? si deve calibrare come documentazione?
Calibration the CMPS10
I would recommend evaluating the CMPS10 performance first before implementing this function. Its purpose is to remove offsets
caused by constant magnetic sources around the CMPS10. First of all you need to determine North and align the CMPS10 with it, then
write a sequence of 3 commands in the correct order with a small delay between bytes, 100ms will be more than adequate. The
sequence to enter calibration mode is 0x31,0x45,0x5A, then calibrate the first point by sending 0x5E to the command register, this
should also light the LED. The Compass should then be rotated 90° and 0x5E sent to the command register again, repeat for two
further 90° rotations and the calibration completes and the LED turns off. Please make sure that the CMPS10 is not located nearto
ferrous objects as this will distort the magnetic field and induce errors in the reading.
Ciao tedesco,
partiamo dai dati di puntamento che sono memorizzati nella eeprom(cella 0 e 1). La prima volta bisogna cercare il satellite con la ricerca manuale sui valori ricavati da tabelle che si trovano in Internet o da programmi tipo Satfinder (Android), valori che dipendono dalla tua posizione che in genere sono intorno a 38 e 173 gradi. Successivamente, memorizzare questi valori con "Salva" per poi ritrovarseli in fase di ricerca automatica. Per quanto riguarda Astra, io non l'ho previsto in quanto uso il decoder TivuSat con scheda che usa esclusivamente Hotbird, ho però notato che senza cambiare elevazione, basta ruotare l'antenna di circa 5 gradi e si prende, anche più forte di Hotbird e puoi sempre memorizzare questi valori per la ricerca successiva.
La cmps10 non l'ho dovuta calibrare in quanto andava benissimo, mentre per la cmps03 che usavo prima e che si era scalibrata, ho usato questo sketch in modalità I2C che penso dovrebbe andare bene anche per la cmps10:
// CALIBRAZIONE CMPS03
#include <Wire.h>
#define sensorAddress 0x60 // indirizzo bussola:
#define resultRegister 0x02 // indirizzo di memoria per il risultato
#define calibrationRegister 22
#define calibrationValue 0xF5
int delayTime = 20000;
void setup()
{
// avvia il bus I2C:
Wire.begin();
// avvia Seriale:
Serial.begin(9600);
}
void loop()
{
setRegisterRX(sensorAddress, resultRegister);
int bearing = readData(sensorAddress, 2); // legge il risultato:
Serial.print("posizione: "); // stampa il risultato:
Serial.print(bearing/10);
Serial.println(" gradi");
delay(10000); // attesa 10 secondi per prossima lettura
// calibrazione:
Serial.println("puntare a NORD");
delay (delayTime);
setRegisterTX(sensorAddress, calibrationRegister, calibrationValue);
Serial.println("NORD scritto");
Serial.println("puntare a EST");
delay (delayTime);
setRegisterTX(sensorAddress, calibrationRegister, calibrationValue);
Serial.println("EST scritto");
Serial.println("puntare a SUD");
delay (delayTime);
setRegisterTX(sensorAddress, calibrationRegister, calibrationValue);
Serial.println("SUD scritto");
Serial.println("puntare a OVEST");
delay (delayTime);
setRegisterTX(sensorAddress, calibrationRegister, calibrationValue);
Serial.println("CALIBRAZIONE FINITA");
while(true) {};
}
void setRegisterRX(int address, int thisRegister) {// start I2C transmission:
Wire.beginTransmission(address); // write address to read from:
Wire.write(thisRegister); // end I2C transmission:
Wire.endTransmission();
}
void setRegisterTX(int address, int thisRegister, int thisValue) {
// start I2C transmission:
Wire.beginTransmission(address); // write address to write to:
Wire.write(thisRegister); // write value to write to:
Wire.write(thisValue); // end I2C transmission:
Wire.endTransmission();
}
int readData(int address, int numBytes) {
int result = 0; // the result is two bytes long
Wire.requestFrom(address, numBytes); // write I2C request for data:
while (Wire.available() < 2 ) { // wait for two bytes to return:
} // wait for result
result = Wire.read() * 256; // read the two bytes, and combine them into one int:
result = result + Wire.read();
return result; // return the result:
}
Per quanto riguarda il montaggio della bussola, ho solo una foto a montaggio ultimato dell'antenna. Essendo ora integrati bussola e inclinometro, io l'ho sistemata nello scatolotto al posto dell' inclinometro, cercando di tenerla in piano in corrispondenza dei 38 gradi di elevazione e con il nord in avanti (usare solo viti in ottone o acciaio inox per via dei campi magnetici).