Hi again ..
Below you can download the hand in for the course assignment.
The archive contains the report, the source code and exported file (.apk) just in case you can not use the code.
Have fun.
/ Gabriel
Course Assignment - Gabriel Florentin Hirjoi
Wednesday, April 25, 2012
Sunday, March 25, 2012
SQLite Exception and Course Assignment Requirements
The SQLite Exception:
java.lang.RuntimeException: Unable to destroy activity: android.database.sqlite.SQLiteException: unable to close due to unfinalised statements I have gotten this exception while trying to close the SQLite database. It was due to trying to close the database before closing the Cursor object I was using to operate on the database. Closing the cursor before trying to close the database did the job here. ;)The Course Assignment Requirements:You can download the pdf version for the requirements using the link below:BeatSketch - requirements - pdf
Sunday, March 18, 2012
Course Project Ideas
The first idea is a game, more specifically an enhanced version of PacMan where the character would not only eat ghosts but also use special skills to make catch the ghosts or easily escape from them. The game would be a combination of PacMan, Mortal Kombat and Dragon Ball .. the last two being my favorite game and respectively tv series when I was a child.
The second idea refers to a translation application. It should be possible to take a picture of a text and than the application should extract the text from the picture using an OCR (Optical Character Recognition) and post it in a TextEdit view for the user to modify it in case there are errors in the spelling of the text. Then the user should be able to select the language the text is written in and the language he wants the text to be translated into and send the text to google translate API using the internet connection from the phone. It should also display the translated text the google API sent back to the application.
The third idea is a beat sketch application. The user should be able the sketch a beat by using a certain number of pads on the screen of the device. He should be able to make changes when it comes to the tempo of the beat, the samples selected to play when the pads are touched and also to save the file as mp3. The user should also receive help when it comes to the timing part of the beat, meaning a special sound indicating the bar number he is on at the moment and also after he finishes recording a small part of the beat the app should play the recorded layers of the beat so the user hears what he composed. Below is a screen shot of what the application could look like (this is the screen shot of the iMaschine, the iPhone application Native Instruments released not a long time ago; this application is designed to work with Maschine, a Native Instruments product ):
The second idea refers to a translation application. It should be possible to take a picture of a text and than the application should extract the text from the picture using an OCR (Optical Character Recognition) and post it in a TextEdit view for the user to modify it in case there are errors in the spelling of the text. Then the user should be able to select the language the text is written in and the language he wants the text to be translated into and send the text to google translate API using the internet connection from the phone. It should also display the translated text the google API sent back to the application.
The third idea is a beat sketch application. The user should be able the sketch a beat by using a certain number of pads on the screen of the device. He should be able to make changes when it comes to the tempo of the beat, the samples selected to play when the pads are touched and also to save the file as mp3. The user should also receive help when it comes to the timing part of the beat, meaning a special sound indicating the bar number he is on at the moment and also after he finishes recording a small part of the beat the app should play the recorded layers of the beat so the user hears what he composed. Below is a screen shot of what the application could look like (this is the screen shot of the iMaschine, the iPhone application Native Instruments released not a long time ago; this application is designed to work with Maschine, a Native Instruments product ):
Monday, March 12, 2012
Draw App
For this week's hand in we have had to
develop an application which allows the user to draw on the screen
using gestures.
Our application (see the screen shot
below) includes several controls for the app, controls like clearing
the screen, increase or decrease the size of the dots, selecting the
color, draw different shapes (dots, lines, squares and circles) and
also controls for undoing and redoing.
The main view of the application
consists of a view with a linear layout (with vertical orientation)
which nests a draw view as it's first child and the controls as the
second child. The reason for using a nested view is purely esthetic,
its role being to hold the drawing area on top of the screen and the
controls on the bottom.
The drawing is done by using an
ArrayList of DrawingElement and than in the onDraw() method just
paint all the elements in the ArrayList on the screen.
In order to be able to store all the
elements the application uses the onTouch() method together with the
actions provides by the event calling this method.
Each DrawingElement contains fields for
storing the x and y coordinates when the user touches the screen and
x and y coordinates for the case when the user releases the screen
(these last two coordinates help in drawing all the figures except
for the dots). The element also contains fields for storing the color
the element has, its type and also its size (in case of the dots).
After each element is stored the
invalidate() method is called in order to tell the AndroidOS to draw
every element on the screen.
One could thing is that the user can select the color to draw with by using the color control. This control opens a ColorPickerDialog which changes the value of the color field in the DrawView (the nested view):
It is able to change the value of the color field because the nested view implements the ColorPickerDialog.OnColorChangedListener and than because I have implemented the colorChanged() method required for the listener:
public void colorChanged(int color) {
paint.setColor(color);
}
Another cool thing about the application is
that it allows the user to see the size of the shapes he is drawing
while dragging on the screen. This is done by implementing the case
of MotionEvent.MOTION_MOVE. The code snippet below demonstrates this
case:
in the onTouch(View view, MotionEvent
event) method:
if (event.getAction() == MotionEvent.ACTION_MOVE) {
if (moveFlag == 1 && elements.size() != 0) {
if (lineFlag == 1 || circleFlag == 1 || squareFlag == 1) {
// remove the element added by the move event
elements.remove(elements.size() - 1);
}
}
// get the end point of the line
element.x2 = event.getX();
element.y2 = event.getY();
element.color = paint.getColor();
element.size = drawSize;
if (dotFlag == 1) {
element.type = 1;
} else if (lineFlag == 1) {
element.type = 2;
} else if (circleFlag == 1) {
element.type = 3;
} else if (squareFlag == 1) {
element.type = 4;
}
// prepare a temporary element to be added to the array of
// drawing elements
DrawingElement tempElement = new DrawingElement();
tempElement.x1 = element.x1;
tempElement.y1 = element.y1;
tempElement.x2 = element.x2;
tempElement.y2 = element.y2;
tempElement.type = element.type;
tempElement.color = element.color;
tempElement.size = element.size;
elements.add(tempElement);
invalidate();
if (moveFlag == 0) {
if (lineFlag == 1 || circleFlag == 1 || squareFlag == 1) {
moveFlag = 1;
}
}
}
One other cool thing for the
application is that the user can undo and redo all of his actions.
This has been achieved using a stack to push the elements when the
user select the undo control and pop the elements when the user
selects the redo control.
public boolean onUndoClick() {
if (elements.size() != 0) {
// get the last elements
DrawingElement d = elements.get(elements.size() - 1);
// remove last element from the elements array
elements.remove(elements.size() - 1);
// insert the element in the stack
undoStack.push(d);
invalidate();
// add the action to the log
Log.d(TAG, "Undo: " + d);
}
return true;
}
public boolean onRedoClick() {
if (!undoStack.isEmpty()) {
// get the last element added to the undoStack
DrawingElement d = undoStack.pop();
// add the element to the elements array
elements.add(d);
// draw the screen
invalidate();
// add the action to the log
Log.d(TAG, "Redo: " + d);
}
return true;
}
The application can be downloaded from
the link below:
The apk:
The source code:
It took me approximately 10 hours to
develop this app. I have spent most of the time dealing with the
motion event. The undo and redo controls took me approximately 10
minutes to implement. ;)
Tuesday, March 6, 2012
Swipe Application
For this third hand in we have made a Swipe Application which offers the user the ability to swipe between three activities.
The activities themselves follow the same design meaning each of them contain a list.
The list is displayed on the screen using a ListView which offers the ability to scroll up and down on the list.
Below there is a screen shot of the first activity of the application (AVD targer Android 2.2 SDK version 8):
All the activities in the application follow the same layout set in the list_item.xml file.
The layout contains on a text view in which we have to specify the attributes for each of the list's element:
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/textview"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:padding="16sp"
android:textSize="20dp" />
The steps the activities follow on their creation are very simple ones, with the first being retrieving the array of Strings used to populate the ListView from the strings.xml file in the res/values folder of the application.
<string-array name="countries_array">
<item>Bahrain</item>
<item>Bangladesh</item>
<item>Barbados</item>
<item>Belarus</item>
<item>Belgium</item>
<item>Belize</item>
<item>Benin</item>
<item>Romania</item>
<item>Bulgaria</item>
<item>Sweden</item>
<item>Denmark</item>
<item>Norvway</item>
<item>Irak</item>
<item>Afganistan</item>
</string-array>
Then the ListView is populated with the retrieved values:
//get the String array from the strings.xml
String[] countries = getResources().getStringArray(
R.array.countries_array);
// populate the list
setListAdapter(new ArrayAdapter<String>(this, R.layout.list_item,
countries));
final ListView lv = getListView();
lv.setTextFilterEnabled(true);
These steps being done the running activity has a populated ListView which allows the user to scroll up and down through the displayed data.
Now all that is left to do is to add the swipe capabilities so that the user would swipe when he / she wants to change the activity / screen.
We do that by setting the OnTouchListener for the ListView since it covers all the screen of the activity.
// set the listener for the swipe
lv.setOnTouchListener(new OnTouchListener() {
public boolean onTouch(View view, MotionEvent event) {
synchronized (event) {
try {
// wait for 16 ms
event.wait(16);
// when the user touches the screen
if (event.getAction() == MotionEvent.ACTION_DOWN) {
// reset deltaX
deltaX = 0;
// get initial positions
initialX = event.getRawX();
event.getRawY();
}
// when the user releases the screen
if (event.getAction() == MotionEvent.ACTION_UP) {
deltaX = event.getRawX() - initialX;
// swiped to the left
if (deltaX > 100) {
Toast.makeText(getApplicationContext(),
"Swipe to the other direction",
Toast.LENGTH_SHORT).show();
}
// swiped to the right
else if (deltaX < -100) {
startActivity(new Intent(activity, SecondActivity.class));
}
return false;
}
} catch (InterruptedException e) {
return false;
}
}
return false;
}
});
By calculating the difference of the x coordinate of the point when the user touches the screen and the x coordinate of the point when the user releases the screen we can determine on which direction the user swiped. If the deltaX is greater than 0 the user swiped to the left otherwise the user swiped to the right.
When we test if there was a right or a left swipe we compare deltaX with the absolute value of 100, if deltaX is greater than 100 the activity will launch the code for swiping to the left otherwise if deltaX is smaller than -100 is launches the code for swiping to the right. The reason we chose to compare deltaX with the absolute value of 100 is so that we will be sure that the user wants to swipe and not only touching the screen since the android phone can not detect with very high precision the point the user touches the screen on.
After the OnTouchListener is set the application displays a simple text informing the user what type of information the ListView is displaying, and this is done with the help of a Toast object:
Toast.makeText(getApplicationContext(), "Countries", Toast.LENGTH_SHORT).show();
The project can be downloaded by following the link below:
http://uploading.com/files/c6ma4618/Swipper.rar/
Surprisingly the development time of this application is around 5 hours, give or take some minutes, most of this time being spent on creating the swipe capabilities.
The activities themselves follow the same design meaning each of them contain a list.
The list is displayed on the screen using a ListView which offers the ability to scroll up and down on the list.
Below there is a screen shot of the first activity of the application (AVD targer Android 2.2 SDK version 8):
The layout contains on a text view in which we have to specify the attributes for each of the list's element:
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/textview"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:padding="16sp"
android:textSize="20dp" />
The steps the activities follow on their creation are very simple ones, with the first being retrieving the array of Strings used to populate the ListView from the strings.xml file in the res/values folder of the application.
<string-array name="countries_array">
<item>Bahrain</item>
<item>Bangladesh</item>
<item>Barbados</item>
<item>Belarus</item>
<item>Belgium</item>
<item>Belize</item>
<item>Benin</item>
<item>Romania</item>
<item>Bulgaria</item>
<item>Sweden</item>
<item>Denmark</item>
<item>Norvway</item>
<item>Irak</item>
<item>Afganistan</item>
</string-array>
Then the ListView is populated with the retrieved values:
//get the String array from the strings.xml
String[] countries = getResources().getStringArray(
R.array.countries_array);
// populate the list
setListAdapter(new ArrayAdapter<String>(this, R.layout.list_item,
countries));
final ListView lv = getListView();
lv.setTextFilterEnabled(true);
These steps being done the running activity has a populated ListView which allows the user to scroll up and down through the displayed data.
Now all that is left to do is to add the swipe capabilities so that the user would swipe when he / she wants to change the activity / screen.
We do that by setting the OnTouchListener for the ListView since it covers all the screen of the activity.
// set the listener for the swipe
lv.setOnTouchListener(new OnTouchListener() {
public boolean onTouch(View view, MotionEvent event) {
synchronized (event) {
try {
// wait for 16 ms
event.wait(16);
// when the user touches the screen
if (event.getAction() == MotionEvent.ACTION_DOWN) {
// reset deltaX
deltaX = 0;
// get initial positions
initialX = event.getRawX();
event.getRawY();
}
// when the user releases the screen
if (event.getAction() == MotionEvent.ACTION_UP) {
deltaX = event.getRawX() - initialX;
// swiped to the left
if (deltaX > 100) {
Toast.makeText(getApplicationContext(),
"Swipe to the other direction",
Toast.LENGTH_SHORT).show();
}
// swiped to the right
else if (deltaX < -100) {
startActivity(new Intent(activity, SecondActivity.class));
}
return false;
}
} catch (InterruptedException e) {
return false;
}
}
return false;
}
});
By calculating the difference of the x coordinate of the point when the user touches the screen and the x coordinate of the point when the user releases the screen we can determine on which direction the user swiped. If the deltaX is greater than 0 the user swiped to the left otherwise the user swiped to the right.
When we test if there was a right or a left swipe we compare deltaX with the absolute value of 100, if deltaX is greater than 100 the activity will launch the code for swiping to the left otherwise if deltaX is smaller than -100 is launches the code for swiping to the right. The reason we chose to compare deltaX with the absolute value of 100 is so that we will be sure that the user wants to swipe and not only touching the screen since the android phone can not detect with very high precision the point the user touches the screen on.
After the OnTouchListener is set the application displays a simple text informing the user what type of information the ListView is displaying, and this is done with the help of a Toast object:
Toast.makeText(getApplicationContext(), "Countries", Toast.LENGTH_SHORT).show();
The project can be downloaded by following the link below:
http://uploading.com/files/c6ma4618/Swipper.rar/
Surprisingly the development time of this application is around 5 hours, give or take some minutes, most of this time being spent on creating the swipe capabilities.
Sunday, February 26, 2012
MediaPlayer Application (week no 8)
We have added a simple User Interface for the MediaPlayer application.
The UI includes a TextView for the name of the application, another TextView for the name of the audio file and 4 buttons for opening, playing, pausing and stopping the playback of the file.
We have also included a seek bar in the UI, seek bar which should show the progress of the playback and also let the user seek trough the audio file.
Below there is a screenshot of the application ran on an AVD (target Android 2.2 SDK version 8)
Nothing too fancy until this point. ;)
In order to be able to use the elements we have added in the UI in our java classes we have had to add IDs to all elements in the interface and also add the onClickListener methods for the buttons.
For the Play, Pause and Stop button the story is pretty much the same, when pressing the Play button the Media element should start the playback (in our case it is only effective to press the Play button after we have pressed the Pause button to pause the playback), the Pause button to temporarily pause the playback and the Stop button to stop playing the file. The only exception here is that when we stop the playback of the file we should also release the resources the Media Element was using to play the file.
public void onPlay(View view) {
if (mMediaPlayer != null) {
mMediaPlayer.start();
}
}
public void onPause(View view) {
if (mMediaPlayer != null) {
mMediaPlayer.pause();
}
}
public void onStop(View view) {
mMediaPlayer.stop();
mMediaPlayer.release();
mMediaPlayer = null;
}
When we start the application the first thing we must do in order to play a file is to select the file we want to play. In our case unfortunately we could not figure out how exactly to fetch the files from the internal memory and the SD card of the AVD, even if we have tried doing so with Uris and also by pushing files to the mnt/sdcard folder of the AVD. Instead we play a file located in the res/raw folder of the application, folder which we have created and which is supposed to include raw media elements, but for the moment we have added encoded files for a better quality of the playback.
In order to play a MediaPlayer element we must first set the DataSource for it by calling one of the setDataSource methods presented on the element. Calling this method can result in the main thread trowing illegal state, illegal argument and IO exceptions.
After we set the data source Android requires that we prepare the MediaPlayer element for playback by calling the prepare() methods.
All that is left to do at this moment is to start the playback of the file by calling the MediaPlayer.start() method.
public void onOpen(View view) {
if (mMediaPlayer == null) {
mMediaPlayer = new MediaPlayer();
}
AssetFileDescriptor afd = getApplicationContext().getResources()
.openRawResourceFd(R.raw.fever);
seekBar = (SeekBar) findViewById(R.id.seekBar);
mediaTitle = (TextView) findViewById(R.id.titleTextView);
try {
mMediaPlayer.setDataSource(afd.getFileDescriptor(),
afd.getStartOffset(), afd.getLength());
afd.close();
mMediaPlayer.prepare();
mediaTitle.setText("Title: fever");
seekBar.setMax(mMediaPlayer.getDuration());
seekBar.setProgress(0);
progressThread.start();
mMediaPlayer.start();
} catch (IllegalArgumentException e) {
...
} catch (IllegalStateException e) {
...
} catch (IOException e) {
...
}
}
As it can be seen in the code above there are also some settings we make for the seek bar we have placed in the UI.
We set the maximum length of the bar to the length of the MediaPlayer element and we also set the shown progress to 0.
Just before we start the playback of the file we also start a Thread (the progressThread) which updates the seek bar. Here is the definition of this thread:
Thread progressThread = new Thread() {
@Override
public void run() {
int currentPosition = 0;
int total = mMediaPlayer.getDuration();
seekBar.setIndeterminate(false);
while (mMediaPlayer != null && currentPosition < total) {
try {
currentPosition = mMediaPlayer.getCurrentPosition();
Thread.sleep(1000);
} catch (InterruptedException e) {
return;
} catch (Exception e) {
return;
}
seekBar.setProgress(currentPosition);
}
}
};
At this point if the user opens a file the seek bar will be updated automatically every 1 second with the progress in the playback of the file. But all this could have been done using just a progress bar. Instead we also want the user to be able to seek to various parts of the file. To do so our activity class has to implement the SeekBar.OnSeekBarChangeListener. By acting in this manner we also get to implement the methods on the OnSeekBarChangeListener interface, from which the most noticeable is the onProgressChanged method:
public void onProgressChanged(SeekBar seekBar, int progress,
boolean fromUser) {
if (fromUser && mMediaPlayer.isPlaying()) {
mMediaPlayer.seekTo(progress);
}
}
The boolean fromUser parameter helps us identify if the event has occurred due to a action from the user or due to the progress thread we have defined earlier.
Unfortunately this last step is not working for our application due to unknown reasons. Not even some hours spent on Google and on the Android documentation could not help us in this manner since we have fallowed all the steps required and still no results were produced.
The full project can be downloaded from the link below:
MediaPlayer Application Project
We can not say that developing this simple application was hard or easy, but what we can say is that is was time consuming work, most of it being spent on researching various methods which could have helped us with the seek bar and also with the use of internal / sdcard memory of the device.
The UI includes a TextView for the name of the application, another TextView for the name of the audio file and 4 buttons for opening, playing, pausing and stopping the playback of the file.
We have also included a seek bar in the UI, seek bar which should show the progress of the playback and also let the user seek trough the audio file.
Below there is a screenshot of the application ran on an AVD (target Android 2.2 SDK version 8)
Nothing too fancy until this point. ;)
In order to be able to use the elements we have added in the UI in our java classes we have had to add IDs to all elements in the interface and also add the onClickListener methods for the buttons.
For the Play, Pause and Stop button the story is pretty much the same, when pressing the Play button the Media element should start the playback (in our case it is only effective to press the Play button after we have pressed the Pause button to pause the playback), the Pause button to temporarily pause the playback and the Stop button to stop playing the file. The only exception here is that when we stop the playback of the file we should also release the resources the Media Element was using to play the file.
public void onPlay(View view) {
if (mMediaPlayer != null) {
mMediaPlayer.start();
}
}
public void onPause(View view) {
if (mMediaPlayer != null) {
mMediaPlayer.pause();
}
}
public void onStop(View view) {
mMediaPlayer.stop();
mMediaPlayer.release();
mMediaPlayer = null;
}
When we start the application the first thing we must do in order to play a file is to select the file we want to play. In our case unfortunately we could not figure out how exactly to fetch the files from the internal memory and the SD card of the AVD, even if we have tried doing so with Uris and also by pushing files to the mnt/sdcard folder of the AVD. Instead we play a file located in the res/raw folder of the application, folder which we have created and which is supposed to include raw media elements, but for the moment we have added encoded files for a better quality of the playback.
In order to play a MediaPlayer element we must first set the DataSource for it by calling one of the setDataSource methods presented on the element. Calling this method can result in the main thread trowing illegal state, illegal argument and IO exceptions.
After we set the data source Android requires that we prepare the MediaPlayer element for playback by calling the prepare() methods.
All that is left to do at this moment is to start the playback of the file by calling the MediaPlayer.start() method.
public void onOpen(View view) {
if (mMediaPlayer == null) {
mMediaPlayer = new MediaPlayer();
}
AssetFileDescriptor afd = getApplicationContext().getResources()
.openRawResourceFd(R.raw.fever);
seekBar = (SeekBar) findViewById(R.id.seekBar);
mediaTitle = (TextView) findViewById(R.id.titleTextView);
try {
mMediaPlayer.setDataSource(afd.getFileDescriptor(),
afd.getStartOffset(), afd.getLength());
afd.close();
mMediaPlayer.prepare();
mediaTitle.setText("Title: fever");
seekBar.setMax(mMediaPlayer.getDuration());
seekBar.setProgress(0);
progressThread.start();
mMediaPlayer.start();
} catch (IllegalArgumentException e) {
...
} catch (IllegalStateException e) {
...
} catch (IOException e) {
...
}
}
As it can be seen in the code above there are also some settings we make for the seek bar we have placed in the UI.
We set the maximum length of the bar to the length of the MediaPlayer element and we also set the shown progress to 0.
Just before we start the playback of the file we also start a Thread (the progressThread) which updates the seek bar. Here is the definition of this thread:
Thread progressThread = new Thread() {
@Override
public void run() {
int currentPosition = 0;
int total = mMediaPlayer.getDuration();
seekBar.setIndeterminate(false);
while (mMediaPlayer != null && currentPosition < total) {
try {
currentPosition = mMediaPlayer.getCurrentPosition();
Thread.sleep(1000);
} catch (InterruptedException e) {
return;
} catch (Exception e) {
return;
}
seekBar.setProgress(currentPosition);
}
}
};
At this point if the user opens a file the seek bar will be updated automatically every 1 second with the progress in the playback of the file. But all this could have been done using just a progress bar. Instead we also want the user to be able to seek to various parts of the file. To do so our activity class has to implement the SeekBar.OnSeekBarChangeListener. By acting in this manner we also get to implement the methods on the OnSeekBarChangeListener interface, from which the most noticeable is the onProgressChanged method:
public void onProgressChanged(SeekBar seekBar, int progress,
boolean fromUser) {
if (fromUser && mMediaPlayer.isPlaying()) {
mMediaPlayer.seekTo(progress);
}
}
The boolean fromUser parameter helps us identify if the event has occurred due to a action from the user or due to the progress thread we have defined earlier.
Unfortunately this last step is not working for our application due to unknown reasons. Not even some hours spent on Google and on the Android documentation could not help us in this manner since we have fallowed all the steps required and still no results were produced.
The full project can be downloaded from the link below:
MediaPlayer Application Project
We can not say that developing this simple application was hard or easy, but what we can say is that is was time consuming work, most of it being spent on researching various methods which could have helped us with the seek bar and also with the use of internal / sdcard memory of the device.
Monday, February 13, 2012
Mobile Application Development MOB I1 (week no 7)
Mobile Applications Development - MOB I1
First Week
Group members:
Gabriel - Florentin Hirjoi, student no: 126645
Claudiu - Lucian Manea, student no: 130935
Georgiana Barbu, student no: 127811
Teacher e-mail address:
fx@cs.au.dk
First Week
Group members:
Gabriel - Florentin Hirjoi, student no: 126645
Claudiu - Lucian Manea, student no: 130935
Georgiana Barbu, student no: 127811
Teacher e-mail address:
fx@cs.au.dk
Subscribe to:
Posts (Atom)