亲自上手,用原生 JavaScript 打造简易电影选座系统

87b6aa6c7e5898f2b42b690467ef71dc.jpeg

大家有没有在使用App选电影票的经历呢?我们今天就来实现一个简易版的电影选座系统。这个系统的功能包括:

  • 显示电影列表,并且可以选择不同的电影

  • 展示座位图,并且可以选择座位

  • 实时计算和显示已选座位数量和总价

  • 使用本地缓存保存用户的选择状态,并在页面刷新后保持状态

案例展示

我们来看一下最终实现的效果,如图所示:19ac3a50a5cf4a15c00ce4621b172563.png

项目实现思路

接下来我们需要按照这个思路制作我们的项目,我会一一进行详细介绍。

  • 创建基础HTML结构:定义选择电影的下拉菜单和座位布局。

  • 添加CSS样式:美化页面,使其更符合实际的电影院选座系统。

  • 实现JavaScript逻辑:处理座位选择、电影选择、价格计算和状态保存等功能。

  • 本地存储:使用浏览器的localStorage保存用户选择的电影和座位信息,在页面刷新时重新加载这些信息。

创建基础HTML结构

首先,我们需要一个基础的HTML结构来展示电影列表和座位布局。这一步主要是定义页面的基本框架,让用户能够看到选择电影和座位的界面。

a. 确定页面的主要部分

一个简单的电影选座系统主要包含以下几个部分:

  • 电影选择区域

  • 座位展示区域

  • 已选座位信息展示区域

HTML代码

以下是完整的HTML代码,并附有注释帮助理解:

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <link rel="stylesheet" href="style.css">
  <title>Movie Seat Booking</title>
</head>
<body>
  <!-- 电影选择区域 -->
  <div class="movie-container">
    <label for="movie">Pick a movie:</label>
    <select id="movie">
      <option value="10">Avengers: Endgame ($10)</option>
      <option value="12">Joker ($12)</option>
      <option value="8">Toy Story 4 ($8)</option>
      <option value="9">The Lion King ($9)</option>
    </select>
  </div>

  <!-- 座位展示区域 -->
  <ul class="showcase">
    <li>
      <div class="seat"></div>
      <small>N/A</small>
    </li>
    <li>
      <div class="seat selected"></div>
      <small>Selected</small>
    </li>
    <li>
      <div class="seat occupied"></div>
      <small>Occupied</small>
    </li>
  </ul>

  <!-- 座位布局 -->
  <div class="container">
    <div class="screen"></div> <!-- 屏幕 -->
    
    <!-- 一排座位 -->
    <div class="row">
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
      <div class="seat"></div>
    </div>
    <!-- 更多排座位可以按需添加 -->
  </div>

  <!-- 已选座位信息展示区域 -->
  <p class="text">
    You have selected <span id="count">0</span> seats for a price of $<span id="total">0</span>
  </p>

  <script src="script.js"></script>
</body>
</html>

添加CSS样式

为了使我们的页面更美观且易于使用,我们需要一些CSS样式。样式的目的是让页面看起来更像电影院的座位选择界面。

@import url('https://fonts.googleapis.com/css?family=Lato&display=swap');

* {
  box-sizing: border-box;
}

body {
  background-color: #242333;
  color: #fff;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  height: 100vh;
  font-family: 'Lato', sans-serif;
  margin: 0;
}

.movie-container {
  margin: 20px 0;
}

.movie-container select {
  background-color: #fff;
  border: 0;
  border-radius: 5px;
  font-size: 14px;
  margin-left: 10px;
  padding: 5px 15px 5px 15px;
  -moz-appearance: none;
  -webkit-appearance: none;
  appearance: none;
}

.container {
  perspective: 1000px;
  margin-bottom: 30px;
}

.seat {
  background-color: #444451;
  height: 12px;
  width: 15px;
  margin: 3px;
  border-top-left-radius: 10px;
  border-top-right-radius: 10px;
}

.seat.selected {
  background-color: #6feaf6;
}

.seat.occupied {
  background-color: #fff;
}

.seat:nth-of-type(2) {
  margin-right: 18px;
}

.seat:nth-last-of-type(2) {
  margin-left: 18px;
}

.seat:not(.occupied):hover {
  cursor: pointer;
  transform: scale(1.2);
}

.showcase .seat:not(.occupied):hover {
  cursor: default;
  transform: scale(1);
}

.showcase {
  background: rgba(0, 0, 0, 0.1);
  padding: 5px 10px;
  border-radius: 5px;
  color: #777;
  list-style-type: none;
  display: flex;
  justify-content: space-between;
}

.showcase li {
  display: flex;
  align-items: center;
  justify-content: center;
  margin: 0 10px;
}

.showcase li small {
  margin-left: 2px;
}

.row {
  display: flex;
}

.screen {
  background-color: #fff;
  height: 70px;
  width: 100%;
  margin: 15px 0;
  transform: rotateX(-45deg);
  box-shadow: 0 3px 10px rgba(255, 255, 255, 0.7);
}

p.text {
  margin: 5px 0;
}

p.text span {
  color: #6feaf6;
}

JavaScript功能实现

JavaScript是实现页面交互的核心部分,通过JavaScript我们可以实现座位选择、电影选择、价格计算和状态保存等功能。以下是详细的实现思路:

a. 获取DOM元素

首先,我们需要获取页面上的一些重要元素,以便后续操作。

const container = document.querySelector('.container');
const seats = document.querySelectorAll('.row .seat:not(.occupied)');
const count = document.getElementById('count');
const total = document.getElementById('total');
const movieSelect = document.getElementById('movie');

b. 初始化票价和电影数据

通过读取电影选择框的值来初始化票价。

let ticketPrice = +movieSelect.value;

c. 事件监听

为电影选择框和座位添加事件监听器,用户选择电影或座位时触发相应的处理逻辑。

movieSelect.addEventListener('change', e => {
  ticketPrice = +e.target.value;
  setMovieData(e.target.selectedIndex, e.target.value);
  updateSelectedCount();
});

container.addEventListener('click', e => {
  if (
    e.target.classList.contains('seat') &&
    !e.target.classList.contains('occupied')
  ) {
    e.target.classList.toggle('selected');
    updateSelectedCount();
  }
});

d. 更新选中座位数和总价

当用户选择或取消选择座位时,更新座位数量和总价,并将选中状态保存到本地存储。

function updateSelectedCount() {
  const selectedSeats = document.querySelectorAll('.row .seat.selected');
  const seatsIndex = [...selectedSeats].map(seat => [...seats].indexOf(seat));

  localStorage.setItem('selectedSeats', JSON.stringify(seatsIndex));

  const selectedSeatsCount = selectedSeats.length;

  count.innerText = selectedSeatsCount;
  total.innerText = selectedSeatsCount * ticketPrice;
}

e. 本地存储

为了保持用户的选择状态,我们使用localStorage来保存和读取数据。

保存选中电影和座位的信息:

function setMovieData(movieIndex, moviePrice) {
  localStorage.setItem('selectedMovieIndex', movieIndex);
  localStorage.setItem('selectedMoviePrice', moviePrice);
}

从本地存储中读取数据并更新页面:

function populateUI() {
  const selectedSeats = JSON.parse(localStorage.getItem('selectedSeats'));

  if (selectedSeats !== null && selectedSeats.length > 0) {
    seats.forEach((seat, index) => {
      if (selectedSeats.indexOf(index) > -1) {
        seat.classList.add('selected');
      }
    });
  }

  const selectedMovieIndex = localStorage.getItem('selectedMovieIndex');

  if (selectedMovieIndex !== null) {
    movieSelect.selectedIndex = selectedMovieIndex;
  }
}

完整的JS代码

const container = document.querySelector('.container');
const seats = document.querySelectorAll('.row .seat:not(.occupied)');
const count = document.getElementById('count');
const total = document.getElementById('total');
const movieSelect = document.getElementById('movie');

populateUI();

let ticketPrice = +movieSelect.value;

// 保存选中的电影索引和价格
function setMovieData(movieIndex, moviePrice) {
  localStorage.setItem('selectedMovieIndex', movieIndex);
  localStorage.setItem('selectedMoviePrice', moviePrice);
}

// 更新选中座位数和总价
function updateSelectedCount() {
  const selectedSeats = document.querySelectorAll('.row .seat.selected');

  const seatsIndex = [...selectedSeats].map(seat => [...seats].indexOf(seat));

  localStorage.setItem('selectedSeats', JSON.stringify(seatsIndex));

  const selectedSeatsCount = selectedSeats.length;

  count.innerText = selectedSeatsCount;
  total.innerText = selectedSeatsCount * ticketPrice;
}

// 从本地存储中获取数据并更新UI
function populateUI() {
  const selectedSeats = JSON.parse(localStorage.getItem('selectedSeats'));

  if (selectedSeats !== null && selectedSeats.length > 0) {
    seats.forEach((seat, index) => {
      if (selectedSeats.indexOf(index) > -1) {
        seat.classList.add('selected');
      }
    });
  }

  const selectedMovieIndex = localStorage.getItem('selectedMovieIndex');

  if (selectedMovieIndex !== null) {
    movieSelect.selectedIndex = selectedMovieIndex;
  }
}

// 电影选择事件
movieSelect.addEventListener('change', e => {
  ticketPrice = +e.target.value;
  setMovieData(e.target.selectedIndex, e.target.value);
  updateSelectedCount();
});

// 座位点击事件
container.addEventListener('click', e => {
  if (
    e.target.classList.contains('seat') &&
    !e.target.classList.contains('occupied')
  ) {
    e.target.classList.toggle('selected');

    updateSelectedCount();
  }
});

// 初始计数和总价设置
updateSelectedCount();

通过以上步骤,我们实现了一个基本的电影选座系统,用户可以选择电影和座位,并且在页面刷新后仍然保留之前的选择状态。这一系列的操作将帮助初学者理解如何通过前端技术来实现一个功能完备的小项目。

结束

这就是一个简易的电影选座系统的实现过程啦!你是不是也觉得自己能动手做一个了呢?赶快试试看吧!如果你有任何问题或者改进建议,欢迎在评论区留言哦!希望大家都能学有所成,成为代码大神!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值