Skip to content

An MERN stack application developed to help photographers find and share lesser-known photo locations.

Notifications You must be signed in to change notification settings

T1LT/ViewFinder

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Description

ViewFinder is a web application that allows photographers to find specific spots in California that would be a great place to take a picture perfect photo. The photographers can share photos with others by posting their photo and tagging the conditions in which they took it. It allows users to view the page of a specific photographer and like the photos that they love the most.

splashpage_compressed

clustering

List of technologies used

ViewFinder uses a MERN stack (Mongo, Express, React, and Node).

Significant Code

Searchbar Logic

The search feature is made possible through this simple router. The route accepts a type and body as query keys, in which the type must match the one of the type within the included types array. A second check is done to ensure that the body key value is not empty. After passing the the two checks, The if else block is hit in which it will enter a specific condition based on the type. After entering a condition, the it will utilize regex to search through the different fields of the collection. If any data matched the regex pattern, that document is pulled and added to the the query array. The queried data is then normalized using the standardizeData and sent back to the client.

searchbar

router.get('/', async (req, res, next) => {
    //expect query to come in as ?body=val
    try {
      
      //ensures body is not empty 
      if (!req.query.body) return res.json({ message: "Missing Body" });

      //query photos collection 
      let photos = await Photo.find({
        $or: [
          {
            description: {
              $regex: new RegExp(req.query.body, "i"),
            },
          }
        ],
      });

      //query users collection 
      let users = await User.find({
        $or: [
          {
            username: {
              $regex: new RegExp(req.query.body, "i"),
            },
          },
          {
            email: {
              $regex: new RegExp(req.query.body, "i"),
            },
          },
        ],
      }).select("_id username email createdAt updatedAt");

      //query locations collection 
      let locations = await Location.find({
        $or: [
          {
            county: {
              $regex: new RegExp(req.query.body, "i"),
            },
          },
        ],
      });

      //normalize the array into object and deconstruct all the objects into one object  
      if (!users.length && !locations.length && !photos.length) return res.json({ search: [] });
      return res.json({...standardizeData(users), ...standardizeData(locations), ...standardizeData(photos)});
    }
    catch(err) {
        const error = new Error("Search failed");
        error.statusCode = 404;
        error.errors = {
          message: `Failed to find a photo, a location, or/and user; ${err}`,
        };
        return next(error);
    }
});

Photo upload logic

When a user adds a photo, there are two main cases that can happen. They can either upload a picture to an existing or non existing spot. A photo is added to a spot if the photo’s latitude and longitude is within a one mile distance of an existing spot. The spotSearchRectangle returns an object with the keys of top and bottom points, which are utilized within the Spot find. If the spot exists, the photo is created and saved, and all the corresponding collections storing a ref for photos are updated as well before the data is sent back to the client. If the spot does not exist, then a new spot must be create. A new spot is created based on the request latitude and longitude, and the new field is filled using the request description ment for the photo. Once the spot is saved to the collection, a specific document must be updated within the location collection. In order to complete this action, a server side google map’s reverse geocode api is utilized. The request latitude and longitude are imputed into the api in which a lot of information is returned from the fetch. All the locations have a specific county name, so I utilize the countySearch function to retrieve the county name form date requested from the google map’s api. The county name is used to find the location and updates its spots field to include the newly created spot. Once creating a new spot and updating the location’s spots field is updated, a new photo is created & saved, and all the corresponding collections storing a ref for photos are updated as well. The data is sent back to the client.

router.post(
 "/",
 singleMulterUpload("images"),
 requireUser,
 validatePhotoInput,
 async (req, res, next) => {
   try {
     const client = new Client({});
     recArea = spotSearchRectangle(+req.body.latitude, +req.body.longitude);
     let spot = await Spot.find({
       $and: [
         { latitude: { $gte: recArea.top1[0], $lte: recArea.bottom1[0] } },
         { longitude: { $gte: recArea.bottom1[1], $lte: recArea.top1[1] } },
       ],
     });
     if (!spot.length) {
       const newSpot = new Spot({
         latitude: +req.body.latitude,
         longitude: +req.body.longitude,
         name: req.body.description,
       });
       spot = await newSpot.save();
       const mapResult = await client.reverseGeocode({
         params: {
           latlng: [+req.body.latitude, +req.body.longitude],
           key: process.env.GOOGLE_MAPS_API_KEY,
         },
         timeout: 1000,
       });
       let county = countySearch(mapResult.data.results);
       await Location.updateOne(
         { county: county },
         { $push: { spots: spot._id } }
       );
     } else {
       spot = spot.pop();
     }
     const imgUrl = await singleFileUpload({ file: req.file, public: true });
     const newPhoto = new Photo({
       url: imgUrl,
       spotId: spot._id,
       userId: req.user._id,
       latitude: +req.body.latitude,
       longitude: +req.body.longitude,
       genre: req.body.genre,
       description: req.body.description,
       condition: req.body.condition,
       transportation: req.body.transportation,
       bestTimeOfDay: req.body.bestTimeOfDay,
       payment: req.body.payment,
       likes: [],
     });
     let photo = await newPhoto.save();
     await spot.photos.addToSet(photo._id);
     await spot.save();
     await User.updateOne(
       { _id: photo.userId },
       { $push: { photos: photo._id } }
     );
     return res.json(photo);
   } catch (err) {
     const error = new Error("New Photo Upload failed");
     error.statusCode = 404;
     error.errors = { message: `Failed to create a new photo, ${err} ` };
     return next(error);
   }
 }
);

Screenshots

Modal for inspecting a photo

Screen Shot 2023-01-29 at 5 21 55 PM

Spot Show Page

spotshowpage

User Profile Page

Screen Shot 2023-01-29 at 5 20 18 PM

Technical implementation details

We had to think about the flow of the website in general, where to post photos and how a photo is nested in a spot and nested into a location. We had multiple talks about how the backend should be implemented and how it affects the frontend.

To-dos and future features

  • Allow users to see photos they liked
  • Heatmaps
  • Fix likes on the users show page
  • Add more seeds
  • Update the photo upload feature on the form
  • Add loading spinner

About

An MERN stack application developed to help photographers find and share lesser-known photo locations.

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 94.5%
  • CSS 5.1%
  • HTML 0.4%