ID311 Final Peoject
Go to file
2025-06-11 23:57:41 +09:00
src fix type error 2025-06-11 23:57:41 +09:00
static fix type error 2025-06-11 23:57:41 +09:00
.gitignore Initial commit 2025-05-26 13:38:50 +09:00
.npmrc Initial commit 2025-05-26 13:38:50 +09:00
netlify.toml add netlify config 2025-06-10 20:59:53 +09:00
package-lock.json change to netlify 2025-06-10 20:50:41 +09:00
package.json change to netlify 2025-06-10 20:50:41 +09:00
README.md fix type error 2025-06-11 23:57:41 +09:00
svelte.config.js change to netlify 2025-06-10 20:50:41 +09:00
tsconfig.json Initial commit 2025-05-26 13:38:50 +09:00
vite.config.ts added map @ itinerary page 2025-05-31 23:39:27 +09:00

Home Page Screenshot

SvelteKit    Firebase    OpenAI    Netlify

Student Information

Name: Adelia Putri
Student ID: 20210782
Email: adelia@kaist.ac.kr

Name: Chaebean Yang
Student ID: 20230412
Email: kazed0102@kaist.ac.kr

Git Repository: Triptide Repository
Live Demo: Triptide
Demo Video: Triptide Demo Video

Table of Contents

App Description

Triptide isn't just a planner, it's your personal travel story. It weaves together the memories of where you've been with the excitement of where you'll go next. Relive your journeys through vibrant photo galleries and color mood boards, then let those experiences inspire your next adventure through AI-powered recommendations.

How to Run Locally

  1. Prerequisites
    • Node.js (v18 or later)
  2. Clone the repository
    git clone http://git.prototyping.id/20210782/travel-app
    cd travel-app
    
  3. Install dependencies
    npm install
    
  4. Set up environment variables
    Create a .env file in the project root and add the following API keys.
     # Firebase Configuration
     VITE_FIREBASE_API_KEY="your_firebase_api_key"
     VITE_FIREBASE_AUTH_DOMAIN="your_firebase_auth_domain"
     VITE_FIREBASE_DATABASE_URL="your_firebase_database_url"
     VITE_FIREBASE_PROJECT_ID="your_firebase_project_id"
     VITE_FIREBASE_STORAGE_BUCKET="your_firebase_storage_bucket"
     VITE_FIREBASE_MESSAGING_SENDER_ID="your_firebase_messaging_sender_id"
     VITE_FIREBASE_APP_ID="your_firebase_app_id"
    
     # Google Maps API
     VITE_GOOGLE_MAPS_API_KEY="your_google_maps_api_key"
    
     # OpenAI API
     VITE_OPENAI_API_KEY="your_openai_api_key"
    
     # Unsplash API
     VITE_UNSPLASH_API_KEY="your_unsplash_api_key"
    
     # Uploadcare API
     VITE_UPLOADCARE_PUBLIC_KEY="your_uploadcare_public_key"
    
  5. Run the development server
    npm run dev
    
  6. Open the app

Core Features

Planner: Plan Your Next Adventure

Home Page

Visualize your global travel history on an interactive world map. Markers indicate every location you've visited. Clicking a marker reveals a list of trips to that location and clicking a trip label navigates directly to the detailed itinerary page for that trip.

Home Page Screenshot

Users can add a new trip by clicking the Plan a new trip button at the bottom right and filling out the trip creation form.

Itinerary Page

Plan your trip by requesting AI-generated recommendations for places to visit. If you have previously visited the destination, Triptide will ask for your preference regarding the type of recommendations (new, old, or mixed).

Recommendation Popup Screenshot

After receiving recommendations, Triptide helps you organize them by dividing the places into your trip days, optimizing for travel time and location with a help from LLM.

My Trips Page

View all your upcoming and past trips in one place.

My Trips Screenshot

Memory: Relive Your Journeys

Home Page

On the map, you can see all the locations where you've added photos. Additionally, if there are trips planned through the Triptide Planner that havent had any photos added yet, they will appear as markers. This visual connection between the planner and memory sections helps transform your travel plans into lasting memories.

Memory Home Screenshot

View Photos by Location Page

You can collect and revisit your travel photos all in one place. By uploading your images, you can visualize the mood of your journey through colors. When youve visited the same place more than once, you can compare each set of photos side by side, exploring visual similarities through shared color tones. This page offers a new, engaging way to reflect on your travels—through a uniquely visual and interactive experience.

View Photos Screenshot

My Memories Page

Browse a gallery of all your past trips, each represented by a photo deck.

View My Memories Screenshot

Code Organization

Tech Stack

  • Frontend: SvelteKit
  • Database: Firebase Realtime Database
  • Hosting: Netlify
  • Libraries:
    • D3.js to render the interactive WorldMap.svelte
  • APIs Used:

Component & File Structure

.
├── README.md                  # Project documentation
├── package.json               # Project dependencies and scripts
├── src
│   ├── app.css                # Global styles
│   ├── firebase.js            # Firebase configuration
│   ├── lib
│   │   ├── components         # Svelte UI components
│   │   │   ├── AddPlaces.svelte      # Place autocomplete and selection
│   │   │   ├── Button.svelte         # Reusable button component
│   │   │   ├── MemoryPopup.svelte    # Popup for adding/viewing memories
│   │   │   ├── WorldMap.svelte       # Interactive world map visualization
│   │   │   └── ...                   # Other UI components
│   │   └── constants
│   │       ├── Colors.ts             # Color palette and theme
│   │       ├── CountryMappings.ts    # Country code mappings
│   │       └── Interfaces.ts         # Shared TypeScript interfaces
│   ├── routes
│   │   ├── itinerary/[tid]           # Itinerary page for a specific trip
│   │   ├── memories                  # Memory-related pages
│   │   ├── mymemory                  # User's personal memory pages
│   │   ├── trips                     # List and management of trips
│   │   ├── viewimage/[tripId]/[memoryId] # View a specific memory image
│   │   └── +page.svelte              # Home page route
│   └── services
│       ├── openai.ts                 # OpenAI API integration and prompts
│       └── unsplash.ts               # Unsplash API integration
├── static                            # Photos used in the project
├── svelte.config.js
├── tsconfig.json
└── vite.config.ts

Database Structure

Recommendation Popup Screenshot

Design Patterns

MVC (Model-View-Controller)

  • Model: Data is managed in Firebase and accessed via service modules.
  • View: Svelte components render the UI and respond to user interaction.
  • Controller: Logic in Svelte scripts and service files acts as the controller, handling user actions and data flow.

Observer Pattern

The observer pattern is used for real-time data updates

  • Subject: Firebase Realtime Database nodes (e.g., trips, memories)
  • Observer: Svelte components subscribe to changes using Firebase's onValue method, automatically updating the UI when data changes.

Singleton Pattern

  • The Firebase app instance is implemented as a singleton in firebase.js, ensuring only one instance is used throughout the app.

Service Pattern

  • API integrations (OpenAI, Unsplash) are abstracted into service modules in src/services, promoting code reuse and separation of concerns.

Limitations

  • API Rate Limiting: The Unsplash free tier limits API calls to 50 requests/hour, which can temporarily disrupt the image-fetching feature during heavy use.
  • UAPI Rate Limiting: Due to limitations of the free version of Uploadcare, EXIF metadata (such as timestamp and location) is stripped during image upload. As a result, we were unable to provide contextual information extracted from images.
  • Map Interactions: Markers on the itinerary page map are currently for display only and are not clickable.
  • Memory Photo Limit: Adding a large number of photos to a trip can result in slow page rendering and decreased performance.

Acknowledgement

  • Logo icons were obtained from Font Awesome
  • The world map GeoJSON data was retrieved from GeoJSON Maps
  • The implementation of the Place Autocomplete widget was guided by the official Google Maps Platform documentation
  • Cursor was used for initial page scaffolding from Figma designs and for generating code comments to improve readability.
  • The official SvelteKit Docs were an essential resource for resolving deprecation problems