Friday, February 25, 2011

What more can you ASCII for?

Greetings

I am not the artistic type, but that does not mean that I do not appreciate art. While I do like art created by others, I sometimes wish that I too was gifted with artistic abilities. Once again, using the awesome powers of Java, I was able to realize this dream. Using simple Java code, I created an image to ASCII converter program which takes an image and outputs a text file with ASCII characters, which looks like the input image when viewed all at once. In this post I will explain the basic principle behind computer generated ASCII art, and how to implement it using Java. I will release my own code for the program that I built, the program itself and a short video of the program in action. From now on, I will release all my code under the Open Source Initiative (OSI) MIT License (MIT), because the license is clean, short and easy to understand. Before I explain the principle behind ASCII art, here are some of the ASCII art created by the program: 
That's me!!

The Simpsons are ASCIIfied!
p-51 Mustang- A classic

The principle behind ASCII art is easy to understand. An image can be thought of as an 2 dimension array with rows and columns that correspond to the width and height of the image (In reality, it is more complex than that but this is a good enough understanding for the time being). Each cell in this array holds a pixel. When all these pixels are viewed together at once, you get an image. So a 400x400 image has 400 rows and 400 columns, and has 160000 pixels. Now to convert an image to text, the program has to iterate through all the pixels, get its RGB value, convert it to its gray scale value, and append a suitable ASCII character. Java has a class called the BufferedImage class, which has the methods necessary to obtain the RGB values of a pixel. Now once the RGB values(ranging from 0 to 255) of the pixel is obtained, there is a simple formula to convert it to a gray scale value. Simply take ~30% of the Red Value,  ~59% of the Green value and ~11% of the blue value and add them up together to a value ranging from 0 to 255, 0 being the darkest and 255 being the brightest. Don't worry, the BufferedImage class has the necessary methods to obtain these values,which you can see in my annotated code posted below.This value is your gray scale value. Generally gray scale value range from 0 to 1 with decimals in between, but there is no reason for them to be between 0 and 255. You could divide it by 255 to range it from 0 to 1, but why the extra hassle? After obtaining the gray scale values, the creative part begins. As you know, some ASCII characters appear darker than the others. For example, # appears darker than 0.So what the program does is obtain the gray scale value of each pixel, and according to the value, append an ASCII character to the text file. For example, if the gray scale value is in between 180 and 200(very bright), it appends a ".", and if it is 50 and 70(very dark), it appends  a "#". Currently there are nine different ASCII characters that the program chooses from to append. However, I urge you to edit the code and experiment by adding more characters, changing the existing characters, etc. For a detailed explanation, read the comments in my code which explain the function and workings of each method. The results are stored in a file called result.txt, and when viewing the results in a text editor, choose a small bold mono-spaced font, with word wrapping DISABLED. I recommend choosing images not bigger than 600x600 as the task will become more and more resource intensive with bigger images. While the current program does have an area where the results are appended, for spectacular results, open up the result.txt file and choose the best font and size to view the art. I find that Prestige Std Elite and OCR A work really well, almost giving the illusion of colors. This program is only a test, and soon I will be releasing code to a program with better GUI, more options and ultimately more robust. But until then, here is a small video below which runs you through the code of the program and explains how the program works. 


And finally here are the links to the code of the program and the actual program and the Netbeans Project. Please go through the code, experiment with it and hope you learn something out of this.


This is not the end

Aravind Rao
                  

6 comments:

  1. i like what you just did, really cool man... something have to be said about the post, PLEASE SEPARATE YOUR PARAGRAPH, ITS REALLY HARD TO READ! i know you are not too into the design thing, but come one man.. Besides that.. Good job.

    ReplyDelete
  2. Yeah, it is kind of muddled isn't it.. thanks for the input, I will change the font color and size and edit my paragraphs better the next time.

    ReplyDelete
  3. Thanks for posting this! I've changed the structure of the class a bit to make it easier to convert images to ascii from other classes. The main() method now has all the GUI code, which instantiates an ASCII object to convert a selected file.

    This also uses a StringBuilder to build the string, which is very fast (also, we know the exact length of the finished String, so we can initialize the StringBuilder's buffer size accurately).

    https://gist.github.com/3859200

    ReplyDelete
  4. Thanks for this. I'm using it as a starting point to convert webcam to ascii art. I'll post a link when I'm done.

    ReplyDelete