DEV Community

Cover image for Character editor
Adrian
Adrian

Posted on

Character editor

Intro

Learn about binary and character representation while implementing this useful character editor. The tool can be useful in defining custom sprite characters for your games.

There are other tutorials on CodeGuppy that shows you how to use custom sprites in your programs.

What do you need?

To run the code in this article you don’t need to install anything on your local computer.

Just open the online JavaScript playground from codeguppy.com/code.html and copy and paste the following code in the integrated editor.

When ready, press the “Play” button to run the code.

Source code

var mapX = 50;
var mapY = 50;
var squareSize = 25;
var color0 = "White";
var color1 = "Black";

var valuesX = 290;
var valuesY = 50;

var instructionsX = 450;
var instructionsY = 62;

var map = [ 
        [0, 0, 0, 0, 0, 0, 0, 0],
        [0, 0, 1, 1, 1, 1, 0, 0],
        [0, 1, 0, 0, 0, 0, 1, 0],
        [0, 1, 0, 0, 0, 0, 1, 0],
        [0, 1, 1, 1, 1, 1, 1, 0],
        [0, 1, 0, 0, 0, 0, 1, 0],
        [0, 1, 0, 0, 0, 0, 1, 0],
        [0, 0, 0, 0, 0, 0, 0, 0]
    ];

var chars = [];
var charDot = 2;
var charsY = 300;

function loop()
{
    clear();

    displayMap();
    displayValues();
    displayInstructions();

    displayChars();    

    checkMouse();
}

function keyPressed()
{
    switch(keyCode)
    {
        case UP_ARROW:
            shiftLeft(map);
            break;
        case DOWN_ARROW:
            shiftRight(map);
            break;
        case LEFT_ARROW:
            shiftMapLeft();
            break;
        case RIGHT_ARROW:
            shiftMapRight();
            break;
    }

    switch(key)
    {
        case "H":
            mirrorMap();
            break;
        case "V":
            mirrorArray(map);
            break;
        case "R":
            rotateRight();
            break;
        case "L":
            rotateLeft();
            break;
        case "X":
            var ar = getChar();
            chars.push(ar);
            break;
    }
}


function checkMouse()
{
    if (!mouseIsPressed)
        return;

    var o = findRowCol(mouseX, mouseY);
    if (!o)
        return;

    map[o.row][o.col] = mouseButton == LEFT ? 1 : 0;
}

function findRowCol(x, y)
{
    var col = floor( (x - mapX) / squareSize );
    var row = floor( (y - mapY) / squareSize);

    if (row < 0 || row >= map.length)
        return null;

    if (col < 0 || col >= map[0].length)
        return null;

    return { row : row, col : col };
}

function displayMap()
{
    stroke(0);

    for(var row = 0; row < map.length; row++)
    {
        var arRow = map[row];

        for(var col = 0; col < arRow.length; col++)
        {
            var x = mapX + col * squareSize;
            var y = mapY + row * squareSize;

            fill( arRow[col] == 0 ? "White" : "Black" );
            rect(x, y, squareSize, squareSize);
        }
    }
}

function displayValues()
{
    fill(0);
    noStroke();
    textAlign(LEFT, CENTER);

    for(var row = 0; row < map.length; row++)
    {
        var arRow = map[row];
        var number = arrayToNumber(arRow);

        var y = valuesY + squareSize / 2 + squareSize * row;

        // Dec
        text(number.toString(), valuesX, y);

        // Hex
        text(number.toString(16).toUpperCase(), valuesX + 30, y);

        // Bin
        text(toBinary(number, 8), valuesX + 60, y);
    }
}

// Returns an array of 8 numbers representing the defined character
function getChar()
{
    var ar = [];

    for(var row = 0; row < map.length; row++)
    {
        var number = arrayToNumber(map[row]);
        ar.push(number);
    }

    return ar;
}

function displayInstructions()
{
    var delta = 20;

    text("Use mouse to edit", instructionsX, instructionsY + delta * 0);
    text("↑ - Rotate up", instructionsX, instructionsY + delta * 1);
    text("↓ - Rotate down", instructionsX, instructionsY + delta * 2);
    text("← - Rotate left", instructionsX, instructionsY + delta * 3);
    text("→ - Rotate right", instructionsX, instructionsY + delta * 4);
    text("H - Mirror horizontal", instructionsX, instructionsY + delta * 5);
    text("V - Mirror vertical", instructionsX, instructionsY + delta * 6);
    text("R - Rotate right", instructionsX, instructionsY + delta * 7);
    text("L - Rotate left", instructionsX, instructionsY + delta * 8);
    text("X - Add to list", instructionsX, instructionsY + delta * 9);
}

function toBinary(number, n)
{
    var txt = number.toString(2);

    while(txt.length < n)
        txt = "0" + txt;

    return txt;
}

function arrayToNumber(ar)
{
    var n = ar.length;
    var number = 0;

    for(var i = 0; i < n; i++)
    {
        number += ar[i] * pow(2, n - i - 1)
    }

    return number;
}

function displayChar(ar, x, y)
{
    noStroke();

    for(var i = 0; i < ar.length; i++)
    {
        var t = toBinary(ar[i], 8);

        for(var j = 0; j < t.length; j++)
        {
            fill( t[j] == "0" ? color0 : color1 );
            rect(x + j * charDot, y + i * charDot, charDot, charDot);
        }
    }
}

function displayChars()
{
    var x = 10;
    var y = charsY;

    for(var i = 0; i < chars.length; i++)    
    {
        var ar = chars[i];
        displayChar(ar, x, y);
        x += charDot * ar.length * 2;

        if ( x > width - 30 ) 
        {
            y += 30;
            x = 10;
        }
    }

}

function shiftMapLeft()
{
    for(var ar of map)
    {
        shiftLeft(ar);
    }
}

function shiftMapRight()
{
    for(var ar of map)
    {
        shiftRight(ar);
    }
}

function shiftLeft(ar)
{
    var el = ar.shift();
    ar.push(el);
}

function shiftRight(ar)
{
    var el = ar.pop();
    ar.unshift(el);
}

function mirrorMap()
{
    for(var ar of map)
    {
        mirrorArray(ar);
    }
}

function mirrorArray(ar)
{
    var n = ar.length;
    for(var i = 0; i < n / 2; i++)
    {
        var el = ar[n - 1 - i];
        ar[n - 1 - i] = ar[i];
        ar[i] = el;
    }
}

function rotateLeft()
{
    var n = map.length;
    var map2 = newMap(n);


    for(var i = 0; i < n; i++)
    {
        for(var j = 0; j < n; j++)
        {
            map2[n - 1 - j][i] = map[i][j];
        }
    }

    map = map2;
}

function rotateRight()
{
    var n = map.length;
    var map2 = newMap(n);

    for(var i = 0; i < n; i++)
    {
        for(var j = 0; j < n; j++)
        {
            map2[j][n - 1 - i] = map[i][j];
        }
    }

    map = map2;
}

function newMap(n)
{
    var ar = new Array(n);
    for(var i = 0; i < n; i++)
    {
        ar[i] = new Array(n);
    }

    return ar;
}

Feedback

If you liked the article, please follow @codeguppy on Twitter and / or visit codeguppy.com for more tutorials and projects.

Also, if you want to extend this article with detailed instructions explaing how to build the program step by step, please leave feedback in the comments.

Top comments (0)