├── views ├── partials │ ├── footer.ejs │ └── header.ejs ├── comments │ ├── new.ejs │ └── edit.ejs ├── login.ejs ├── register.ejs ├── campgrounds │ ├── edit.ejs │ ├── index.ejs │ ├── new.ejs │ └── show.ejs └── landing.ejs ├── models ├── user.js ├── comment.js └── campground.js ├── package.json ├── README.md ├── public └── assets │ ├── newCampground.css │ ├── showCampground.css │ ├── campgrounds.css │ └── landing.css ├── routes ├── index.js ├── comments.js └── campgrounds.js ├── app.js ├── middleware └── index.js └── seeds.js /views/partials/footer.ejs: -------------------------------------------------------------------------------- 1 |

2 | 3 | 4 | -------------------------------------------------------------------------------- /models/user.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | var passportLocalMongoose=require("passport-local-mongoose"); 3 | 4 | var UserSchema = new mongoose.Schema({ 5 | username: String, 6 | password: String 7 | }) 8 | 9 | UserSchema.plugin(passportLocalMongoose); 10 | 11 | module.exports=mongoose.model("User",UserSchema); -------------------------------------------------------------------------------- /models/comment.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | var commentSchema = mongoose.Schema({ 3 | text: String, 4 | author: { 5 | id: { 6 | type: mongoose.Schema.Types.ObjectId, 7 | ref: "User" 8 | }, 9 | username: String 10 | } 11 | }); 12 | 13 | module.exports=mongoose.model("Comment",commentSchema); -------------------------------------------------------------------------------- /models/campground.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | 3 | var campgroundSchema = new mongoose.Schema({ 4 | name: String, 5 | image: String, 6 | description: String, 7 | author: { 8 | id: { 9 | type: mongoose.Schema.Types.ObjectId, 10 | ref: "User" 11 | }, 12 | username: String 13 | }, 14 | comments: [ 15 | { 16 | type: mongoose.Schema.Types.ObjectId, 17 | ref: "Comment" 18 | } 19 | ] 20 | }); 21 | 22 | module.exports = mongoose.model("Campground",campgroundSchema); 23 | -------------------------------------------------------------------------------- /views/comments/new.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 | 4 |
5 |
6 |

Add New Comment to <%= campground.name %>

7 |
8 |
9 | 10 | 11 |
12 |
13 | 14 |
15 |
16 |
17 |
18 | 19 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /views/comments/edit.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 |
4 |
5 |

Edit Comment

6 |
7 |
8 | 9 | 10 |
11 |
12 | 13 |
14 |
15 |
16 |
17 | 18 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "yelpcamp", 3 | "version": "1.0.0", 4 | "description": "Its a Full Stack Website for finding Camping grounds.", 5 | "main": "app.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1", 8 | "start": "node app.js" 9 | }, 10 | "author": "Arnab Poddar", 11 | "license": "ISC", 12 | "dependencies": { 13 | "body": "^5.1.0", 14 | "body-parser": "^1.19.0", 15 | "connect-flash": "^0.1.1", 16 | "ejs": "^3.0.2", 17 | "express": "^4.17.1", 18 | "express-session": "^1.17.0", 19 | "method-override": "^3.0.0", 20 | "mongodb": "^3.5.5", 21 | "mongoose": "^5.9.7", 22 | "parser": "^0.1.4", 23 | "passport": "^0.4.1", 24 | "passport-local": "^1.0.0", 25 | "passport-local-mongoose": "^6.0.1" 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /views/login.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('./partials/header') %> 2 | 3 |
4 |
5 |

Login Form

6 |
7 |
8 | 9 | 10 |
11 |
12 | 13 | 14 |
15 |
16 | 17 |
18 |
19 |

Don't have an account yet?Create one by clicking here

20 |
21 |
22 | 23 | 24 | 25 | 26 | <%- include ('\partials/footer') %> -------------------------------------------------------------------------------- /views/register.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('./partials/header') %> 2 | 3 |
4 |
5 |

Create an AngelCamp account and gain access to Campkeeper,comments, and much more.

6 |

SignUp Form

7 |
8 |
9 | 10 | 11 |
12 |
13 | 14 | 15 |
16 |
17 | 18 |
19 |
20 |
21 |
22 | 23 | 24 | <%- include ('\partials/footer') %> -------------------------------------------------------------------------------- /views/campgrounds/edit.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 |
4 |
5 |

Edit <%= campground.name %>

6 |
7 |
8 | 9 | 10 |
11 |
12 | 13 | 14 |
15 |
16 | 17 | 18 |
19 |
20 | 21 |
22 |
23 |
24 |
25 | 26 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /views/landing.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | AngelCamp 10 | 11 | 12 | 13 |
14 |

AngelCamp

15 |

Rediscover The Joy Of Camping

16 |
17 | Enter 18 |
19 |
20 | 21 | 28 | 29 | <%- include('partials/footer'); -%> 30 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # AngelCamp 2 | This is a node.js application which let's users browse through various camping areas posted by other users and can post their own campground too.Users can also post comments to any campground. 3 | 4 | Live Demo 5 | 6 | To see the app in action, go to https://mysterious-thicket-79829.herokuapp.com/ 7 | 8 | # Features 9 | 10 | ## Authentication: 11 | 12 | User login with username and password 13 | 14 | Admin sign-up 15 | 16 | 17 | ## Authorization: 18 | 19 | One cannot edit or delete posts and comments created by other users 20 | 21 | 22 | ## Manage campground posts and comments 23 | 24 | Create campground and post comments 25 | 26 | Edit or delete campground by it's owner 27 | 28 | Edit or delete comments by it's writer 29 | 30 | Flash messages responding to users' interaction with the app 31 | 32 | Responsive web design 33 | 34 | 35 | # Built with 36 | 37 | ## Front-end 38 | 39 | ejs 40 | 41 | Bootstrap 42 | 43 | ## Back-end 44 | 45 | node.js 46 | 47 | express 48 | 49 | mongoDB 50 | 51 | mongoose 52 | 53 | passport 54 | 55 | passport-local 56 | 57 | express-session 58 | 59 | method-override 60 | 61 | connect-flash 62 | 63 | ## Deployment 64 | 65 | Heroku 66 | 67 | -------------------------------------------------------------------------------- /views/campgrounds/index.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 | 4 | 5 |
6 | 13 | 14 |
15 | <% campgrounds.forEach(function(camp) 16 | { 17 | %>
18 |
19 | 20 |
21 |
22 |

<%= camp.name %>

23 |
24 |
25 |
26 |
<% 27 | }) %> 28 |
29 | 30 | 31 |
32 | 33 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /views/campgrounds/new.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 |
4 | 28 |
29 | 30 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /public/assets/newCampground.css: -------------------------------------------------------------------------------- 1 | .campdetails 2 | { 3 | margin-top: 100px; 4 | } 5 | #header-login .campdetails 6 | { 7 | margin-top: 40px; 8 | } 9 | .campdetails h3 10 | { 11 | font-family: 'Milonga', cursive; 12 | font-size: 30px; 13 | text-align: center; 14 | margin-bottom: 50px; 15 | } 16 | 17 | form div 18 | { 19 | text-align: center; 20 | margin: 10px; 21 | } 22 | 23 | form div label 24 | { 25 | font-size: 20px; 26 | margin: 20px; 27 | } 28 | 29 | 30 | form div input 31 | { 32 | height: 40px; 33 | width: 400px; 34 | border: 2px solid rgb(92, 61, 3); 35 | border-radius: 20px; 36 | padding: 5px 20px; 37 | font-size: 20px; 38 | outline: none; 39 | font-family: Georgia, 'Times New Roman', Times, serif; 40 | } 41 | 42 | 43 | form div button 44 | { 45 | height: 40px; 46 | width: 100px; 47 | border: 2px solid rgb(92, 61, 3); 48 | border-radius: 20px; 49 | margin: 10px; 50 | outline: none; 51 | font-family: Georgia, 'Times New Roman', Times, serif; 52 | color: rgb(92, 61, 3); 53 | font-size: 15px; 54 | transition: all 0.5s; 55 | } 56 | 57 | form div button:hover 58 | { 59 | background-color: rgb(92, 61, 3); 60 | color: #fff; 61 | } 62 | 63 | #header-login p 64 | { 65 | margin-top: 40px; 66 | text-align: center; 67 | } -------------------------------------------------------------------------------- /routes/index.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var router = express.Router(); 3 | var passport = require("passport"); 4 | var User =require("../models/user"); 5 | 6 | router.get("/",function(req,res) 7 | { 8 | res.render("landing"); 9 | }) 10 | 11 | 12 | 13 | 14 | 15 | //AUTH ROUTES 16 | 17 | router.get("/register",function(req,res) 18 | { 19 | res.render("register"); 20 | }) 21 | 22 | router.post("/register",function(req,res) 23 | { 24 | var newUser= new User({username: req.body.username}); 25 | User.register(newUser,req.body.password,function(err,user){ 26 | if(err) 27 | { 28 | req.flash("error", err.message); 29 | return res.render("register") 30 | } 31 | passport.authenticate("local")(req,res,function() 32 | { 33 | req.flash("success", "Welcome "+newUser.username+"! You have Successfully Signed Up") 34 | res.redirect("/campgrounds"); 35 | }) 36 | }); 37 | }); 38 | 39 | //show login form 40 | router.get("/login",function(req,res) 41 | { 42 | res.render("login"); 43 | }) 44 | 45 | 46 | router.post("/login",passport.authenticate("local", 47 | { 48 | successRedirect: "/campgrounds", 49 | failureRedirect: "/login" 50 | }),function(req,res){ 51 | }) 52 | 53 | router.get("/logout",function(req,res) 54 | { 55 | req.logout(); 56 | req.flash("success", "Logged you out!"); 57 | res.redirect("/campgrounds"); 58 | }) 59 | 60 | function isLoggedIn(req,res,next){ 61 | if(req.isAuthenticated()) 62 | { 63 | return next(); 64 | } 65 | res.redirect("/login") 66 | } 67 | 68 | module.exports=router; -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var app = express(); 3 | app.set("view engine","ejs"); 4 | var bodyParser = require("body-parser"); 5 | var mongoose = require("mongoose"); 6 | var passport=require("passport"), 7 | LocalStrategy=require("passport-local"), 8 | Comment=require("./models/comment"); 9 | var User=require("./models/user"); 10 | mongoose.set('useNewUrlParser', true); 11 | mongoose.set('useUnifiedTopology', true); 12 | var flash=require("connect-flash"); 13 | 14 | methodOverride = require("method-override"); 15 | 16 | //requiring routes 17 | var commentRoutes = require("./routes/comments"); 18 | var campgroundRoutes = require("./routes/campgrounds"); 19 | var indexRoutes = require("./routes/index"); 20 | 21 | 22 | 23 | Campground=require("./models/campground"); 24 | seedDB=require("./seeds") 25 | seedDB(); 26 | 27 | var commentRoutes = require("./routes/comments") 28 | 29 | var url=process.env.DATABASEURL || "mongodb://localhost/yelpcamp" 30 | mongoose.connect(url); 31 | 32 | 33 | app.use(require("express-session")({ 34 | secret: "Its a secret", 35 | resave: false, 36 | saveUninitialized: false 37 | })) 38 | app.use('/public', express.static('public')); 39 | app.use(passport.initialize()); 40 | app.use(passport.session()); 41 | passport.use(new LocalStrategy(User.authenticate())); 42 | passport.serializeUser(User.serializeUser()); 43 | passport.deserializeUser(User.deserializeUser()); 44 | 45 | app.use(methodOverride("_method")); 46 | app.use(flash()); 47 | 48 | 49 | app.use(bodyParser.urlencoded({ 50 | extended: true 51 | })); 52 | app.use(function(req,res,next) 53 | { 54 | res.locals.currentUser = req.user; 55 | res.locals.error=req.flash("error"); 56 | res.locals.success=req.flash("success"); 57 | next(); 58 | }) 59 | 60 | 61 | app.use("/", indexRoutes); 62 | app.use("/campgrounds", campgroundRoutes); 63 | app.use("/campgrounds/:id/comments",commentRoutes); 64 | 65 | 66 | 67 | app.listen(process.env.PORT, process.env.IP,function() //replace process.env.PORT, process.env.IP with 5000 if running in local 68 | { 69 | console.log("Serving AngelCamp"); 70 | }) 71 | -------------------------------------------------------------------------------- /middleware/index.js: -------------------------------------------------------------------------------- 1 | var Campground=require("../models/campground"); 2 | var Comment = require("../models/comment"); 3 | 4 | var middlewareObj={}; 5 | 6 | middlewareObj.checkCampgroundOwnership = function(req,res,next) 7 | { 8 | if(req.isAuthenticated()) 9 | { 10 | Campground.findById(req.params.id, function(err, foundCampground) 11 | { 12 | if(err) 13 | res.redirect("back"); 14 | else 15 | { 16 | if(req.user._id.equals(foundCampground.author.id)) //we cant use === because campground author is a mongoose object while user id is a string 17 | next(); 18 | else 19 | { 20 | req.flash("error", "You don't own the Campground!") 21 | res.redirect("back"); 22 | } 23 | } 24 | }) 25 | } 26 | else 27 | { 28 | req.flash("error", "Please Login First"); 29 | res.redirect("back"); 30 | } 31 | 32 | } 33 | 34 | middlewareObj.checkCommentOwnership=function(req,res,next) 35 | { 36 | if(req.isAuthenticated()) 37 | { 38 | Comment.findById(req.params.comment_id, function(err, foundComment) 39 | { 40 | if(err) 41 | { 42 | req.flash("error", "Comment Not Found!") 43 | res.redirect("back"); 44 | } 45 | else 46 | { 47 | if(req.user._id.equals(foundComment.author.id)) //we cant use === because campground author is a mongoose object while user id is a string 48 | next(); 49 | else 50 | { 51 | req.flash("error", "You have not written this comment!") 52 | res.redirect("back"); 53 | } 54 | } 55 | }) 56 | } 57 | else 58 | { 59 | req.flash("error", "Please Login First") 60 | res.redirect("back"); 61 | } 62 | } 63 | 64 | middlewareObj.isLoggedIn = function(req, res, next) 65 | { 66 | if(req.isAuthenticated()) 67 | { 68 | return next(); 69 | } 70 | req.flash("error","Please Login First!") 71 | res.redirect("/login") 72 | } 73 | 74 | module.exports = middlewareObj -------------------------------------------------------------------------------- /public/assets/showCampground.css: -------------------------------------------------------------------------------- 1 | body .col-md-9 2 | { 3 | padding: 0; 4 | } 5 | 6 | .camp-show 7 | { 8 | background-color: #E2C74B; 9 | } 10 | .camp .thumbnail 11 | { 12 | width: 100%; 13 | background-color: #E2C74B; 14 | border: none; 15 | } 16 | 17 | .camp .thumbnail img 18 | { 19 | width: 100%; 20 | } 21 | 22 | body .lead 23 | { 24 | margin-top: 20vh; 25 | font-family: 'Milonga', cursive; 26 | text-align: center; 27 | background-color: #E2C74B; 28 | font-size: 40px; 29 | } 30 | body .well 31 | { 32 | background-color: #E2C74B; 33 | border: none; 34 | box-shadow: none; 35 | } 36 | 37 | body .list-group li 38 | { 39 | background-color: #E2C74B; 40 | font-family:Georgia, 'Times New Roman', Times, serif; 41 | font-size: 30px; 42 | border: none; 43 | } 44 | 45 | body .caption-full 46 | { 47 | margin: 10px; 48 | } 49 | 50 | body .caption-full p 51 | { 52 | font-size: 18px; 53 | } 54 | 55 | body .well .btn 56 | { 57 | display: inline-block; 58 | } 59 | 60 | body .camp form 61 | { 62 | display: inline; 63 | text-align: right; 64 | } 65 | 66 | .new-comment-but 67 | { 68 | text-align: right; 69 | margin-top: 30px; 70 | } 71 | 72 | .new-comment-but .but 73 | { 74 | height: 80px; 75 | font-size: 15px; 76 | padding: 5px; 77 | } 78 | 79 | body form input .com-del 80 | { 81 | display: inline; 82 | } 83 | 84 | .com-sec 85 | { 86 | margin-top: 80px; 87 | margin-left: 300px; 88 | } 89 | 90 | .comments-sec 91 | { 92 | background-color: #E2C74B;; 93 | } 94 | 95 | .comment 96 | { 97 | border-bottom: 1px solid rgb(92, 61, 3); 98 | margin-bottom: 5px; 99 | padding: 7px; 100 | } 101 | 102 | .author 103 | { 104 | font-family: 'Milonga', cursive; 105 | font-size: 15px; 106 | color: #000; 107 | } 108 | 109 | @media screen and (max-width: 1100px) 110 | { 111 | body .col-md-3 .lead 112 | { 113 | margin-top: 10vh; 114 | } 115 | 116 | .list-group 117 | { 118 | text-align: center; 119 | } 120 | } 121 | 122 | @media screen and (max-width: 800px) 123 | { 124 | body .col-md-3 .lead 125 | { 126 | margin-top: 5vh; 127 | } 128 | body .com-sec 129 | { 130 | margin-left: 0px; 131 | margin-top: 110px; 132 | } 133 | } 134 | @media screen and (max-width: 500px) 135 | { 136 | .text-center div 137 | { 138 | width: 100%; 139 | } 140 | form div input 141 | { 142 | width: 250px; 143 | height: 35px; 144 | font-size: 10px; 145 | } 146 | #header-login 147 | { 148 | width: 90%; 149 | } 150 | } -------------------------------------------------------------------------------- /routes/comments.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var router = express.Router({mergeParams: true}); 3 | var Campground = require("../models/campground") 4 | var Comment = require("../models/comment") 5 | var middleware = require("../middleware") 6 | 7 | //CREATE NEW COMMENT 8 | 9 | router.get("/new",middleware.isLoggedIn, function(req,res) 10 | { 11 | Campground.findById(req.params.id,function(err,campground) 12 | { 13 | if(err) 14 | console.log(err); 15 | else 16 | res.render("comments/new",{campground: campground}); 17 | }) 18 | }) 19 | 20 | router.post("/",middleware.isLoggedIn,function(req,res) 21 | { 22 | Campground.findById(req.params.id,function(err,campground) 23 | { 24 | if(err) 25 | { 26 | console.log(err); 27 | redirect("/campgrounds"); 28 | } 29 | else 30 | { 31 | Comment.create(req.body.comment,function(err,comment) 32 | { 33 | if(err) 34 | console.log(err); 35 | else 36 | { 37 | comment.author.id = req.user._id; 38 | comment.author.username = req.user.username; 39 | comment.save(); 40 | campground.comments.push(comment); 41 | campground.save(); 42 | res.redirect("/campgrounds/"+campground._id); 43 | } 44 | }) 45 | } 46 | }) 47 | 48 | }) 49 | 50 | //EDIT COMMENT 51 | 52 | router.get("/:comment_id/edit", middleware.checkCommentOwnership, function(req,res) 53 | { 54 | Comment.findById(req.params.comment_id, function(err,foundComment) 55 | { 56 | if(err) 57 | res.redirect("back"); 58 | else{ 59 | res.render("comments/edit", {campground_id: req.params.id, comment: foundComment}); 60 | } 61 | }) 62 | }) 63 | 64 | router.put("/:comment_id", middleware.checkCommentOwnership, function(req,res) 65 | { 66 | Comment.findByIdAndUpdate(req.params.comment_id,req.body.comment,function(err, updatedComment) 67 | { 68 | if(err) 69 | res.redirect("back"); 70 | else 71 | { 72 | res.redirect("/campgrounds/" + req.params.id); 73 | } 74 | }) 75 | }) 76 | 77 | //DESTROY COMMENT 78 | 79 | router.delete("/:comment_id",middleware.checkCommentOwnership, function(req,res) 80 | { 81 | Comment.findByIdAndDelete(req.params.comment_id,function(err) 82 | { 83 | if(err) 84 | { 85 | res.redirect("back"); 86 | } 87 | else 88 | { 89 | res.redirect("/campgrounds/"+req.params.id) 90 | } 91 | }) 92 | }) 93 | 94 | 95 | 96 | module.exports=router; -------------------------------------------------------------------------------- /views/partials/header.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | AngelCamp 17 | 18 | 19 | 49 |
50 | <% if(error && error.length>0){ %> 51 | 54 | <%}%> 55 | <% if(success && success.length>0){ %> 56 | 59 | <%}%> 60 |
61 | -------------------------------------------------------------------------------- /views/campgrounds/show.ejs: -------------------------------------------------------------------------------- 1 | <%- include ('../partials/header') %> 2 | 3 |
4 |
5 |
6 |

Camp Details

7 |
8 |
  • Camp Name - <%= campground.name %>
  • 9 |
  • Camp Owner - <%= campground.author.username %>
  • 10 |
    11 |
    12 |
    13 |
    14 | 15 |
    16 |

    17 |

    <%= campground.description %>

    18 | <% if(currentUser && campground.author.id.equals(currentUser._id)){%> 19 | Edit 20 |
    21 | 22 |
    23 | <% }%> 24 |
    25 |
    26 |
    27 |
    28 |
    29 |
    30 |

    Comments

    31 |
    32 | Add New Comment 33 |
    34 |
    35 |
    36 | <% campground.comments.forEach(function(comment){%> 37 |
    38 |

    <%= comment.author.username %>

    39 | 40 |

    41 | <%= comment.text %> 42 |

    43 | <% if(currentUser && comment.author.id.equals(currentUser._id)){%> 44 | 46 | Edit 47 | 48 |
    49 | 50 |
    51 | <% }%> 52 |
    53 | <%}) 54 | %> 55 |
    56 |
    57 |
    58 |
    59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | <%- include ('../partials/footer') %> -------------------------------------------------------------------------------- /public/assets/campgrounds.css: -------------------------------------------------------------------------------- 1 | body .navbar-inverse 2 | { 3 | background-color: #E2C74B; 4 | border-color: #E2C74B; 5 | } 6 | body .navbar-inverse .navbar-brand 7 | { 8 | color: rgb(92, 61, 3); 9 | font-family: 'Milonga', cursive; 10 | } 11 | body .navbar-inverse .navbar-nav>li>a 12 | { 13 | color: rgb(92, 61, 3); 14 | font-family:Georgia, 'Times New Roman', Times, serif; 15 | } 16 | body .navbar-inverse .navbar-toggle:focus, 17 | body .navbar-inverse .navbar-toggle:hover 18 | { 19 | background-color: rgb(92, 61, 3); 20 | } 21 | #header 22 | { 23 | background-color: #E2C74B; 24 | color: rgb(92, 61, 3); 25 | } 26 | #header h1 27 | { 28 | font-family: 'Milonga', cursive; 29 | 30 | } 31 | #header-login 32 | { 33 | width: 700px; 34 | background-color: #E2C74B; 35 | color: rgb(92, 61, 3); 36 | } 37 | 38 | .captions 39 | { 40 | font-family:Georgia, 'Times New Roman', Times, serif; 41 | color: #fff; 42 | margin-bottom: 20px; 43 | } 44 | 45 | .but 46 | { 47 | height: 40px; 48 | width: 250px; 49 | border: 2px solid rgb(92, 61, 3); 50 | color: rgb(92, 61, 3); 51 | text-align: center; 52 | line-height: 35px; 53 | font-size: 20px; 54 | border-radius: 20px; 55 | transition: all 0.5s; 56 | font-family:Georgia, 'Times New Roman', Times, serif; 57 | margin-top: 35px; 58 | } 59 | 60 | .caption 61 | { 62 | transition: all 0.5s; 63 | } 64 | 65 | body .but a, 66 | .caption a 67 | { 68 | text-decoration: none; 69 | color: rgb(92, 61, 3); 70 | } 71 | 72 | .but:hover 73 | { 74 | background-color: rgb(92, 61, 3); 75 | color: #fff; 76 | cursor: pointer; 77 | text-decoration: none; 78 | } 79 | 80 | body .but a:hover, 81 | .caption a:hover 82 | { 83 | color: #fff; 84 | text-decoration: none; 85 | } 86 | 87 | .caption:hover 88 | { 89 | background-color: rgb(92, 61, 3); 90 | } 91 | 92 | body .thumbnail 93 | { 94 | height: 292px; 95 | width: 350px; 96 | padding: 0; 97 | } 98 | 99 | body .thumbnail a img 100 | { 101 | margin: 0; 102 | width: 100%; 103 | height: 240px; 104 | } 105 | 106 | .camp_block_bottom 107 | { 108 | background-color: #E2C74B; 109 | } 110 | 111 | .camp_block_bottom p 112 | { 113 | margin-bottom: 0; 114 | font-family:Georgia, 'Times New Roman', Times, serif; 115 | } 116 | 117 | body .thumbnail .caption 118 | { 119 | padding-bottom: 5px; 120 | } 121 | 122 | body .caption h4 123 | { 124 | font-family: 'Milonga', cursive; 125 | font-size: 20px; 126 | } 127 | 128 | .camp_block_bottom .but 129 | { 130 | width: 130px; 131 | height: 30px;; 132 | margin: auto auto; 133 | line-height: 25px; 134 | } 135 | 136 | .del-form 137 | { 138 | display: inline; 139 | } 140 | 141 | 142 | @media screen and (max-width: 1100px) 143 | { 144 | body .col-md-4 .thumbnail 145 | { 146 | width: 105%; 147 | } 148 | } 149 | 150 | @media screen and (max-width: 800px) 151 | { 152 | body .col-md-4 .thumbnail 153 | { 154 | width: 100%; 155 | } 156 | } 157 | 158 | -------------------------------------------------------------------------------- /public/assets/landing.css: -------------------------------------------------------------------------------- 1 | body { 2 | background-color: #000; 3 | background-image: linear-gradient(to right, rgba(77, 72, 72, 0.9), rgb(77, 72, 72,0.9)); 4 | } 5 | 6 | div h1{ 7 | font-family: 'Milonga', cursive; 8 | font-size: 70px; 9 | } 10 | 11 | div p{ 12 | font-size: 20px; 13 | font-family:Georgia, 'Times New Roman', Times, serif; 14 | font-weight: 400; 15 | color:#fff; 16 | margin-bottom: 20px; 17 | } 18 | 19 | .landing-button 20 | { 21 | height: 30px; 22 | width: 100px; 23 | border: 2px solid #E2C74B; 24 | color: #E2C74B; 25 | text-align: center; 26 | margin: 0; 27 | margin: 0 auto; 28 | font-size: 20px; 29 | border-radius: 15px; 30 | transition: all 0.5s; 31 | } 32 | 33 | .landing-button:hover 34 | { 35 | background-color: #E2C74B; 36 | } 37 | 38 | .landing-button a 39 | { 40 | color: #E2C74B; 41 | text-decoration: none; 42 | } 43 | 44 | .landing-button a:hover 45 | { 46 | color: #fff; 47 | text-decoration: none; 48 | } 49 | 50 | 51 | #landing-header { 52 | z-index: 1; 53 | position: relative; 54 | text-align: center; 55 | padding-top: 30vh; 56 | } 57 | 58 | #landing-header h1 { 59 | color:#fff; 60 | } 61 | 62 | .slideshow { 63 | position: fixed; 64 | width: 100%; 65 | height: 100%; 66 | top: 0; 67 | left: 0; 68 | z-index: 0; 69 | list-style: none; 70 | margin: 0; 71 | padding: 0; 72 | } 73 | 74 | .slideshow li { 75 | width: 100%; 76 | height: 100%; 77 | position: absolute; 78 | top: 0; 79 | left: 0; 80 | background-size: cover; 81 | background-position: 50% 50%; 82 | background-repeat: no-repeat; 83 | opacity: 0; 84 | z-index: 0; 85 | animation: imageAnimation 25s linear infinite; 86 | } 87 | 88 | .slideshow li:nth-child(1) { 89 | background-image: linear-gradient(to right, rgba(48, 45, 45, 0.5), rgb(48, 45, 45,0.5)), url(http://i.imgur.com/K3mPv14.jpg) 90 | } 91 | .slideshow li:nth-child(2) { 92 | background-image: linear-gradient(to right, rgba(48, 45, 45, 0.5), rgb(48, 45, 45,0.5)),url(http://i.imgur.com/SBEmFpv.jpg); 93 | animation-delay: 5s; 94 | } 95 | .slideshow li:nth-child(3) { 96 | background-image: linear-gradient(to right, rgba(48, 45, 45, 0.5), rgb(48, 45, 45,0.5)),url(http://i.imgur.com/emvhOnb.jpg); 97 | animation-delay: 10s; 98 | } 99 | .slideshow li:nth-child(4) { 100 | background-image: linear-gradient(to right, rgba(48, 45, 45, 0.5), rgb(48, 45, 45,0.5)),url(https://images.unsplash.com/photo-1484960055659-a39d25adcb3c?ixlib=rb-1.2.1&auto=format&fit=crop&w=1350&q=80); 101 | animation-delay: 15s; 102 | } 103 | .slideshow li:nth-child(5) { 104 | background-image: linear-gradient(to right, rgba(48, 45, 45, 0.5), rgb(48, 45, 45,0.5)),url(http://i.imgur.com/TVGe0Ef.jpg); 105 | animation-delay: 20s; 106 | } 107 | 108 | @keyframes imageAnimation { 109 | 0% { 110 | opacity: 0; 111 | animation-timing-function: ease-in; 112 | } 113 | 10% { 114 | opacity: 1; 115 | animation-timing-function: ease-out; 116 | } 117 | 20% { 118 | opacity: 1 119 | } 120 | 30% { 121 | opacity: 0 122 | } 123 | } 124 | 125 | /* Older browser support - .no-cssanimations class added by modernizr */ 126 | .no-cssanimations .slideshow li { 127 | opacity: 1; 128 | } -------------------------------------------------------------------------------- /routes/campgrounds.js: -------------------------------------------------------------------------------- 1 | var express = require("express"); 2 | var router = express.Router(); 3 | var Campground = require("../models/campground"); 4 | var middleware = require("../middleware") 5 | 6 | 7 | //SHOW ALL CAMPGROUNDS 8 | 9 | router.get("/",function(req,res) 10 | { 11 | Campground.find({},function(err, allCampgrounds) 12 | { 13 | if(err){ 14 | console.log(err); 15 | }else{ 16 | res.render("campgrounds/index",{campgrounds:allCampgrounds, currentUser: req.user}); 17 | } 18 | }) 19 | }) 20 | 21 | //CREATE NEW CAMPGROUND 22 | 23 | router.get("/new",middleware.isLoggedIn,function(req,res) 24 | { 25 | res.render("campgrounds/new.ejs"); 26 | }) 27 | 28 | router.post("/",middleware.isLoggedIn,function(req,res) 29 | { 30 | var name=req.body.name; 31 | var image=req.body.image; 32 | var desc=req.body.description; 33 | var author = { 34 | id: req.user._id, 35 | username: req.user.username 36 | } 37 | var newCampground = {name: name, image: image, description: desc, author:author} 38 | Campground.create(newCampground, function(err, newlyCreated) 39 | { 40 | if(err) 41 | { 42 | console.log(err); 43 | } 44 | else 45 | { 46 | req.flash("success", "Successfully created your Campground!") 47 | res.redirect("/campgrounds"); 48 | } 49 | }); 50 | }); 51 | 52 | //SHOW A SPECIFIC CAMPGROUND 53 | 54 | router.get("/:id",function(req,res) 55 | { 56 | Campground.findById(req.params.id).populate("comments").exec(function(err, foundCampground) 57 | { 58 | if(err) 59 | console.log(err); 60 | else 61 | { 62 | res.render("campgrounds/show",{campground: foundCampground, currentUser: req.user}); 63 | } 64 | }) 65 | }); 66 | 67 | //EDIT CAMPGROUND 68 | router.get("/:id/edit",middleware.checkCampgroundOwnership, function(req,res) 69 | { 70 | Campground.findById(req.params.id, function(err, foundCampground) 71 | { 72 | if(req.user._id.equals(foundCampground.author.id)) //we cant use === because campground author is a mongoose object while user id is a string 73 | { 74 | res.render("campgrounds/edit", {campground: foundCampground}) 75 | } 76 | }); 77 | }); 78 | 79 | router.put("/:id", middleware.checkCampgroundOwnership, function(req,res) 80 | { 81 | Campground.findByIdAndUpdate(req.params.id, req.body.campground,function(err, updatedCampground) 82 | { 83 | if(err) 84 | res.redirect("/campgrounds"); 85 | else 86 | { 87 | req.flash("success", "You have Successfully edited your Campground!"); 88 | res.redirect("/campgrounds/"+ req.params.id); 89 | } 90 | }) 91 | }) 92 | 93 | //Destroy Campground 94 | 95 | router.delete("/:id",middleware.checkCampgroundOwnership, function(req,res) 96 | { 97 | Campground.findByIdAndRemove(req.params.id, function(err) 98 | { 99 | if(err) 100 | res.redirect("/campgrounds"); 101 | else 102 | { 103 | req.flash("success", "You have Successfully deleted your Campground!") 104 | res.redirect("/campgrounds"); 105 | } 106 | }) 107 | }) 108 | 109 | 110 | 111 | 112 | module.exports=router; -------------------------------------------------------------------------------- /seeds.js: -------------------------------------------------------------------------------- 1 | var mongoose = require("mongoose"); 2 | var Campground=require("./models/campground"); 3 | var Comment=require("./models/comment"); 4 | 5 | var data = [/* 6 | { 7 | name: "Cloud's Rest", 8 | image: "https://encrypted-tbn0.gstatic.com/images?q=tbn%3AANd9GcSuzTFxpOhC0VVryrqB9kWry7CDSa_-85Budr0Zn9iLFnGymkbE", 9 | description: "blah blah blah" 10 | }, 11 | { 12 | name: "Desert Mesa", 13 | image: "", 14 | description: "blah blah blah" 15 | }, 16 | { 17 | name: "Canyon floor", 18 | image: "", 19 | description: "blah blah blah" 20 | },*/ 21 | ] 22 | 23 | function seedDB() 24 | {/* 25 | Campground.remove({},function(err) 26 | { 27 | if(err) 28 | console.log(err); 29 | else 30 | console.log("Campgrounds Removed!"); 31 | data.forEach(function(seed) 32 | { 33 | Campground.create(seed,function(err,campground) 34 | { 35 | if(err) 36 | console.log(err); 37 | else 38 | { 39 | 40 | Comment.create({ 41 | text: "A great place!", 42 | author: "Homer" 43 | },function(err,comment) 44 | { 45 | if(err) 46 | { 47 | console.log(err); 48 | console.log("Going to create comment!"); 49 | } 50 | else 51 | { 52 | console.log("Comment Added!"); 53 | campground.comments.push(comment); 54 | campground.save(); 55 | } 56 | }) 57 | } 58 | }) 59 | }) 60 | }) 61 | */} 62 | 63 | module.exports=seedDB --------------------------------------------------------------------------------