javascriptphpgrapesjs

GrapesJs and PHP - store and load data to show in editor and as HTML page as well


I am using GrapesJS to build a simple webpage.

I included the script in the following way inside head part :

<script type="text/javascript" src="js/jquery-1.10.2.js"></script>
<link rel="stylesheet" type="text/css" href="grapesjs-dev/dist/css/grapes.min.css">
<script type="text/javascript" src="grapesjs-dev/dist/grapes.min.js"></script>

HTML:

<div id="gjs" style="height:0px; overflow:hidden;">
</div>

Javascript :

<script>      
       var editor = grapesjs.init({
        showOffsets: 1,
        noticeOnUnload: 0,
        container: '#gjs',

        fromElement: true,

        height: '100%',
        fromElement: true,
        storageManager: { autoload: 0 },


   assetManager: {

     assets: [
     'http://placehold.it/350x250/78c5d6/fff/image1.jpg',
     // Pass an object with your properties
     {
       type: 'image',
       src: 'http://placehold.it/350x250/459ba8/fff/image2.jpg',
       height: 350,
       width: 250
     },
     {
       // As the 'image' is the base type of assets, omitting it will
       // be set as `image` by default
       src: 'http://placehold.it/350x250/79c267/fff/image3.jpg',
       height: 350,
       width: 250
     },
    ],

  },


   storageManager: {
    type: 'remote',
    stepsBeforeSave: 1,
    autosave: true,         // Store data automatically
    autoload: true,
    urlStore: 'save_now.php',
    urlLoad: 'load_now.php',
    // ContentType: 'application/json',
    // For custom parameters/headers on requests
    //params: { _some_token: '....' },
    contentTypeJson: true,
      storeComponents: true,
    storeStyles: true,
    storeHtml: true,
    storeCss: true,
     headers: {
    'Content-Type': 'application/json'
    },
    json_encode:{
    "gjs-html": [],
    "gjs-css": [],
    }
  //headers: { Authorization: 'Basic ...' },
  }


      });

 window.editor= editor;




var blockManager = editor.BlockManager;

// 'my-first-block' is the ID of the block
blockManager.add('my-first-block', {
  label: 'Simple block',
  content: '<div class="my-block">This is a simple block</div>',
});


 </script>

So I get in the blocks panel a block namely Simple block which I can drag and drop on the editor. When ever any change is made then the autosave is trigerred with an ajax call to save.php file. Inside save.php, I have:

$content_found="";
$content_found= file_get_contents('php://input');

mysqli_real_escape_string($conn, $content_found);
echo " content found = ".$content_found;
$sql = "INSERT INTO `grapes_content` (`content_found`)
VALUES ('".$content_found."')";

But in Chrome developer tool network tab, I can see :

enter image description here

It is not clear what payload variables I should save in database and how . I used $content_found= file_get_contents('php://input'); to get the full content instead.

After saving it into DB, on page refresh, I load the page with load_now.php. Inside load_now.php, I have :

$content_found="";
$sql = "SELECT * FROM  `grapes_content`";
$result=$conn->query($sql);
$content_found="";
while($row=mysqli_fetch_assoc($result)){

    $content_found=$row['content_found'];

}

echo $content_found;

But the editor shows no saved data.

I am pretty sure that the way I save and retrieve data is not correct. So points are:

Q1) What things should I save in database ? And how can I get the variables or data from ajax payload or in any other way ?

Q2) How can I show the saved data into the editor after page reload ?

In the editor, I see a preview option with an eye image that can show me the HTML page without any editor.

Q3) After saving data into database, how can I show the data simply just as a HTML page and not inside any editor ?


Solution

  • I found the solution : I used the following table :

    CREATE TABLE IF NOT EXISTS `pages` (
       `id` int(20) NOT NULL AUTO_INCREMENT,
       `assets` TEXT  NULL ,
       `components` TEXT NULL,
       `css` TEXT NULL ,
       `html` TEXT  NULL ,
       `styles` TEXT NULL ,
        PRIMARY KEY (`id`)
    ) ENGINE=InnoDB DEFAULT CHARSET=latin1 ;
    

    In Javascrpt , the storageManager part was changed in the following way :

    storageManager: {
        type: 'remote',
        stepsBeforeSave: 1,
        autosave: true,         // Store data automatically
        autoload: true,
        urlStore: 'save_now.php',
        urlLoad: 'load_now.php',
        params: { page_id: 11111 },
        contentTypeJson: true,
          storeComponents: true,
        storeStyles: true,
        storeHtml: true,
        storeCss: true,
         headers: {
        'Content-Type': 'application/json'
        }
    

    In save_now.php, I have:

    header("Content-Type:application/json");
    include('db.php');
    $data = json_decode(file_get_contents("php://input"),true);
    
    $assets = $data['gjs-assets'];
    $assets=json_encode($assets);
    $components = $data['gjs-components'];
    $components=json_encode($components);
    $css = $data['gjs-css'];
    $css = json_encode($css);
    $html = $data['gjs-html'];
    $html= json_encode($html);
    $styles = $data['gjs-styles'];
    $styles = json_encode($styles);
    
    //$page_id = $data['page_id']; **I did not use it in my code here.. but you might need it. See the last part of this answer.**
    
     $result = mysqli_query(
     $con,
     "INSERT INTO `pages` (assets, components, css, html, styles)
      VALUES ($assets, $components, $css, $html, $styles)") or die(mysqli_error($con));
    
    echo "success";
    

    In load_now.php, I have:

    header("Content-Type:application/json");
    
      $result = mysqli_query(
      $con,
      "SELECT * FROM `pages` ");
      if(mysqli_num_rows($result)>0){
        $row = mysqli_fetch_array($result);
        $assets = $row['assets'];
        $components = $row['components'];
        $css = $row['css'];
        $html = $row['html'];
        $styles = $row['styles'];
        $id=$row['id']; 
        response($id, $assets, $components, $css, $html, $styles);
        mysqli_close($con);
      }else{
        response(NULL, NULL,NULL,NULL, 200,"No Record Found");
      }
    function response($id, $assets, $components, $css, $html, $styles){
     $response['id'] = $id;
     $response['gjs-assets'] = $assets;
     $response['gjs-components'] = $components;
     $response['gjs-css'] = $css;
     $response['gjs-html'] = $html;
     $response['gjs-styles'] = $styles;
    
     $json_response = json_encode($response);
     echo $json_response;
    }
    

    Notice that in storageManager, I used params: { page_id: 11111 },. You can use this page_id in a column value to identify the certain row for your page as you can have multiple pages as well. I wrote the very basic functionalities i.e: I only showed the insert part without using the update query in case the expected row for the certain page already exists in DB. As to the loading part, you should use the page_id to fetch the certain row. This part I also did not show leaving it to your normal logical sense.

    N.B.: I came to the solution from q GitHub question here .

    EDIT: As to Q3) , the solution is:

    Check out grapesjs-plugin-export for a client side solution. Since the html and css are saved into database, you can also do the same thing the plugin does using a server side script. You can use something like file_put_contents.