How to do it...

  1. Install bower using npm:
      npm install –g bower
  1. Verify the bower installation:
      which bower && bower -v
Note that both commands should return some values. On Windows, you need Cygwin or Git Bash to run the preceding commands.
  1. Install Bootstrap 4 alpha 6:
      bower install bootstrap#v4.0.0-alpha.6
  1. Install Harp globally:
      npm install -g harp
  1. Make two new folders in the root, and add these files to them as per the following commands:
      mkdir app grunt && 
touch app/index.html grunt/package.json grunt/Gruntfile.js main.scss
  1. Add the following code to main.scss:
      $white: #ffffff;

@import "./bower_components/bootstrap/scss/bootstrap.scss";
  1. CD into the grunt folder:
      cd grunt
  1. Open the package.json file and add the following code to it:
      {
"name": "customgrunt",
"version": "",
"devDependencies": {
"grunt": "^1.0.1"
}
}

Then, save the file.

  1. In workspace/grunt, install grunt-contrib-sass:
      npm install grunt-contrib-sass --save-dev
  1. Also, install grunt CLI (Command Line Interface):
      npm install grunt-cli -g
  1. Add the following code to your Gruntfile.js:
      'use strict';
// Load Grunt
module.exports = function (grunt) {
grunt.initConfig({
pkg: grunt.file.readJSON('package.json'),

// Tasks
sass: { // Begin Sass Plugin
dist: {
options: {
sourcemap: 'inline'
},
files: [{
expand: true,
cwd: '../',
src: ['main.scss'],
dest: '../app/css',
ext: '.css'
}]
}
},
});
// Load Grunt plugins
grunt.loadNpmTasks('grunt-contrib-sass');

// Register Grunt tasks
grunt.registerTask('default', ['sass']);

};

Then, save all files.

  1. The installation of grunt-cli does not install the grunt task runner. The CLI will only run Grunt if it is installed as a project dependency. We do it by running this command:
      npm install grunt --save-dev

The --save-dev flag lists grunt devDependency to your package.json.

  1. In workspace/grunt, run the grunt command with verbose logging:
      grunt -v
Note that this is a very important step, as it will compile Bootstrap's SCSS into a new .css file in app/css/main.css, as instructed in the sass grunt task in Gruntfile.js.
  1. Install the grunt-contrib-copy plugin:
      npm install grunt-contrib-copy --save-dev
  1. Verify that the package.json file was updated with the new grunt plugin devDependencies; save all files.
  2. Update your Gruntfile.js, to include the copy task:
      'use strict';
// Load Grunt
module.exports = function(grunt) {
grunt.initConfig({
pkg: grunt.file.readJSON('package.json'),

// Tasks
sass: { // Begin Sass Plugin
dist: {
options: {
sourcemap: 'inline'
},
files: [{
expand: true,
cwd: '../',
src: ['main.scss'],
dest: '../app/css',
ext: '.css'
}]
}
},
copy: {
main: {
files: [{
cwd: '../bower_components/bootstrap/dist/js/',
expand: true,
src: ['**'],
dest: '../app/js/'
}]
}
},
});
// Load Grunt plugins
grunt.loadNpmTasks('grunt-contrib-sass');
grunt.loadNpmTasks('grunt-contrib-copy');


// Register Grunt tasks
grunt.registerTask('default', ['sass', 'copy']);

};

Save the file.

  1. Run grunt without verbose logging:
      grunt

This time, Bootstrap's JS files, bootstrap.js and bootstrap.min.js, will be copied into the app/js/ folder, as described in the new 'copy' task in our Gruntfile.js.

  1. Now, make sure that Harp is installed by running the version check:
      harp version
  1. Go into the app folder and add the following code to the index.html file:
      <!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-
scale=1, shrink-to-fit=no">
<meta name="description" content="">
<meta name="author" content="">
<link rel="icon" href="../../favicon.ico">

<title>Bootstrap 4 Grid Recipe</title>

<!-- Bootstrap core CSS -->
<link href="./css/main.css" rel="stylesheet" type="text/css" >

<!-- Custom styles for this template -->
<link href="starter-template.css" rel="stylesheet">
</head>

<body>

<nav class="navbar navbar-toggleable-md navbar-light bg-faded">
<button class="navbar-toggler navbar-toggler-right"
type="button" data-toggle="collapse" data-
target="#navbarsExampleDefault" aria-
controls="navbarsExampleDefault" aria-expanded="false" aria-
label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<a class="navbar-brand" href="#">Bootstrap 4 Grid Recipe</a>

<div class="collapse navbar-collapse"
id="navbarsExampleDefault">
<ul class="navbar-nav mr-auto">
<li class="nav-item active">
<a class="nav-link" href="#">Home <span class="sr-only">
(current)</span></a>
</li>
<li class="nav-item">
<a class="nav-link" href="#">Link</a>
</li>
<li class="nav-item">
<a class="nav-link disabled" href="#">Disabled</a>
</li>
<li class="nav-item dropdown">
<a class="nav-link dropdown-toggle" href="http://example.com"
id="dropdown01" data-toggle="dropdown" aria-haspopup="true"
aria-expanded="false">Dropdown</a>
<div class="dropdown-menu" aria-labelledby="dropdown01">
<a class="dropdown-item" href="#">Action</a>
<a class="dropdown-item" href="#">Another action</a>
<a class="dropdown-item" href="#">Something else here</a>
</div>
</li>
</ul>
<form class="form-inline my-2 my-lg-0">
<input class="form-control mr-sm-2" type="text"
placeholder="Search">
<button class="btn btn-outline-success my-2 my-sm-0"
type="submit">Search</button>
</form>
</div>
</nav>

<div class="container">

<div class="starter-template">
<h1>Bootstrap 4 Grid Recipe</h1>
<p class="lead">Use this document as a way to quickly start any
new project.<br> All you get is this text and a mostly barebones
HTML document.</p>
</div>

</div><!-- /.container -->


<!-- Bootstrap core JavaScript
================================================== -->
<!-- Placed at the end of the document so the pages load faster
-->
<script src="https://code.jquery.com/jquery-3.1.1.slim.min.js"
integrity="sha384-A7FZj7v+d/sdmMqp/nOQwliLvUsJfDHW+k9Omg/a/
EheAdgtzNs3hpfag6Ed950n"
crossorigin="anonymous"></script>
<script>window.jQuery || document.write('<script
src="../../assets/js/vendor/jquery.min.js"><\/script>')</script>
<script
src="https://cdnjs.cloudflare.com/ajax/libs/tether/1.4.0/js/
tether.min.js"integrity="sha384-DztdAPBWPRXSA/3eYEEUWrWCy7G5
KFbe8fFjk5JAIxUYHKkDx6Qin1DkWx51bBrb"
crossorigin="anonymous"></script>
<script src="js/bootstrap.min.js"></script>
</body>
</html>

Then, save the file.

  1. Navigate to the workspace/app folder, and run the following code:
      harp compile
Running this command for the first time will generate a www folder inside the app folder. The www folder will have all the necessary files to run and preview the results in the browser of your choice.
  1. Open the compiled www folder, and the index.html file inside. Click on the round, green Run button at the top of the Cloud9 workspace. A new tab will open next to the existing bash tab(s); there will be the Starting Apache httpd, serving text inside it. The text is followed by a clickable link, in the following format: https://<project-name>-<c9username>.c9users.io/app/www/index.html.
Note that this way of running our Harp server is not the suggested one. This is just an intermediary step to enable us to make sure that our setup at this point is running as smoothly as possible. In the next steps, we will look into the preferred approach of serving harp-generated files.
  1. Click on the provided link and a new tab will open in your browser, with the following web page:
  1. Go to main.scss, and change the value of the $white variable:
      $white: #bada55;

Save all files.

To make sure that your main.scss file compiles properly, you could use the grep and less tools available in the Bash console. Once you have the main.scss file saved as per the previous step, all you have to do is run  sass main.scss | grep bada55 | less.  This command compiles the main.scss code, and then, instead of outputting it right into the console, it pipes it to grep. It is told to look for the  bada55 string and, the result of this grep command is output to the less viewer. The less viewer lists all the instances of the  bada55 string in the compiled CSS code, which proves that our main.scss file indeed copies the bootstrap/scss files and alters them with the new value of the $white variable.
  1. Now, navigate to the workspace/grunt folder, and run the grunt command:
      grunt

This will recompile the .scss file and copy the .js file, from the bower_components/js/dist folder, into the app folder.

  1. Go to the app folder, and run harp compile again. Open the generated www/index.html and run it with the green Run button; click on the link and verify that the colors have changed to green, based on the changes made to the main.scss file.
  2. Now is the time to override more Bootstrap variables in the main.scss file. Above the @include, replace the override of the $white variable with the following code:
      $enable-rounded: false;
$white: #ccc;
$border-width: 5px;
$font-size-base: 3rem;

Save the file. Go back to the grunt folder, and run the grunt command. Go back to the app folder, and run the harp compile command.

Obviously, it is not sustainable to keep manually switching between the grunt folder and the app folder, and to keep saving, recompiling, and rerunning the files. In the next steps, we will automate this process.
  1. Now that our setup is working, it is time to organize the Cloud9 workspace in such a way that it makes it easier for us to run the code. This will involve several simple actions. The following screenshot will give us an overview of the actions to do:
Organizing Cloud9 Workspace

The first thing to do (marked by number 1) is click on the little icon at the top-left corner, which is the pane submenu. Next, click on the split four ways button at the bottom of the menu (marked by number 2). The third action involves dragging and dropping several open tabs, namely, index.html to the top-left corner, _custom.scss to the top-right corner, the Bash console tab that will be used to run the harp server to the lower-left corner, and finally, the bash tab that will run the grunt watch command to the lower-right corner. 

  1. CD into the grunt folder, and install the grunt-contrib-watch grunt plugin:
      npm install grunt-contrib-watch --save-dev
  1. Update the Gruntfile.js file with the following code:
      'use strict';
// Load Grunt
module.exports = function(grunt) {
grunt.initConfig({
pkg: grunt.file.readJSON('package.json'),

// Tasks
sass: { // Begin Sass Plugin
dist: {
options: {
sourcemap: 'inline'
},
files: [{
expand: true,
cwd: '../',
src: ['main.scss'],
dest: '../app/css',
ext: '.css'
}]
}
},
copy: {
main: {
files: [
{
cwd: '../bower_components/bootstrap/dist/js/',
expand: true,
src: ['**'],
dest: '../app/js/'
}]
}
},
watch: {
files:
['../main.scss','../bower_components/bootstrap/dist/js/*.js'],
tasks: ['sass', 'copy']
},
});
// Load Grunt plugins
grunt.loadNpmTasks('grunt-contrib-sass');
grunt.loadNpmTasks('grunt-contrib-copy');
grunt.loadNpmTasks('grunt-contrib-watch');

// Register Grunt tasks
grunt.registerTask('default', ['watch']);

};
  1. Everything is now set up to track all the changes. In the bash tab dedicated for running grunt, type the grunt -v command. Due to the way it is set up, it will watch for changes, and report its operations with verbose logging. In the lower-left bash tab, we will start running the harp server, with the following command:
      harp server ./ --port 8080
  1. Finally, to view your site, open a new browser tab and type the https://<project-name>-<username>.c9users.io address. This should serve the updated website. To change the look of the website, change the variable values in _custom.scss and refresh the page. With grunt watching in the lower-right console tab, and harp serving in the lower-left console tab, you only need to save changes you make to _custom.scss, and refresh the browser tab that points to your app to view the updates as they are saved.
It is possible that your page will not serve as expected. That has to do with what folder you are in when you run the harp server ./ --port 8080 command. If you are in the workspace folder, then you can see your running site at the  https://<project-name>-<username>.c9users.io address. However, if you are in the workspace/app folder, you can see your running site at the  https://<project-name>-<username>.c9users.io/app/ address. To even further improve this workflow, we can make our browser live-reload the harp site we just created. A quick solution is to use the  Auto Refresh Chrome extension, from  http://bit.ly/1CvA52G.