从0使用TCP手撸http服务器六

html模板设计:

        上篇我们说到通过路由去返回不同的html页面,如果每一个页面都是一个数组的话,那么我们可能需要很多数组,里面很多内容都是一样的,这样子会浪费我们单片机很多flash,我们需要把共同的部分固定出来,每个页面不同的部分才定义一个数组。

        要想设计html模板,我们的先了解html的结构,更加具体的可以百度查阅。我这边把一个html大致分成一下部分:
    -模板文件

    0-html标准格式
    1-style
    2-导航
    3-body
    4-js
对于一个页面,其实html标准格式是不需要变化的,style,导航,body,js可能需要变化,于是我们可以设计一个标准的匹配模板:里面的%s就代表着style,导航,body,js    

//模板页面
static char *page_template="<!DOCTYPE html>\
<html>\
<head>\
<title>webserver</title>\
%s\
</head>\
<meta charset=\"utf-8\">\ 
<body>\
%s\
%s\
%s\
</body>\
</html>";

有了模板后,我们需要分别定义出style,导航,body,js ,JS还没用的,可以先不定义

style设计:

//style
static char *html_style="<style> \
.button {\
background-color: #4CAF50;\
border: none;\
color: white;\
padding: 15px 15px;\
text-align: center;\
text-decoration: none;\
display: inline-block;\
font-size: 16px;\
margin: 4px;\
border-radius: 12px;\
width:30%;\
height:200px;\
}\
.button1 {\
background-color: #AAA220;\
border: none;\
color: white;\
padding: 15px 15px;\
text-align: center;\
text-decoration: none;\
display: inline-block;\
font-size: 16px;\
margin: 4px;\
border-radius: 12px;\
width:18%;\
}\
progress {\
    width: 200px;\
	height:30px;\
}\
</style>";

导航设计:

//导航
static char *html_navigation="<div style='width:100%;height:1000px;'>\
<div style='background-color:#4CAFA0;text-align:center;height:150px;border-radius: 12px;'>\
<h1>嵌入式设备</h1>\
<div style='text-align: center;'>\
<button class='button1' onclick=\"window.location.href = '/'\">首页</button>\
<button class='button1' onclick=\"window.location.href = 'info'\">信息</button>\
<button class='button1' onclick=\"window.location.href = 'para'\">参数</button>\
<button class='button1' onclick=\"window.location.href = 'debug'\">调试</button>\
<button class='button1' onclick=\"window.location.href = 'ota'\">升级</button>\
</div>\
</div>";

4个body设计:

//index的body
static char *html_index_body="<div style='text-align: center;'>\
<button class='button'>功能1</button>\
<button class='button'>功能2</button>\
<button class='button'>功能3</button>\
</div>\
<div style='text-align: center;'>\
<button class='button'>功能4</button>\
<button class='button'>功能5</button>\
<button class='button'>功能6</button>\
</div>\
<div style='text-align: center;'>\
<button class='button'>功能7</button>\
<button class='button'>功能8</button>\
<button class='button'>功能9</button>\
</div>";
//info的body
static char *html_info_body="<div style='text-align:center;'>\
<p style='font-size: 26px;color:red;'>信息开发中</p>\
</div>";
//para的body
static char *html_para_body="<div style='text-align:center;'>\
<p style='font-size: 26px;color:red;'>参数开发中</p>\
</div>";
//debug的body
static char *html_debug_body="<div style='text-align:center;'>\
<p style='font-size: 26px;color:red;'>调试开发中</p>\
</div>";
//OTA的body
static char *html_ota_body="<div style='text-align:center;'>\
<p style='font-size: 26px;color:red;'>只能选择OTA文件</p>\
<p>\
<label style='font-size: 26px;' for='file'>完成度:</label>\
<progress max='100' value='70'> 70% </progress>\
</p>\
<form action='/otafile' method='post' enctype='multipart/form-data'>\
<div style='text-align:center;'>\
<input class='button1' style='text-align:center; background-color: #4CAF50;' type='file' accept='*' name='ota'>\
<input class='button1' style='text-align:center; background-color: #4CAF50;' type='submit' value='开始升级'>\
</div>\
</form>\
</div>\
</div>";

我们有了数据后,需要把数据通过模板构建成一个完整的html。先设计一个html文件缓存数组,这个数组代表着你能返回多大的页面。

static u8 html_file[4*1024]={0};

定义构建页面的函数,js部分没有定义,全部为空("")

//生成index页面
static void create_index_page()
{
	memset(html_file,0,strlen((char*)html_file));
	sprintf((char*)html_file,page_template,html_style,html_navigation,html_index_body,"");
	
}
static void create_info_page()
{
	memset(html_file,0,strlen((char*)html_file));
	sprintf((char*)html_file,page_template,html_style,html_navigation,html_info_body,"");
	
}
static void create_para_page()
{
	memset(html_file,0,strlen((char*)html_file));
	sprintf((char*)html_file,page_template,html_style,html_navigation,html_para_body,"");
	
}
static void create_debug_page()
{
	memset(html_file,0,strlen((char*)html_file));
	sprintf((char*)html_file,page_template,html_style,html_navigation,html_debug_body,"");
	
}
//生成ota页面
static void create_ota_page()
{
	memset(html_file,0,strlen((char*)html_file));
	sprintf((char*)html_file,page_template,html_style,html_navigation,html_ota_body,"");
	
}

下面就是可以通过路由反馈不同页面的,和文章1一样的效果。

                    if(strcmp("/ota",route)==0)
					{

						create_ota_page();
						send_html(remote_sock,(char *)html_file);
					}
					if(strcmp("/",route)==0)
					{
						create_index_page();
						send_html(remote_sock,(char *)html_file);
						
					}
					if(strcmp("/info",route)==0)
					{
						create_info_page();
						send_html(remote_sock,(char *)html_file);
						
					}
					if(strcmp("/para",route)==0)
					{
						create_para_page();
						send_html(remote_sock,(char *)html_file);
						
					}
					if(strcmp("/debug",route)==0)
					{
						create_debug_page();
						send_html(remote_sock,(char *)html_file);
						
					}

效果:点击不同的导航进入不同的页面

源码:

https://download.csdn.net/download/HES_C/87620599

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/5079.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

神经网络之反向传播算法(加入Nesterov动量的误差反向传播算法)

文章目录1、Nesterov动量2、结合Nesterov动量的误差反向传播算法3、算法实现3.1 训练过程3.2 测试过程4、参考源码及数据集1、Nesterov动量 在动量法提出不久之后&#xff0c;Nesterov动量也随之被提了出来&#xff0c;此方法属于动量法的进一步发展&#xff0c;与动量法不同的…

基于windows11配置深度学习环境包含WSL2配置ubuntu20.04

基于windows11配置深度学习环境包含WSL2配置ubuntu20.04目录平台WSL2 系统准备(Windows Subsystem of Linux)安装WSL2基于WSL2中的Ubuntu安装CUDA和cudnnCUDA 安装目录 平台 系统 : windows11 专业版 CPU Intel I7 8750hq 显卡&#xff1a;Nvidia GTX1060移动端 显卡驱动版本…

安装Windows11提示这台电脑不符合安装此版本的Windows所需的最低系统要求

现在很多用户都会选择用U盘来安装系统&#xff0c;最新有用户在使用U盘安装Win11系统的时候&#xff0c;结果安装到第一步就提示这台电脑无法运行Windows11&#xff0c;这台电脑不符合安装此版本的Windows所需的最低系统要求。下面小编就来教大家解决此问题的方法。 问题解析&a…

BIO/NIO/AIO/IO多路复用简介

bio、nio、aio、io多路复用、reactor模式io&#xff0c;在将IO的时候&#xff0c;是不是都遇到过这些概念&#xff0c;也有种傻傻分不清&#xff1f;甚至别人在大谈特谈的时候&#xff0c;一会nio&#xff0c;一会io多路复用&#xff0c;一会又搞到reactor模式上去了&#xff1…

一文搞懂原型和原型链

在了解原型和原型链之前首先得明确它俩是什么东西&#xff1a; 原型&#xff1a;prototype 又称显示原型 1、原型是一个普通对象 2、只有构造函数才具备该属性 3、公有属性可操作 隐式原型&#xff1a;__proto__ 1、只有对象(普通对象、函数对象&#xff09;具备 2、私有的对…

《Spring系列》第2章 解析XML获取Bean

一、基础代码 Spring加载bean实例的代码 public static void main(String[] args) throws IOException {// 1.获取资源Resource resource new ClassPathResource("bean.xml");// 2.获取BeanFactoryDefaultListableBeanFactory factory new DefaultListableBeanFa…

Airtest自动化测试工具实战演练

一开始知道Airtest大概是在年初的时候&#xff0c;当时&#xff0c;看了一下官方的文档&#xff0c;大概是类似Sikuli的一个工具&#xff0c;主要用来做游戏自动化的&#xff0c;通过截图的方式用来解决游戏自动化测试的难题。最近&#xff0c;移动端测试的同事尝试用它的poco库…

iwebsec靶场-命令执行漏洞

漏洞简介 命令执行漏洞&#xff08;Command Injection&#xff09;是一种常见的安全漏洞&#xff0c;也被称为代码注入漏洞。它允许攻击者将恶意代码注入到受攻击的应用程序中&#xff0c;从而可以在应用程序的上下文中执行任意命令。 命令执行漏洞通常出现在Web应用程序中&…

好的表单设计应该遵循什么规则?

在数字化时代&#xff0c;表单已经成为了人们生活中不可或缺的一部分。它们可能是网站注册表格、调查问卷、订单表格或者其他类型的表格。无论表单的类型是什么&#xff0c;都必须经过精心设计才能提供良好的用户体验。在本文中&#xff0c;我们将探讨如何设计一份用户体验好的…

Redis缓存双写一致性

目录双写一致性Redis与Mysql双写一致性canal配置流程代码案例双写一致性理解缓存操作细分缓存一致性多种更新策略挂牌报错,凌晨升级先更新数据库,在更新缓存先删除缓存,在更新数据库先更新数据库,在删除缓存延迟双删策略总结双写一致性 Redis与Mysql双写一致性 canal 主要是…

低代码开发公司:用科技强力开启产业分工新时代!

实现办公自动化&#xff0c;是不少企业的共同追求。低代码开发公司会遵循时代发展规律&#xff0c;注入强劲的科技新生力量&#xff0c;在低代码开发市场厚积爆发、努力奋斗&#xff0c;推动企业数字化转型升级&#xff0c;为每一个企业的办公自动化升级创新贡献应有的力量。 一…

【数据结构与算法】堆的实现(附源码)

目录 一.堆的概念及结构 二.接口实现 A.初始化 Heapinit 销毁 Heapdestroy B.插入 Heappush 向上调整 AdjustUp 1.Heappush 2.AdjustUp C.删除 Heappop 向下调整 AdjustDown D.堆的判空 Heapempty 堆顶数据 Heaptop 堆的大小 Heapsize 三.源码 Heap.h He…

【模板】带权并查集

文章目录1. 奇偶游戏2. 银河英雄传说1. 奇偶游戏 239. 奇偶游戏 题意&#xff1a; 依次给出多个区间的含 111 的个数的奇偶性&#xff0c;找出第一个不符合的答案的回答。 思路&#xff1a; 已知区间[a,b][a,b][a,b][b,c][b,c][b,c]的奇偶性&#xff0c;那么具有传递性&…

分享一个国内可用的免费ChatGPT网站(自己写的)

背景 ChatGPT作为一种基于人工智能技术的自然语言处理工具&#xff0c;近期的热度直接沸腾&#x1f30b;。 作为一个程序员&#xff0c;我也忍不住做了一个基于ChatGPT的网站&#xff0c;免费&#xff01;免登陆&#xff01;&#xff01;国内可直接对话ChatGPT&#xff0c;也…

10.线性表代码实战

10.1 与408关联解析及本节内容介绍 链表比顺序表出现的顺序更加的频繁。 10.2线性表地顺序表示原理解析 线性表的特点&#xff1a; &#xff08;1&#xff09;表中的元素的个数是有限的 &#xff08;2&#xff09;表中元素的数据类型相同。意味着每一个元素占用相同大小的空…

使用Dism++和360安全卫士搞定Windows10离线升级

Windows10有很多版本&#xff0c;常见的由1903、1909、20H1、21H2等&#xff0c;在离线状态下&#xff0c;很难下载到匹配的升级补丁。期间尝试多种方法均失败&#xff0c;最后用Dism和360安全卫士组合拳搞定。 1、使用下载补丁&#xff0c;升级失败 比如这里介绍了常见补丁&a…

【SL101】 传感器接入chirpstack平台

【SL101】 传感器接入chirpstack平台使用硬件SL100工程师答疑chirpstack 中 net-server 使能 80-87 频段网关开启80-87 频段设备传感器端配置频点连接成功测试结果---chirpstackSL100系列温湿度传感器产品&#xff08;墨水屏版&#xff09;接入chirpstack 平台笔记记录 使用硬件…

mysql学习之数据系统概述

☀️马上要成为打工人&#xff0c;这几天把前面的知识都捡了捡&#xff0c;发现自己对关系数据库这块的学习还有所缺失&#xff0c;于是本章开始学习mysql 这里写目录标题1. 数据库系统的发展1.1 人工管理阶段1.2 文件系统阶段1.3 数据库阶段1.4 大数据阶段2 数据库系统的组成2…

了解这7个Node.js库,让你的开发效率提升不止一点点

Node.js是一个流行的JavaScript运行时环境&#xff0c;拥有庞大的生态系统和丰富的库&#xff0c;使得在Node.js上构建高效、可靠的应用程序变得非常容易。在这篇文章中&#xff0c;我们将分享七个有用的Node.js库&#xff0c;它们可以提高您的工作效率&#xff0c;让您更轻松地…

android:手搓一个即时消息聊天框(包含消息记录)

先看一下效果 1.后端 要实现这个&#xff0c;先说一下后端要实现的接口 1.创建会话id 传入“发送id”和“接收id”给服务端&#xff0c;服务端去创建“会话id” 比如 get请求&#xff1a;http://xxxx:8110/picasso/createSession?fromUserId1&toUserId2 返回seesionId…