Hello and welcome to our community! Is this your first visit?
Register
Enjoy an ad free experience by logging in. Not a member yet? Register.
Results 1 to 6 of 6
  1. #1
    New to the CF scene
    Join Date
    Jul 2012
    Posts
    4
    Thanks
    1
    Thanked 0 Times in 0 Posts

    Saving a Multidimensional Array into an XML using JavaScript

    So I have a multidimensional array that looks like this:

    Code:
        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:

    Code:
       <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:

    Code:
      <Level><map> ARRAY HERE </map></Level>
    How would i be able to add the map variable from the Javascript into a <map></map> node within the XML file? I would need to be able to return the <map> number aswel. For example if i already had 4 <level><map>[[0,0],[0,0]]</map></level> then i would need to add it to the end of those and return the number 5 to show thats what the level number is.

    Thanks

  • #2
    Supreme Master coder! Old Pedant's Avatar
    Join Date
    Feb 2009
    Posts
    27,650
    Thanks
    80
    Thanked 4,638 Times in 4,600 Posts
    JavaScript has no ability to WRITE to files.

    Not 100% true; MSIE can write to files ON THE USER'S COMPUTER (only), but only if the user marks your site as being a "safe" site and only if he/she agrees to let your page run an "unsafe for scripting" ActiveX control.

    But even here, the JS code is only writing to the USER's computer. No matter what, JavaScript has no ability to write to files ON THE SERVER, which is surely what you would want to do for your map game.

    You will need server-side code (PHP/ASP/JSP) to do this. And then the JavaScript code *can* send the data to the server and let the server do the file writing.
    An optimist sees the glass as half full.
    A pessimist sees the glass as half empty.
    A realist drinks it no matter how much there is.

  • #3
    Senior Coder
    Join Date
    Sep 2010
    Posts
    2,451
    Thanks
    17
    Thanked 275 Times in 275 Posts
    Go with whatever server side you're using, if at all possible. It will have no problem writing to the file. You have to open the file for writing, you can make a template file for the heading, copy it to the folder you want it in, with rename, write what you want to it, and append the closing tags. Done.

  • #4
    Supreme Master coder! Old Pedant's Avatar
    Join Date
    Feb 2009
    Posts
    27,650
    Thanks
    80
    Thanked 4,638 Times in 4,600 Posts
    Actually, if the server side system you are using understands XML well, you can load the XML file into an XML document, *append* the new <Level> with its subnodes, and then write the XML document back out as XML text. Trivial to do in ASP.NET or JSP. Not very hard to do in classic ASP. You then don't have to parse the XML in the file by hand to try and find the closing tag.

    (I don't use PHP, so I don't know where this is easy or hard there. If it's not easy--assuming you have the right libraries--I would be surprised.)
    An optimist sees the glass as half full.
    A pessimist sees the glass as half empty.
    A realist drinks it no matter how much there is.

  • #5
    Regular Coder
    Join Date
    Apr 2005
    Location
    Texas
    Posts
    448
    Thanks
    24
    Thanked 63 Times in 63 Posts
    Quote Originally Posted by Old Pedant View Post
    JavaScript has no ability to WRITE to files.
    But since this is such a fun exercise, let us assume a few things here:
    - The OP is another victim of the ambiguous JavaScript != JScript
    - The OP has control over the server and is able to flag server directories as writable
    - The deployment server supports classic ASP (and thus JScript (which is *remarkably* similar to JavaScript))
    - Only 1 user at a time can access the "savemapdata" page (it is not a data base, although i leave that as and exercise for the better qualified)


    inc/json.asp
    Code:
    <%
    	// thanks of course to Old Pedant
        var Clone = function (a) {if(null==a||"object"!=typeof a)return [];var c=a.constructor(),b;for(b in a)a.hasOwnProperty(b)&&(c[b]=a[b]);return c};
    /*
    here is where you would include JSON i believe the version i use can be found here
    https://github.com/douglascrockford/JSON-js/blob/master/json2.js
    
        See http://www.JSON.org/js.html
    
        This code should be minified before deployment.
        See http://javascript.crockford.com/jsmin.html
    */
    %>
    inc/maps.asp
    Code:
    <%
    var maps = [
        {
            level:1,
            terrain:[
                [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]
            ]
        }
    ];
    %>
    savemapdata.asp
    Code:
    <%@LANGUAGE="JavaScript"%><!--Dont let the name fool ya...-->
    
      <!--#include file="inc/json.asp"-->
      <!--#include file="inc/maps.asp"-->
    <%
    	//retrieve array in object notation
        mapFromURL = new String(Request("map")).replace(/[^\[\]\,0-9]/g,'');
        levelFromURL = new String(Request("level")).replace(/[^0-9]/g,'');
    
        var FSO = new ActiveXObject("Scripting.FileSystemObject")
        var ForReading = 1, ForWriting = 2, ForAppending = 3;
    
        maps.push(
            {
                map:JSON.parse(mapFromURL) ,
                level:levelFromURL
            }
        );
        var MapsFileObj = FSO.OpenTextFile('inc/maps.asp' , ForWriting);
    
        MapsFileObj.WriteLine('<' + '%')
        MapsFileObj.WriteLine('maps = ' + JSON.stringify(maps));
        MapsFileObj.WriteLine('%' + '>');
        MapsFileObj.Close();
    %>
    maps.asp (outputs xml data...)
    Code:
    <%@LANGUAGE="JavaScript"%>
      <!--#include file="inc/maps.asp"-->
      <!--#include file="inc/json.asp"-->
    
    <%
        var FilteredMaps = [];
        var Levels = '' , write =  function (d){Levels+=d+'\n'};
    
        for (var i=0 , ii = maps.length ; i<ii ; ++i) {
            FilteredMaps[maps[i].level] = Clone(maps[i]);		// read only the most recent version of the map, conversly one might only save one version of each level... but let's say that we intentionally allowed the level versions to co-exist for historical reasons
        }
    
        for (var i=0 , ii = FilteredMaps.length ; i<ii ; ++i) {
    
            if (FilteredMaps[i].terrain===undefined)		// or some other hiddeous malformation...
                continue;
    
            write('    <Level> <!-- Level ' + i + '  -->');		// Automation is no excuse for sloppy formatting ;)
            write('      <map level="' + i + '">');
            write('        '  + JSON.stringify(FilteredMaps[i].terrain));
            write('      </map>');
            write('    </Level>');
       }
    %>
      <TileMaps>
    <%=Levels%>
      </TileMaps>
    now all you need is a form to populate the URL variables (or you could link to the page "savemapdata.asp?level="+JSON.stringify(levelNum)+"map="+JSON.stringify(mapData)" from your javascript)

    At a second glance I suppose the savemapdata.asp and maps.asp could be combined to produce an actual .xml as requested...

    savemapdata.asp
    Code:
    <%@LANGUAGE="JavaScript"%>
    
      <!--#include file="inc/json.asp"-->
      <!--#include file="inc/maps.asp"-->
    <%
    	//retrieve array in object notation
        mapFromURL = new String(Request("map")).replace(/[^\[\]\,0-9]/g,'');
        levelFromURL = new String(Request("level")).replace(/[^0-9]/g,'');
    
        var FSO = new ActiveXObject("Scripting.FileSystemObject")
        var ForReading = 1, ForWriting = 2, ForAppending = 3;
    
        maps.push(
            {
                map:JSON.parse(mapFromURL) ,
                level:levelFromURL
            }
        );
        var MapsDataFileObj = FSO.OpenTextFile('inc/maps.asp' , ForWriting);
    
        MapsDataFileObj.WriteLine('<' + '%')
        MapsDataFileObj.WriteLine('maps = ' + JSON.stringify(maps));
        MapsDataFileObj.WriteLine('%' + '>');
        MapsDataFileObj.Close();
    
    
        var FilteredMaps = [];
        var Levels = '' , write =  function (d){Levels+=d+'\n'};
    
        for (var i=0 , ii = maps.length ; i<ii ; ++i) {
            FilteredMaps[maps[i].level] = Clone(maps[i]);		// read only the most recent version of the map, conversly one might only save one version of each level... but let's say that we intentionally allowed the level versions to co-exist for historical reasons
        }
    
        for (var i=0 , ii = FilteredMaps.length ; i<ii ; ++i) {
    
            if (FilteredMaps[i].terrain===undefined)		// or some other hiddeous malformation...
                continue;
    
            write('    <Level> <!-- Level ' + i + '  -->');		// Automation is no excuse for sloppy formatting ;)
            write('      <map level="' + i + '">');
            write('        '  + JSON.stringify(FilteredMaps[i].terrain));
            write('      </map>');
            write('    </Level>');
       }
    
        var MapsXMLFileObj = FSO.OpenTextFile('inc/maps.xml' , ForWriting);
        MapsXMLFileObj.WriteLine('  <TileMaps>');
        MapsXMLFileObj.WriteLine(Levels);
        MapsXMLFileObj.WriteLine('  <\/TileMaps>');
        MapsXMLFileObj.Close();
    %>
    Gee, that was fun
    anyway, my batteries are dying, I hope I didn't miss anything. I tested all those pages individually (except the last) so they should work in concert.
    Allwisend bin ich nicht, doch viel ist mir bewursst
    -Goethe

  • #6
    New to the CF scene
    Join Date
    Aug 2012
    Posts
    9
    Thanks
    0
    Thanked 0 Times in 0 Posts
    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):


    PHP Code:
    <?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');
    }
    ?>



    <!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>

    Best Regards,
    Navin Patel
    Last edited by VIPStephan; 08-09-2012 at 03:04 PM. Reason: added code BB tags and removed fake signature


  •  

    Tags for this Thread

    Posting Permissions

    • You may not post new threads
    • You may not post replies
    • You may not post attachments
    • You may not edit your posts
    •