giovedì 6 dicembre 2012

Text to Speech (TTS) con Pico

Una delle funzioni meno utilizzate (ma anche delle piu' interessanti) e' quella di poter utilizzare delle API per poter enunciare una stringa

Il motore di TTS di default su Android e' Pico e non sempre ci sono tutti i linguaggi installati ..quindi se si vuole che il telefono parli in inglese non ci sono problemi ma se si vuole una lingua un po' piu' esotica va scaricata a parte

Gli esempi su come usare Pico sono gia' inclusi negli esempi dei developer di Android quindi non c'e' molto da fare se non usare il sistema


La voce e' piuttosto metallica e robotica rispetto ad altri motori TTS ma c'e' da dire che i file sono estremamente compatti

un primo metodo per testare TTS e' il seguente ma di fatto non e' utile in casi reali di programmazione
---------------------------------------------------------------

package com.example.tts;
import android.os.Bundle;
import android.app.Activity;
import android.speech.tts.TextToSpeech;
import android.speech.tts.TextToSpeech.OnInitListener;


public class MainActivity extends Activity implements OnInitListener{

private TextToSpeech talker;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
talker = new TextToSpeech(this, this);
say("Luca Innocenti");
}

public void say(String text2say){
    talker.speak(text2say, TextToSpeech.QUEUE_FLUSH, null);
   }

public void onInit(int status) {
//say("Luca Innocenti");
}

@Override
public void onDestroy() {
if (talker != null) {
talker.stop();
talker.shutdown();
}
super.onDestroy();
}
}
---------------------------------------------------------------

Il secondo metodo e' decisamente piu' interattivo
---------------------------------------------------------------

package com.example.tts2;

import java.util.Locale;

import android.os.Bundle;
import android.app.Activity;
import android.speech.tts.TextToSpeech;
//import android.speech.tts.TextToSpeech.OnInitListener;
import android.util.Log;
import android.view.Menu;
import android.widget.Button;
import android.widget.EditText;
import android.view.View;

public class MainActivity extends Activity implements TextToSpeech.OnInitListener {

private Button pulsante;
private EditText txtv;
private TextToSpeech tts;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

pulsante = (Button) findViewById(R.id.button1);
txtv = (EditText) findViewById(R.id.editText1);

tts = new TextToSpeech(this, this);

pulsante.setOnClickListener(new View.OnClickListener(){

            @Override
            public void onClick(View arg0) {
                speakOut();
            }

        });
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.activity_main, menu);
return true;
}

@Override
public void onInit(int status) {
// TODO Auto-generated method stub
if (status == TextToSpeech.SUCCESS) {
 
            int result = tts.setLanguage(Locale.US);

            if (result == TextToSpeech.LANG_MISSING_DATA
                    || result == TextToSpeech.LANG_NOT_SUPPORTED) {
                Log.e("TTS", "This Language is not supported");
            } else {
                pulsante.setEnabled(true);
                speakOut();
            }

        } else {
            Log.e("TTS", "Initilization Failed!");
        }

}

private void speakOut() {
// TODO Auto-generated method stub
String text = txtv.getText().toString();
 
        tts.speak(text, TextToSpeech.QUEUE_FLUSH, null);

}

}
--------------------------------------------------------

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >

    <EditText
        android:id="@+id/editText1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_centerVertical="true"
        android:layout_marginBottom="66dp"
        android:ems="10" >

        <requestFocus />
    </EditText>

    <Button
        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignTop="@+id/editText1"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="46dp"
        android:text="Button" />

</RelativeLayout>

Android su HTC TyTN II

Avevo gia' montato Android su un HTC TyTN II oltre tre anni fa ma all'epoca conoscevo molto poco di Android e quindi non avevo apprezzato la cosa anche perche' con un processore Arm da 400 MHz Android funziona piuttosto male,mettiamoci anche uno schermo nato per il pennino e non per il touch per rendere il tutto ingestibile

Il telefono era, quando usci' nel 2007, un ottimo oggetto rilasciato pero' con Windows Mobile 6 (aggiornabile al 6.1) con GPS, Bluetooth, WiFi, SdCard ed altro con uno schermo da 240x320 da 2.8 pollici. Non so quale fosse il prezzo (a me e' arrivato come rotto e sono riuscito a renderlo funzionante) ma doveva essere comparabile all'IPhone di oggi


Per inserire Android sul telefono si possono seguire le istruzioni riportate a questo link
Una volta lanciato il file haret. compare il boot di Android che sostituisce completamente Windows Mobile



Il tempo di boot e' piuttosto lento (oltre 2 minuti) ed al termine si entra in Android 2.1








Interrogato via Adb il sistema operativo funziona completamente (si entra in shell e si puo' fare il pull dei file) , si denota solo molto ritardo sensibile sull'interfaccia touch tale da renderlo sostanzialmente inservibile. La rete Wireless ed il GPS funzionano

Purtroppo non e' disponibile la funzione di screenshot mediante DDMS per cui sono dovute ricorrere ad una fotocamera

mercoledì 5 dicembre 2012

Installazione NDK per Android



L'nstallazione dell' NDK per Android e' quanto mai semplice e' molto piu' lineare rispetto a quella del corrispettivo SDK.

E' sufficiente scarica il file zip della piattaforma per cui e' richiesto lo sviluppo dal sito android ndk e successivamente decomprimerlo in una directory a piacere. Si aggiorna quindi la Path con il comando

export PATH=${PATH}:/home/luca/android_ndk

ed il gioco e' fatto

OCR in Linux con Tesseract

Ho iniziato ad usare dei software OCR nel lontano 1992 su un computer 486 DX/25 con un vetusto software Omniscan su Windows 3.1
Il tempo e' passato ed adesso sono disponibili dei software molto piu' perfomanti e gratuiti; per Linux ho provato ad utilizzare il software Tesseract che attualmente e' sponsorizzato da Google e si puo' scaricare da questo link

Esistono diversi pacchetti gia' compilati per tesseract ma visto che ci siamo sono partito dalla compilazione dei sorgenti

per prima cosa e' necessario eseguire
sudo apt-get install autoconf automake libtool
sudo apt-get install libpng12-dev
sudo apt-get install libjpeg62-dev
sudo apt-get install libtiff4-dev
sudo apt-get install zlib1g-dev


successivamente vanno compilate ed installate le librerie del progetto Leptonica tramite la classica sequenza configure/make/make install

terminata la fase preparatoria si scaricano i file dei sorgenti di Tesseract (i piu recenti sono qui) ed il file per il linguaggio di cui si desidera la conversione (italiano o inglese)

./configure
make
make install
ldconfig


terminata la compilazione si spacchetta il file del linguaggio in una directory (per esempio /home/luca/tessdata) ed al termine nella directory devono essere presenti file tipo ita.cube.bigram oppure ita.traindata

si indica quindi al programma dove sono ubicati i file della lingua mediante
export TESSDATA_PREFIX=/home/luca

e si procede da linea di comando ad effettuare una convesione OCR mediante il comando
tesseract immagine.tif testo_scansionato -l ita
(dove con -l ita si indica di utilizzare i file di conversione della lingua italiana, il risultato viene inserito nel file testo_scansionato.txt)

per testare il programma si possono scaricare le immagini di test da questo link

Attenzione: il programma accetta come input solo file TIF e solo con particolari caratteristiche prive di canale alpha. Nel caso di errato riconoscimento caratteri e' probabile che il file immagine non sia nel formato richiesto. Il formato richiesto e' di almeno 200 dpi con una profondita' di colore di 1 bit (quindi immagine a bianco e nero)

Come esempio di prova e' stata usata questa immagine


che ha come risultato della conversione il seguente testo
--------------------------------------------------------------------------------
 ~ Visual Linking allows users to join tables without having to understand
   the underlying table structure.

ReportSmith is available on MS-Windows and works with ASK INGRES/Net for
Windows to provide client/server reporting for ASK INGRES data.

Seeing is believing - - - so we've enclosed a Report Sampler that shows the types
of impressive reports available from ReportSmith.

We're so sure that ReportSmith will answer your reporting requirements that
we're offering a full development copy for $139~~over 50% off the regular price
-~so you can start generating new reports today!

Simply fill out the enclosed reply card, include your credit card number, mail it
back in the postage paid envelope, and we'll ship you a single-user copy of
ReportSmith today!

Once you've had a chance to use this product, I'm sure you'll see why it was
voted Byte Best of Comdex "Finalist" Award, and Windows Sources "Experts Pick"
Award.

For the reporting tool you've been waiting for... send in our reply card or call
1-800-446-4737 today! This special offer ends December 15th.


Regards,


George Kellar
Director Marketing
North America Operations
The ASK Group, Inc.


*This offer is limited to one user per site.
-------------------------------------------------------
facendo la prova invece con una pagina in italiano


e questa e' la ottima conversione in formato testo (gli errori ci sono ma sono abbastanza marginali e principalmente derivati da una cattiva interpretazione dell'apostrofo
------------------------------------------------------------------

ll punto di vista di ARPAT sulla prevenzione, enunciato formalmente fin dal i996, considera
elemento portante della propria politica gestionale la tutela della salute e sicurezza di tutti gli
operatori. Questo nel profondo convincimento che la salute sia un bene primario di cui ognuno ha
diritto, in ogni ambito di vita e di lavoro.

Le azioni intraprese da ARPAT per mettere in pratica questa scelta gestionale si sono concretizzate,
in primo luogo, nellîndividuazione del responsabile del servizio di prevenzione e protezione e,
successivamente, nellîndividuazione di tutte quelle ligure che rientrano nel “Sistema prevenzione”
dellfiAgenzia: fra queste, il medico competente ed autorizzato, che fin dal 1997 opera a fianco del
responsabile del servizio di prevenzione e protezione.

Le attività di quesfiultimo e del medico competente sono fra loro complementari e sinergiche,
Mentre il responsabile del servizio di prevenzione e protezione individua e propone le misure di
prevenzione primaria atte ad eliminare i rischi per la salute e la sicurezza degli operatori
dell’Agenzia (che sono legate ad interventi rivolti in particolare a strutture, impianti, attrezzature ed
organizzazione del lavoro), il medico competente rivolge la sua attenzione alla persona, mettendo in
pratica la sorveglianza sanitaria attraverso Feffettuazione di accertamenti sanitari preventivi e
periodici e l’espressione dei giudizi di idoneità allo svolgimento della mansione specifica di ogni
operatore delPAgenzia, Congiuntamente le due figure operano, invece, per Feffettuazione della
valutazione dei rischi e la formazione del personale delFAgenzia.

ll lavoro portato avanti fino ad oggi dal responsabile del servizio di prevenzione e protezione e dal
medico competente ha portato ARPAT ad accrescere progressivamente e sostanzialmente il suo
impegno nei riguardi delle norme di igiene e sicurezza. La situazione iniziale in cui si trovava
l'Agenzia al momento della sua costituzione, cosi come evidenziato nei primi documenti
predisposti e resi noti dal Direttore generale, vedeva oscillare, infatti, fra il 20 ed il 60% la
percentuale di adeguatezza alle norme tecniche (per le parti strutturali ed impiantistiche delle varie
sedi); d’altra parte anche il controllo sanitario dei lavoratori costituiva Feccezione alla regola,
risultando operativo, nel i996, solo in alcune realtà particolari.

Grazie all’impegno di tutta l’Agenzia, con il supporto tecnico del servizio di prevenzione e
protezione e del medico competente, la situazione iniziale e stata oggetto di un radicale mutamento.

Gli interventi tesi al miglioramento delle strutture e degli impianti, resisi necessari a seguito della
valutazione dei rischi, hanno portato alla realizzazione ed acquisizione di nuove sedi per il
Dipartimento provinciale di Siena, il Servizio sub-provinciale di Empoli, la Sede centrale
delFAgenzia a Firenze, Fimmobile “nuovo” per il dipartimento provinciale di Prato; di “locali
aggiuntivi” per i Dipartimenti provinciali di Livorno ed Arezzo ed a trasformazioni strutturali
(ampia quella del Dipartimento provinciale di Firenze), grazie anche agli interventi previsti dal
Progetto alimenti, per quanto riguarda la maggior parte delle altre sedi.

La politica degli investimenti adottata ha permesso, da un lato, di acquisire attrezzature specifiche
finalizzate alla sicurezza (annadi aspirati per sostanze pericolose, generatori di gas, cappe chimiche
e biologiche ecc.), dall’altro di procedere ad un rinnovamento delle dotazioni strumentali finalizzato
alla sicurezza prima di tutto, oltre che al miglioramento delle prestazioni.

Accanto a queste attività hanno preso vita progetti di formazione ed informazione interna del
personale ARPAT, attraverso incontri informativi gestiti dal responsabile del servizio di
prevenzione e protezione presso ogni sede e corsi di formazione su argomenti generali e specifici, ai
quali hanno preso parte finora circa 600 operatori (pronto soccorso, prevenzione incendi, rischio
cancerogeno e rischio biologico).

Contemporaneamente si e intrapreso un percorso di collaborazione con le altre Agenzie per la
protezione ambientale presenti sul territorio nazionale per accelerare il processo di miglioramento
grazie alla condivisione delle esperienze maturate fino ad ora, percorso che ha avuto recentemente
la sua manifestazione più concreta nel Seminario di Asti del giugno 2000 sullo stato di attuazione
del “sistema sicurezza nelle Agenzie”.




giovedì 29 novembre 2012

Utilizzo SQLite con C

Per terminare la carrellata dell'utilizzo di SQLite (gia' visto qui e qui) si vede l'utilizzo con C sia su Linux che su Windows (mediante Dev-C++ con il DevPack SQLite). Verrano ripetute le operazioni (piu' o meno) gia' eseguite con gli altri linguaggi

Nel progetto Dev-C++ si devono impostare l'utlizzo della libreria SQLite come indicato nella figura sottostante


per utilizzare il programma mediante Dev-C++ e' necessario che la libreria libsqlite3-0.dll sia accessibile (nella stessa directory dove viene ubicato l'eseguibile oppure in una directory di sistema di Windows...attenzione che dal sito di SQLite si scarica una dll nominata sqlite3.dll che non funziona all'interno di Dev-C++..si deve impiegare quella inserita nel DevPack)

Altrimenti dentro Linux il file puo' essere compilato con la seguente riga di comando
gcc -o sqlite_test sqlite_test.c  -Wall -O3 -lsqlite3

di seguito viene riportato il codice sorgente che e' sostanzialmente autoesplicativo in quanto effettua delle exec di comandi SQL
------------------------------------------------------------------------

#include <stdio.h>
#include <stdlib.h>
#include <sqlite3.h>


int main(int argc, char **argv){
sqlite3 *db;
char *zErrMsg = 0;
int rc;

rc = sqlite3_open("utenti.db", &db);
if (rc == SQLITE_OK)
{
rc = sqlite3_exec(db, "CREATE TABLE IF NOT EXISTS users  (Id INTEGER PRIMARY KEY,username TEXT, password TEXT, email TEXT)", NULL, 0, &zErrMsg);
if (rc != SQLITE_OK) fprintf(stderr, "SQL error: %s\n", zErrMsg);

rc = sqlite3_exec(db, "INSERT INTO users VALUES (NULL,\"luca\",\"password\",\"lucainnoc@gmail.com\")", NULL, 0, &zErrMsg);
if (rc != SQLITE_OK) fprintf(stderr, "SQL error: %s\n", zErrMsg);
rc = sqlite3_exec(db, "INSERT INTO users VALUES (NULL,\"chiara\",\"password\",\"chiara@gmail.com\")", NULL, 0, &zErrMsg);
if (rc != SQLITE_OK) fprintf(stderr, "SQL error: %s\n", zErrMsg);

sqlite3_stmt *stmt;
rc = sqlite3_prepare_v2(db, "SELECT * FROM users", -1, &stmt, 0);
if (rc == SQLITE_OK) {
int nCols = sqlite3_column_count(stmt);
if (nCols)
{
for (int nCol = 0; nCol < nCols; nCol++)
printf("%s\t", sqlite3_column_name(stmt, nCol));
printf("\n");
while ((rc = sqlite3_step(stmt)) == SQLITE_ROW)
for (int nCol = 0; nCol < nCols; nCol++)
printf("%s\t", sqlite3_column_text(stmt, nCol));
printf("\n");
}
sqlite3_finalize(stmt);
}

rc = sqlite3_exec(db, "DELETE FROM users", NULL, 0, &zErrMsg);
if (rc != SQLITE_OK) fprintf(stderr, "SQL error: %s\n", zErrMsg);

sqlite3_close(db);
} else {
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(db));
sqlite3_close(db);
return 1;
}
return 0;

}

mercoledì 28 novembre 2012

Utilizzo di SQLite con Android

Riprendendo il precedente post qui verra' mostrato come usare SQLite all'interno di Android partendo da questo esempio

Per prima cosa si deve scrivere una classe di helper in cui sono identificate la struttura del Db e le operazioni da eseguire...per semplicita' viene utilizzata la struttura gia' vista nello script Python

-----------------------------------------------------------------------

package com.example.sqlite;

import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteOpenHelper;

public class database {
SQLiteDatabase mDb;
     DbHelper mDbHelper;
     Context mContext;
     private static final String DB_NAME="users";//nome del db
     private static final int DB_VERSION=1; //numero del db
     
     public database(Context ctx){
             mContext=ctx;
             mDbHelper=new DbHelper(ctx, DB_NAME, null, DB_VERSION);        
     }
     
     public void open(){  
             mDb=mDbHelper.getWritableDatabase();
             
     }
     
     public void close(){ 
             mDb.close();
     }
     
     public void insertUtente(String name,String password, String email){ // genera l'INSERT
             ContentValues cv=new ContentValues();
             cv.put(UsersMetaData.USERNAME_KEY, name);
             cv.put(UsersMetaData.PASSWORD_KEY, password);
             cv.put(UsersMetaData.EMAIL_KEY,email);
             mDb.insert(UsersMetaData.USERS_TABLE, null, cv);
     }
     
     public void deleteUtente(String id) {
         mDb.delete(UsersMetaData.USERS_TABLE, UsersMetaData.ID + "=" + id, null);
     }

    
     public Cursor fetchUtenti(){ //genera la SELECT
             return mDb.query(UsersMetaData.USERS_TABLE, null,null,null,null,null,null);               
     }

     public class UsersMetaData {  // variabili
             static final String USERS_TABLE = "users";
             static final String ID = "_id";
             static final String USERNAME_KEY = "username";
             static final String PASSWORD_KEY = "password";
             static final String EMAIL_KEY = "email";
     }

     private static final String USERS_TABLE_CREATE = "CREATE TABLE IF NOT EXISTS "  //CREATE TABLE
                     + UsersMetaData.USERS_TABLE + " (" 
                     + UsersMetaData.ID+ " integer primary key autoincrement, "
                     + UsersMetaData.USERNAME_KEY + " text not null, "
                     + UsersMetaData.PASSWORD_KEY + " text not null, "
                     + UsersMetaData.EMAIL_KEY + " text not null);";

     private class DbHelper extends SQLiteOpenHelper { //helper

             public DbHelper(Context context, String name, CursorFactory factory,int version) {
                     super(context, name, factory, version);
             }

             public void onCreate(SQLiteDatabase _db) { //solo quando il db viene creato, creiamo la tabella
                     _db.execSQL(USERS_TABLE_CREATE);
             }

             public void onUpgrade(SQLiteDatabase _db, int oldVersion, int newVersion) {

             }

     }
}
---------------------------------------------------------

a questo punto si puo' procedere utilizzando la classe helper 
il codice e' sostanzialmente autoesplicativo
----------------------------------------------------------
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
database utenti =new database(getApplicationContext());
utenti.open();
utenti.insertUtente("luca", "password", "lucainnoc@gmail.com");
utenti.insertUtente("chiara", "password", "chiara@gmail.com");
        
Cursor c=utenti.fetchUtenti(); // query
        
int usernameCol=c.getColumnIndex(UsersMetaData.USERNAME_KEY);  //indici delle colonne
int passwordCol=c.getColumnIndex(UsersMetaData.PASSWORD_KEY);
int mailCol=c.getColumnIndex(UsersMetaData.EMAIL_KEY);
        
       if(c.moveToFirst()){  
            do {
                Log.d("select",c.getString(usernameCol)+" "+c.getString(passwordCol)+" "+c.getString(mailCol));                    
                    } while (c.moveToNext());//iteriamo al prossimo elemento
        }
    
utenti.deleteUtente("1");
        
if(c.moveToFirst()){  
        do {
            Log.d("select",c.getString(usernameCol)+" "+c.getString(passwordCol)+" "+c.getString(mailCol));                    
             } while (c.moveToNext());//iteriamo al prossimo elemento
    }
utenti.close();
        

Utilizzo di base di Sqlite con Python

Sqlite e' un database basato su un unico file che ha bindings in numerosi linguaggi; l'aspetto piu' interessanto dell'impiego con Python e' quello di essere inserito all'interno del linguaggio (per lo meno dalla versione 2.5 e superiori) e per questo motivo non e' necessario utilizzare librerie esterne per il suo impiego

Di seguito un esempio delle operazioni base effettuate impiegando Python...attenzione al commit finale (Sqlite ammette le transazioni atomiche)
-----------------------------------------------------------------

from sqlite3 import dbapi2 as sqlite
con = sqlite.connect('utenti.db')
cur = con.cursor()

#crea una tabella con un indice ad autoincremento
cur.execute('CREATE TABLE users (Id INTEGER PRIMARY KEY,username TEXT, password TEXT, email TEXT)')

#effettua alcuni inserimenti
# NULL serve per l'indice ad autoincremento
cur.execute('INSERT INTO users VALUES (NULL,"luca","password","lucainnoc@gmail.com")')
cur.execute('INSERT INTO users VALUES (NULL,"chiara","password","chiara@gmail.com")')
print "-----------------------"

#estrae tutte le righe
cur.execute('SELECT * FROM users')
for row in cur:
    print row
    print row[0]
    print row[1]
    print row[2]
    print "-----------------------"

#estrae solo una riga
nome = "luca"
cur.execute("SELECT * FROM users WHERE username=:username",  {"username": nome})
for row in cur:
    print row
print "-----------------------"

#cancella una riga
cur.execute("DELETE FROM users WHERE username=:username",  {"username": nome})

cur.execute('SELECT * FROM users')
for row in cur:
    print row
print "-----------------------"

#cancella la tabella
cur.execute('DROP TABLE users')
con.commit()
cur.close()

Change Detection with structural similarity

L'idea di base e' quella di cercare le differenze tra le due immagini sottostanti Non e' immediatamente visibile ma ci sono dei ...