基于华为云主机 CodeArts IDE 与 JavaScript 的活动抽奖网页设计

基于华为云的抽奖网页设计

本案例由开发者:华为2024年第三批次协同育人项目-山东科技大学-崔宾阁老师提供

1 概述

1.1 案例介绍

随着动态交互需求的日益增长,活动抽奖作为一个典型的网页互动功能,已被广泛应用于各类线上活动、促销页面及社交平台中。华为开发者空间云主机平台为前端开发提供了稳定高效的开发环境,使开发者能够在云端快速搭建应用。

本案例依托华为开发者空间提供的云主机与CodeArts IDE,结合JavaScript技术,开发了一个活动抽奖系统。其主要功能包括奖品展示、抽奖、中奖结果展示以及抽奖历史记录查看等。在实现上,首先设计一个九宫格布局来展示奖品。通过JavaScript实现随机抽奖,并在用户点击“开始抽奖”按钮后,呈现动态的抽奖动画效果,然后在抽奖结果区展示用户抽中的奖品。此外,用户可以通过抽奖速度滑块调整抽奖速度,增加互动性。抽奖结果也会记录在历史记录中,方便用户查看之前的抽奖情况。

通过本案例,开发者将深入学习HTML表单元素的构建、CSS布局与定位、视觉效果增强及 JavaScript动态元素运用、事件监听等技术。同时,开发者将熟悉华为云主机平台及CodeArts IDE开发环境,提升项目实战能力,为后续更复杂的Web应用开发打下坚实基础。

1.2 案例时间

本案例总时长预计30分钟。

1.3 案例流程

06983e38c62e4e02bf76e04437291c68

说明:

① 申请并登录华为开发者空间—云主机,打开CodeArts IDE创建工程;

② 在云主机CodeArts IDE for Java中编写案例项目代码;

③ 通过Firefox启动案例项目的入口文件;

1.4 资源总览

本案例预计花费总计0元。

资源名称规格单价(元)时长(分钟)
开发者空间—云主机鲲鹏通用计算增强型 kC2 | 4vCPUs | 8G | Ubuntu免费30
CodeArts IDECodeArts IDE for Java免费30


2 操作步骤

2.1 配置开发者空间—云主机

本案例中,使用华为开发者空间所提供的云主机平台以及CodeArts IDE + JavaScript开发工具,完成活动抽奖案例的开发工作。点击链接可跳转至免费领取云主机指南。

1. 在浏览器中输入华为云开发者空间网址:https://developer.huaweicloud.com/developerspace,进入华为云开发者空间页面。在华为开发者空间页面点击“免费领取”,跳转到开发者空间页面,如未领取根据页面提示进行云主机领取。

52fc4cb49a3d4ce38010deda58aa3d7f

2. 在开发者空间页面,点击左侧“工作台”按钮进入工作台页面,再点击“配置云主机”进行云主机的配置。

563917c8df844aae8375e270bd12deed

3. 在配置云主机窗口中自定义云主机名称,配置完毕后点击“安装”。

4ad412af93ba40ceb753f7b5e08b1b1e

4. 安装完毕后点击“打开云主机”>“进入桌面”即可进入云主机。

06d414fe372648b1adeee803d16bca5d

5. 等待环境云主机下载镜像、安装系统、安装工具集,首次进入云主机大约需要3至5分钟。

fb889024f1894b63902ae192aea8aec5

6. 环境准备完毕后,即可进入云主机,云主机桌面如下所示。

fddf4cd670f5498bbbc0072aada1db87

2.2 创建项目

CodeArts IDE是一个集成开发环境(IDE),兼具源代码编辑器的简易性和开发人员工具的强大功能,如代码补全和调试。它将精简的源代码编辑器与强大的开发者工具结合在一起。

1. 双击打开云主机桌面上的CodeArts IDE for Java(虽然IDE名称带有“Java”,但本案例用于编写JavaScript前端代码)。

06e48eb1912248f79475379c05390f74

2. 首次使用CodeArts IDE创建工程,可直接点击左侧栏目中的“新建工程”,项目名称为“LotteryDraw”(用户可自行取名),工程存放位置、构建系统及JDK直接使用默认配置,接着点击右下角蓝色的“创建”按钮。

f2c66f378a6f4b89a25421ff1eececc8

3. 在新建的“LotteryDraw”工程项目中,点击左上角新建HTML文件,输入:LotteryDraw.html,输入完毕后按下回车键,该HTML文件用于存放实现活动抽奖的代码,代码内容在后续文中进行介绍。

84d7b41176c74cf49feb89ebfcc9e14e

4. 代码编写完成后,鼠标右键单击“LotteryDraw.html”文件,选择“打开所在的文件夹”,找到活动抽奖项目在云主机中的位置。

8827617fddb842ffb41f275d35002982

5. 在打开的项目文件夹中,鼠标右键单击编写好的网页代码文件,依次点击“打开方式”>“Firefox”,在火狐浏览器中查看活动抽奖页面。

6815fbfc934d4715a5805a8ccc3a4be3

2.3 编写代码(LotteryDraw.html)

本案例由一个项目文件组成,文件名为LotteryDraw.html,由HTML + CSS + JS组件构成,其功能是在页面中央显示一个3×3的九宫格,每个格子展示不同的奖品信息。用户点击“开始抽奖”按钮后,奖品格子会有频率的振动,模拟抽奖过程。用户点击“停止抽奖”按钮后,抽奖停止并显示中奖结果,同时在“抽奖结果”区域显示中奖奖品的详细信息,完成一个完整的活动抽奖过程。此外,用户还可以通过滑块调整抽奖速度,增加互动性。系统还会在页面底部的“抽奖记录”区域显示最近四次的抽奖记录,方便用户查看历史抽奖结果。

活动抽奖网页参考代码如下(代码存放在LotteryDraw.html文件中):

(下面代码中有些图标无法显示,请到附件中查看完整代码):

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>活动抽奖</title>
    <meta name="viewport" content="width=device-width,initial-scale=1.0">
    <style>
        /* =========  基础变量  ========= */
        :root {
            --bg0: #0f0c1f;
            --bg1: #1a1142;
            --neonC: #00f9ff;
            --neonM: #ff00a0;
            --glass: rgba(255,255,255,.08);
            --radius: 16px;
        }
        * {
            margin: 0;
            padding: 0;
            box-sizing: border-box;
            font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif
        }
        body {
            min-height: 100vh;
            background: linear-gradient(135deg,var(--bg0),var(--bg1));
            color: #fff;
            display: flex;
            align-items: center;
            justify-content: center;
            padding: 20px;
            overflow-x: hidden;
        }
        .container {
            max-width: 800px;
            width: 100%;
            text-align: center;
            animation: fadeIn .6s
        }
        /* =========  标题  ========= */
        h1 {
            font-size: 2.6rem;
            letter-spacing: 2px;
            margin-bottom: .5em;
            background: linear-gradient(90deg,var(--neonC),var(--neonM));
            -webkit-background-clip: text;
            -webkit-text-fill-color: transparent;
            filter: drop-shadow(0 0 6px var(--neonC));
        }
        /* =========  九宫格  ========= */
        .grid-container {
            display: grid;
            grid-template-columns: repeat(3,1fr);
            gap: 18px;
            max-width: 480px;
            margin: 30px auto;
            padding: 20px;
            background: var(--glass);
            border-radius: var(--radius);
            backdrop-filter: blur(12px);
            border: 1px solid var(--neonC);
            box-shadow: 0 0 25px var(--neonC);
        }
        .grid-item {
            aspect-ratio: 1/1;
            background: var(--glass);
            border: 1px solid var(--neonC);
            border-radius: var(--radius);
            display: flex;
            flex-direction: column;
            justify-content: center;
            align-items: center;
            transition: .3s;
            position: relative;
            overflow: hidden;
        }
            .grid-item::before {
                content: '';
                position: absolute;
                inset: 0;
                background: linear-gradient(45deg,transparent,rgba(0,249,255,.15),transparent);
                opacity: 0;
                transition: .3s;
            }
            .grid-item.active {
                background: rgba(0,249,255,.25);
                transform: scale(1.08)
            }
            .grid-item.win {
                animation: pulse 1s infinite
            }
        @keyframes pulse {
            0%,100% {
                box-shadow: 0 0 10px var(--neonC)
            }
            50% {
                box-shadow: 0 0 30px var(--neonC),0 0 50px var(--neonM)
            }
        }
        .prize-icon {
            font-size: 2.8rem;
            margin-bottom: 6px
        }
        .prize-name {
            font-size: 1rem;
            font-weight: 600
        }
        .prize-value {
            font-size: .8rem;
            opacity: .7
        }
        /* =========  按钮  ========= */
        .button-container {
            display: flex;
            justify-content: center;
            gap: 20px;
            flex-wrap: wrap;
            margin: 25px 0;
        }
        .btn {
            padding: 14px 32px;
            font-size: 1rem;
            font-weight: 700;
            border: none;
            border-radius: 50px;
            cursor: pointer;
            transition: .3s;
            color: #fff;
            position: relative;
            overflow: hidden;
        }
            .btn:disabled {
                opacity: .4;
                cursor: not-allowed
            }
        .btn-start {
            background: linear-gradient(45deg,#00c853,#00ffa2)
        }
        .btn-stop {
            background: linear-gradient(45deg,#ff5252,#ff4081)
        }
        .btn-reset {
            background: linear-gradient(45deg,#2196f3,#21cbf3)
        }
        .btn:hover:not(:disabled) {
            transform: translateY(-4px);
            filter: brightness(1.2)
        }
        /* =========  结果卡片  ========= */
        .result-container, .speed-control, .history {
            background: var(--glass);
            border: 1px solid var(--neonC);
            border-radius: var(--radius);
            padding: 20px;
            margin: 20px auto;
            max-width: 480px;
            backdrop-filter: blur(12px);
        }
        .result-title {
            color: var(--neonC);
            font-size: 1.4rem;
            margin-bottom: 10px
        }
        .result-icon {
            font-size: 3.2rem;
            margin-bottom: 10px;
            animation: float 2s infinite alternate
        }
        @keyframes float {
            from {
                transform: translateY(0)
            }
            to {
                transform: translateY(-8px)
            }
        }
        /* =========  速度滑条  ========= */
        .slider-container {
            display: flex;
            align-items: center;
            gap: 10px
        }
        input[type=range] {
            flex: 1;
            -webkit-appearance: none;
            height: 6px;
            border-radius: 3px;
            background: var(--glass);
            outline: none
        }
            input[type=range]::-webkit-slider-thumb {
                -webkit-appearance: none;
                width: 18px;
                height: 18px;
                border-radius: 50%;
                background: var(--neonM);
                cursor: pointer
            }
        /* =========  抽奖记录(水平居中) ========= */
        .history h2 {
            color: var(--neonM);
            margin-bottom: 12px;
            font-size: 1.4rem
        }
        .history-list {
            display: flex;
            justify-content: center;
            gap: 12px;
            overflow-x: auto;
            padding-bottom: 6px;
        }
        .history-item {
            flex: 0 0 140px;
            background: var(--glass);
            border: 1px solid var(--neonM);
            border-radius: var(--radius);
            padding: 12px;
            display: flex;
            flex-direction: column;
            align-items: center;
            font-size: .85rem;
            min-width: 120px;
        }
        .history-icon {
            font-size: 1.8rem;
            margin-bottom: 6px
        }
        .history-name {
            font-weight: 600;
            margin-bottom: 4px
        }
        .history-time {
            opacity: .6;
            font-size: .75rem
        }
        /* =========  彩色纸屑(简化)  ========= */
        .confetti {
            position: fixed;
            width: 8px;
            height: 8px;
            top: -10px;
            border-radius: 50%;
            animation: fall 3s linear forwards;
        }
        @keyframes fall {
            to {
                transform: translateY(110vh) rotate(720deg);
                opacity: 0
            }
        }
    </style>
</head>
<body>
    <div class="container">
        <h1> 活动抽奖 </h1>
        <!-- 速度滑条 -->
        <div class="speed-control">
            <h3>抽奖速度控制</h3>
            <div class="slider-container">
                <input type="range" id="speedSlider" min="1" max="10" value="5">
                <span id="speedValue">5</span>
            </div>
        </div>
        <!-- 九宫格 -->
        <div class="grid-container">
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">游戏主机</div><div class="prize-value">¥2999</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">智能手机</div><div class="prize-value">¥3999</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">笔记本电脑</div><div class="prize-value">¥5999</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">无线耳机</div><div class="prize-value">¥899</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">智能手表</div><div class="prize-value">¥1299</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">神秘礼盒</div><div class="prize-value">¥1999</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">优惠券</div><div class="prize-value">满1000减200</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">现金红包</div><div class="prize-value">¥888</div></div>
            <div class="grid-item"><div class="prize-icon"></div><div class="prize-name">画板套装</div><div class="prize-value">¥599</div></div>
        </div>
        <!-- 按钮 -->
        <div class="button-container">
            <button class="btn btn-start" id="startButton">开始抽奖</button>
            <button class="btn btn-stop" id="stopButton" disabled>停止抽奖</button>
            <button class="btn btn-reset" id="resetButton">重新开始</button>
        </div>
        <!-- 结果 -->
        <div class="result-container">
            <div class="result-title">抽奖结果</div>
            <div class="result-content" id="resultContent">
                <div class="result-icon"></div>
                <div class="result-text">等待抽奖开始...</div>
            </div>
        </div>
        <!-- 抽奖记录(水平滚动) -->
        <div class="history">
            <h2>抽奖记录</h2>
            <div class="history-list" id="historyList">
                <div class="history-item">
                    <div class="history-icon"></div>
                    <div class="history-name">暂无记录</div>
                    <div class="history-time">--:--:--</div>
                </div>
            </div>
        </div>
    </div>
    <script>
        /* === 基本数据 === */
        const prizes = [...document.querySelectorAll('.grid-item')].map(el => ({
            name: el.querySelector('.prize-name').textContent,
            value: el.querySelector('.prize-value').textContent,
            icon: el.querySelector('.prize-icon').textContent
        }));
        const gridItems = document.querySelectorAll('.grid-item');
        const startBtn = document.getElementById('startButton');
        const stopBtn = document.getElementById('stopButton');
        const resetBtn = document.getElementById('resetButton');
        const resultBox = document.getElementById('resultContent');
        const historyBox = document.getElementById('historyList');
        const speedSlider = document.getElementById('speedSlider');
        const speedValue = document.getElementById('speedValue');
        let isSpin = false, current = 0, win = -1, timer = null;
        let base = 50, factor = 5;
        speedSlider.addEventListener('input', e => { factor = +e.target.value; speedValue.textContent = factor });
        const speed = () => Math.max(10, base / factor);
        /* === 功能函数 === */
        function reset() {
            clearTimeout(timer); isSpin = false; current = 0; win = -1;
            gridItems.forEach(el => el.classList.remove('active', 'win'));
            resultBox.innerHTML = '<div class="result-icon"></div><div class="result-text">等待抽奖开始...</div>';
            startBtn.disabled = false; stopBtn.disabled = true;
        }
        function start() {
            if (isSpin) return;
            reset(); isSpin = true; startBtn.disabled = true; stopBtn.disabled = false;
            win = Math.floor(Math.random() * 9);
            resultBox.innerHTML = '<div class="result-icon"></div><div class="result-text">抽奖进行中...</div>';
            run();
        }
        function run() {
            gridItems.forEach(el => el.classList.remove('active'));
            gridItems[current].classList.add('active');
            current = (current + 1) % 9;
            if (current === win && Math.random() > .7) { setTimeout(end, 100); return }
            timer = setTimeout(run, speed());
        }
        function stopLottery() {
            if (!isSpin) return;
            clearTimeout(timer); isSpin = false;
            if (win < 0) win = Math.floor(Math.random() * 9);
            gridItems.forEach(el => el.classList.remove('active'));
            gridItems[win].classList.add('active', 'win');
            showResult();
            startBtn.disabled = false; stopBtn.disabled = true;
        }
        function end() { stopLottery() }
        function showResult() {
            const p = prizes[win];
            resultBox.innerHTML = `
                <div class="result-icon">${p.icon}</div>
                <div class="result-text">
                    <div class="result-name">${p.name}</div>
                    <div class="result-value">${p.value}</div>
                </div>`;
            confetti(); addHistory(p);
        }
        function confetti() {
            const colors = ['#00f9ff', '#ff00a0', '#ffffff', '#00ffa2', '#ffea00'];
            for (let i = 0; i < 150; i++) {
                const c = document.createElement('div');
                c.className = 'confetti';
                c.style.left = Math.random() * 100 + 'vw';
                c.style.background = colors[Math.floor(Math.random() * colors.length)];
                c.style.width = c.style.height = Math.random() * 8 + 4 + 'px';
                document.body.appendChild(c);
                setTimeout(() => c.remove(), 3000);
            }
        }
        let history = [];
        function addHistory(p) {
            const t = new Date();
            history.unshift({ ...p, time: t.toLocaleTimeString() });
            if (history.length > 5) history.pop();
            updateHistory();
        }
        function updateHistory() {
            if (!history.length) {
                historyBox.innerHTML = '
                    <div class="history-item">
                        <div class="history-icon"></div>
                        <div class="history-name">暂无记录</div>
                        <div class="history-time">--:--:--</div>
                    </div>';
                return;
            }
            historyBox.innerHTML = '';
            history.forEach(h => {
                const d = document.createElement('div');
                d.className = 'history-item';
                d.innerHTML = `
                    <div class="history-icon">${h.icon}</div>
                    <div class="history-name">${h.name}</div>
                    <div class="history-time">${h.time}</div>`;
                historyBox.appendChild(d);
            });
        }
        /* === 事件 === */
        startBtn.addEventListener('click', start);
        stopBtn.addEventListener('click', stopLottery);
        resetBtn.addEventListener('click', reset);
        reset(); updateHistory();
    </script>
</body>
</html>

代码实现了一个活动抽奖网页,主要功能如下:

1. 页面总体布局

(1) 头部信息:显示活动抽奖的标题。

(2) 抽奖速度控制:提供抽奖速度控制滑块组件和速度值显示,用户可以通过滑动滑块来调整抽奖速度。

(3) 九宫格容器:以九宫格形式展示所有奖品,每个奖品都有独立的展示区域,包括奖品图标、名称和价值。

(4) 抽奖按钮区域:包括“开始抽奖”、“停止抽奖”和“重新开始”按钮,用于控制抽奖过程。

(5) 抽奖结果区域:展示抽奖结果,包括中奖奖品的图标、名称和价值。

(6) 抽奖历史记录:显示用户中奖的历史记录,包括奖品图标、名称、价值和抽奖时间。

2. 功能实现

(1) 抽奖控制

① 用户可以通过点击“开始抽奖”按钮来启动抽奖,在此过程中,九宫格中的所有奖品将有频率的振动。

② 用户可以通过点击“停止抽奖”按钮来停止抽奖,系统将会高亮显示一个奖品作为中奖结果。

③ 用户可以通过点击“重新开始”按钮来重置抽奖状态,重新开始抽奖。

(2) 抽奖速度控制

① 用户可以通过滑动“抽奖速度控制”滑块控制抽奖的速度,速度值范围为1—10,速度值越小抽奖的过程越慢。

(3) 抽奖结果显示          

① 根据抽奖结果动态渲染中奖奖品内容,中奖奖品在九宫格中会高亮显示,并在抽奖结果展示区显示奖品图标、名称和价值。

② 抽奖结果展示区支持彩色纸屑飘落动画效果,增加抽奖的趣味性和视觉效果。

(4) 抽奖记录展示

① 抽奖记录区会展示最近四次的抽奖记录,包括奖品图标、名称、价值和抽奖时间,方便用户查看历史中奖记录。

3. 技术要点

(1) HTML结构:页面整体结构简洁明了,核心是“九宫格”布局。每个格子由<div class="grid-item">表示,内部包含奖品图标(<div class="prize-icon">)奖品名称(<div class="prize-name">)和奖品价值(<div class="prize-value">),清晰地展示了九宫格中每个奖品的信息。

(2) CSS布局:通过grid-template-columns: repeat(3,1fr)将容器分为三列,每列宽度相等。每个grid-item设置为正方形(aspect-ratio: 1/1),并使用flexbox布局确保内容居中显示。同时,使用border-radiusbox-shadow等属性让每个格子看起来更加地立体和美观。

(3) DOM 操作:使用innerHTML动态设置抽奖结果区域的内容,展示奖品图标、名称和价值;通过innerHTML初始化抽奖记录列表,并使用document.createElementappendChild更新抽奖记录;在中奖时,使用document.createElement动态创建彩色纸屑元素,并通过style属性设置其样式,然后使用appendChild将纸屑元素添加到页面中,并在一定时间后通过remove方法移除。

(4) 事件监听:为“开始抽奖”按钮添加点击事件,调用start函数开始抽奖;为“停止抽奖”按钮添加点击事件,调用stopLottery函数停止抽奖;为“重新开始”按钮添加点击事件,调用reset函数重置抽奖状态。

2.4 展示活动抽奖网页

1. 活动抽奖网页展示

acbec4ebc80c416a960b962fde7758a7

至此本次实验全部内容完成。

2.5 拓展实验

1. 增强交互性:比如增加轮次选择功能,支持多轮抽奖(用户可以选择抽奖的轮次,每轮抽奖可以有不同的奖品,每轮抽奖结束后自动进入下一轮)、增加抽奖音效(当点击“开始抽奖”、“停止抽奖”、“重置抽奖”按钮时播放不同的音效)。

2. 活动抽奖功能扩展:比如增加抽奖统计功能(显示每个奖品的中奖次数,中奖率等)、增加分享功能(允许用户将中奖信息进行分享)。

3. 欢迎自由发挥


3 释放资源

3.1 关闭云主机

1. 首先点击云主机桌面上面的“关机”按钮,然后在弹出来的提示框点击“确定”按钮,关闭云主机。

ad127a796542429e87dd0b8a6ed205b5

2. 点击“确定”按钮后页面自动跳转到开发者空间页面,可以看到我的云主机显示“关机中”,说明云主机正在关机。

780a55cb2fc14d14bf6f3e83ea2aa147

3.2 检查资源

云主机关机后,等待3至5分钟,当我的云主机显示“已就绪”时,说明云主机成功关机,下次使用云主机只需点击“打开云主机”>“进入桌面”即可,可参考2.1小节。      

70fd1cbdc4b84c1a9f9fe04d95f0010d

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值