Reputation: 938
So I have a multidimensional array that looks like this:
var map = [[0, 0, 0, 0, 0, 0, 0],
[0, 3, 0, 0, 2, 0, 0],
[0, 0, 0, 0, 4, 0, 4],
[0, 0, 0, 0, 5, 0, 5],
[0, 0, 0, 0, 0, 0, 1],
[0, 0, 2, 5, 0, 0, 0],
[0, 0, 0, 2, 0, 0, 0],
[0, 4, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0]];
And i would like to save it into my XML file.
My XML file looks like the following:
<TileMaps>
<Level> <!-- Level 1 -->
<map>[[1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
[1, 3, 2, 4, 0, 0, 0, 0, 0, 1],
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1]]</map>
</Level>
<Level> <!-- Level 2 -->
<map>[[1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
[1, 3, 2, 4, 0, 0, 0, 0, 0, 1],
[1, 0, 2, 4, 0, 0, 0, 0, 0, 1],
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1]]</map>
</Level>
</TileMaps>
So when i add the array i would like it to be placed within the XML file within the:
<Level><map> ARRAY HERE </map></Level>
Thanks
Upvotes: 1
Views: 7266
Reputation: 5687
Here's my solution, it uses PHP in the backend, and JSON, so you'll want to modify your server side code as appropriate (if you wrap the JSON in XML, you'll also need to adjust the syntax in the AJAX pieces)...oh, depending on your target user, you'll want to add some validation to the inputs (both client and server side):
PAGE:
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>Maker</title>
<style>
#can{width:300px; height:90px; display:block; background-color:#666;}
</style>
</head>
<body>
Which level do you want:<input type="text" id="level_request" value="level1"/><button onclick="init()">Go</button>
</body>
<script>
function saveLevel(){
ajax=((window.XMLHttpRequest)?new XMLHttpRequest():new ActiveXObject("Microsoft.XMLHTTP"));
name=document.getElementById('level_name').value; //this grabs the content in the Level Name field
if(name.length>0){
ajax.onreadystatechange=function()
{
if (ajax.readyState==4&&ajax.status==200){
alert('Level saved');
}
}
params='level='+name+'&map='+JSON.stringify(map); //this constructs the message to send, consisting of the name and map
ajax.open("POST","levels.php",true); //this is the file you will be POSTing a message to
ajax.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
ajax.setRequestHeader("Content-length", params.length);
ajax.setRequestHeader("Connection", "close");
ajax.send(params);
}
}
var blocksize=30;
var map=[[1,1,1,1,1,1,1,1,1,1],[1,3,0,0,0,0,2,4,0,1],[1,1,1,1,1,1,1,1,1,1]];
var can;
var ctx;
function init(){
ajax=((window.XMLHttpRequest)?new XMLHttpRequest():new ActiveXObject("Microsoft.XMLHTTP"));
ajax.onreadystatechange=function()
{
if (ajax.readyState==4){ //this test whether the request is complete
l=document.getElementById('level_request').value;
document.body.innerHTML='Level Name:<input type="text" id="level_name" /><br/><canvas id="can" width="300" height="90"></canvas><br/><button onclick="saveLevel()">Save</button>'; //this replaces the initial form
can=document.getElementById('can')
if(can){ctx=can.getContext('2d');}
if(ajax.status==200){ //this test whether it was successful
m=JSON.parse(ajax.responseText);//this overwrites the existing map with the received data
console.log(m);
map=m.tilemaps[l];
for(y=0;y<map.length;y++){
for(x=0;x<map[y].length;x++){
draw(y,x);
}
}
can.addEventListener('click',builder);
}
else{ //this is what we do if the request is done and it was a failure
for(y=0;y<map.length;y++){
for(x=0;x<map[y].length;x++){
draw(y,x);
}
}
can.addEventListener('click',builder);
alert('Something went wrong, loading default level');
}
}
}
ajax.open("GET","levels.php?level="+document.getElementById('level_request').value,true);
ajax.send();
}
function builder(e){
if (e == null) {e = window.event;}
x = e.clientX; //where the click was
y = e.clientY;
offsetX = ExtractNumber(can.offsetLeft);//where the canvas is
offsetY = ExtractNumber(can.offsetTop);
x_grid=Math.floor((x-offsetX)/blocksize); //which block in the canvas was clicked
y_grid=Math.floor((y-offsetY)/blocksize);
map[y_grid][x_grid]++;
if(map[y_grid][x_grid]>4){map[y_grid][x_grid]=0;}
draw(y_grid,x_grid);
}
function draw(y,x){
kind=map[y][x];
switch(kind){
case 0:
ctx.drawImage(floorimg,x*blocksize,y*blocksize);
break;
case 1:
ctx.drawImage(wallimg,x*blocksize,y*blocksize);
break;
case 2:
ctx.drawImage(blockimg,x*blocksize,y*blocksize);
break;
case 3:
ctx.drawImage(playerimg,x*blocksize,y*blocksize);
break;
case 4:
ctx.drawImage(goalimg,x*blocksize,y*blocksize);
break;
}
}
function ExtractNumber(value){
var n = parseInt(value);
return n == null || isNaN(n) ? 0 : n;
}
var floorimg=new Image();
floorimg.src='';
var wallimg=new Image();
wallimg.src='';
var blockimg=new Image();
blockimg.src='';
var playerimg=new Image();
playerimg.src='';
var goalimg=new Image();
goalimg.src='';
</script>
</html>
levels.php:
<?php
if($_REQUEST['level']&&$_REQUEST['map']){
$file=file_get_contents('levels.txt'); //get the existing content
$json=json_decode($file); //convert it from string to object
$tilemaps=$json->tilemaps;
$tilemaps->$_REQUEST['level']=json_decode($_REQUEST['map']);
$o['tilemaps']=$tilemaps;
$str=json_encode($o);
$pos=fopen('levels.txt','w');
fwrite($pos,$str);
fclose($pos);
}
else{
header('content-type:application/json');
echo file_get_contents('levels.txt');
}
?>
Upvotes: 0
Reputation: 25135
Using JSON is recommended for this. Anyway, giving a solution if you want to proceed with XML.
Array to XML
Build XML string using string concatenation. Use Crockford's JSON library for building the array string.
var map = [[0, 0, 0, 0, 0, 0, 0],
[0, 3, 0, 0, 2, 0, 0],
[0, 0, 0, 0, 4, 0, 4],
[0, 0, 0, 0, 5, 0, 5],
[0, 0, 0, 0, 0, 0, 1],
[0, 0, 2, 5, 0, 0, 0],
[0, 0, 0, 2, 0, 0, 0],
[0, 4, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0]];
var xml = '<TileMaps><level><map>';
xml += JSON.stringify(map);
xml += '</map></level></TileMaps>';
alert(xml);
jsfiddle : http://jsfiddle.net/diode/ZfWjp/
Then send it to server side for saving.
XML to Array
Load saved XML from server.
Use jQuery to parse it.
var xml = '<TileMaps><level><map>[[0,0,0,0,0,0,0],[0,3,0,0,2,0,0],[0,0,0,0,4,0,4],[0,0,0,0,5,0,5],[0,0,0,0,0,0,1],[0,0,2,5,0,0,0],[0,0,0,2,0,0,0],[0,4,0,0,0,0,0],[0,0,0,0,0,0,0]]</map></level></TileMaps>';
var map = $.parseJSON($(xml).find("map").text());
alert(map[0]);
alert(map[1]);
jsfiddle : http://jsfiddle.net/kBrCT/1/
note : You have to modify this if there are multiple map nodes in single XML file.
Upvotes: 2
Reputation: 21249
Well if it's just the transformation of the array itself to string you could do this:
var data = [[0, 0, 0, 0, 0, 0, 0],
[0, 3, 0, 0, 2, 0, 0],
[0, 0, 0, 0, 4, 0, 4],
[0, 0, 0, 0, 5, 0, 5],
[0, 0, 0, 0, 0, 0, 1],
[0, 0, 2, 5, 0, 0, 0],
[0, 0, 0, 2, 0, 0, 0],
[0, 4, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0]];
var dataAsString = "[" + data.map(function(item){return "["+item.toString()+"]"}).toString() + "]"
console.log(dataAsString);
Note: map is a recent enough addition to Javascript, it may not work in every browser (c.f. Mozilla Docs)
Upvotes: 0