最近的项目需要在cubieboard3上布署一个web server,用于与设备进行交互,实现通过移动设备的网页控制设备的功能。web server选用了goahead 3.4.3。
GoAHead的前端页面的实现有两种方式:
一种是所有的HTML代码全部由C语言实现。
另一种是一种类似于asp风格的方式,静态相关的网页由HTML编辑器实现,交互部分采用类似这种<%GetTermTime();%>宏的方式实现。在客户端网页请求Goahead服务器网页的时候,GoAHead服务器网页浏览整个网页文件,看到这种宏,就用服务端的一段代码进行替换,最后展现在客户端前面的页面就是替换过的网页文件。
综合比较了一下,还是选用了第二种。第一种方式C语言的体力活太多,而且非常不容易维护,扩展性也比较差,不具有通用性。
项目的交互主要分两个方面:
所谓的后台也就是goahead的c语言处理部分。
1.修改后缀名
在src/route.txt中 :
把 route uri=/
改成route uri=/ extensions=jst,html handler=jst
这样前端访问的网页就支持.jst和.html两种格式了,否则默认是.jst的格式。如果客户端访问GoAhead服务器上的html文件,GoAhead服务器在浏览页面的时候,是不会用服务器端的代码去替换宏的,也就是没办法去获取后台数据。
2.我们用来测试的页面叫做settime.jst,代码如下:
<HTML>
<HEAD>
<TITLE>Time Setting</TITLE>
<SCRIPT language=JavaScript>
function js_getObjectByNameFromForm(form,name)
{
for(var i=0;i<form.length;i++)
{
if(form[i].name == name)
return (form[i]);
}
return (null);
}
function clickedSetButton(form)
{
var date;
var year;
var mon;
var day;
var hour;
var min;
var sec;
year = js_getObjectByNameFromForm(form,"year");
mon = js_getObjectByNameFromForm(form,"month");
day = js_getObjectByNameFromForm(form,"day");
hour = js_getObjectByNameFromForm(form,"hour");
min = js_getObjectByNameFromForm(form,"minute");
sec = js_getObjectByNameFromForm(form,"second");
date = new Date();
year.value=date.getFullYear();
mon.value =date.getMonth()+1;
day.value =date.getDate();
hour.value=date.getHours();
min.value =date.getMinutes();
sec.value =date.getSeconds();
return false;
}
</SCRIPT>
</HEAD>
<BODY text=#000000 vLink=#000080 alink=#000080 link=#000080 bgColor=#FFFFFF onload="">
<TABLE cellSpacing=0 cellPadding=0 width="100%" border=0>
<TBODY>
<TR align=left>
<TD vAlign=bottom>
<FONT size=+2 >
<STRONG>
<h1>设置终端时间</h1>
</STRONG>
</FONT>
</TD>
</TR>
</TBODY>
</TABLE>
<P>点击“获取终端时间”可查看终端时间,点击“获取本地时间”可得到当前系统时间 “设置终端时间”,用当前各输入框内的数据更新终端时间,即时生效
</P>
<FORM id="" action=/goform/formSetTermTime method=POST target="id_iframe">
<TABLE>
<TBODY>
<TR>
<%GetTermTime();%>
</TR>
</TBODY>
</TABLE>
<BR>
<INPUT onClick="location.reload()" type=button value=获取终端时间 name=reloadp >
<INPUT onClick="" type=submit value=设置终端时间 name=Modify>
<INPUT onClick="clickedSetButton(form);" type=button value="获取本地时间" name=B2>
</FORM>
<iframe id="id_iframe" name="id_iframe" style="display:none;"></iframe>
</BODY>
</HTML>
settime.jst中的GetTermTime和formSetTermTime ,就对应下面http.c中新增的aspGetTermTime和formSetTermTime
2.http.c新增如下代码:
int aspGetTermTime(int eid,Webs *wp,int argc,char **argv)
{
time_t ti;
struct tm *timeinfo;
time(&ti);
timeinfo=localtime(&ti);
websWrite(wp,"<TD noWrap><INPUT maxlength=4 size=6 title=时间年 value=%d name=year>年</TD>\n",1900+timeinfo->tm_year);
websWrite(wp,"<TD noWrap><INPUT maxlength=2 size=4 title=时间月 value=%d name=month>月</TD>\n",1+timeinfo->tm_mon);
websWrite(wp,"<TD noWrap><INPUT maxlength=2 size=4 title=时间日 value=%d name=day>日</TD>\n",timeinfo->tm_mday);
websWrite(wp,"<TD noWrap><INPUT maxlength=2 size=4 title=时间时 value=%d name=hour>点</TD>\n",timeinfo->tm_hour);
websWrite(wp,"<TD noWrap><INPUT maxlength=2 size=4 title=时间分 value=%d name=minute>分</TD>\n",timeinfo->tm_min);
websWrite(wp,"<TD noWrap><INPUT maxlength=2 size=4 title=时间秒 value=%d name=second>秒</TD>\n",timeinfo->tm_sec);
return 1;
}
void formSetTermTime(Webs *wp,char *path,char *query)
{
char * ok;
bool bError = 0;
char sText[50];
struct tm *timeinfo;
time_t ti;
ti =time(NULL);
timeinfo=localtime(&ti);
a_assert(wp);
ok = websGetVar(wp,"year","");
websHeader(wp);
if(0==strcmp(ok,"取消"))
websWrite(wp,"取消");
else
{
ok = websGetVar(wp,"year","");
timeinfo->tm_year=atoi(ok)-1900;
ok = websGetVar(wp,"month","");
timeinfo->tm_mon=atoi(ok)-1;
ok = websGetVar(wp,"day","");
timeinfo->tm_mday=atoi(ok);
ok = websGetVar(wp,"hour","");
timeinfo->tm_hour=atoi(ok);
ok = websGetVar(wp,"minute","");
timeinfo->tm_min=atoi(ok);
ok = websGetVar(wp,"second","");
timeinfo->tm_sec=atoi(ok);
struct timeval tv;
struct timezone tz;
gettimeofday(&tv,&tz);
tv.tv_sec=mktime(timeinfo);
if(0!=settimeofday(&tv,&tz))
{
logmsg(2,"#####################settimeofday failed!");
}
else
SerialMain(0,"tang");
websFooter(wp);
websDone(wp);
logmsg(2,"#######%d %d/%d %d:%d:%d",timeinfo->tm_year,timeinfo->tm_mon,timeinfo->tm_mday,timeinfo->tm_hour,timeinfo->tm_min,timeinfo->tm_sec);
}
以上的函数需要在websJstOpen中定义,如下所示
PUBLIC int websJstOpen()
{
websJstFunctions = hashCreate(WEBS_HASH_INIT * 2);
websDefineJst("write", websJstWrite);
websDefineJst("GetTermTime",aspGetTermTime);
websFormDefine("formSetTermTime",formSetTermTime);
websDefineHandler("jst", 0, jstHandler, closeJst, 0);
return 0;
}
备注:参考了http://blog.chinaunix.net/uid-23412956-id-4044201.html,看到两个写好的应答函数,一起拿过来用。
void websReportOK(webs_t wp, const char *url)
{
websHeader(wp);
websWrite(wp,("<script>location.href='../%s'</script>"), url);
websFooter(wp);
websSetStatus(wp, 200);
websDone(wp);
}
void websReportError(webs_t wp, const char *dsc, const char *url)
{
websHeader(wp);
websWrite(wp,("<script>alert('%s');history.back();</script>"), dsc);
websFooter(wp);
websSetStatus(wp, 200);
websDone(wp);
}
3.到目前为止,通过<%GetTermTime();%>得到服务端的数据,它对应的C函数是aspGetTermTime。
通过form的action="goform/formSetTermTime "实现将网页的数据提交到后台的formSetTermTime。正常情况下这时页面会跳转到goform/formSetTermTime 的页面。这是提交刷新的方式。
我这边要讲的是form提交但不刷新本页的方式。
方法如下,修改settime.jst文件:
<FORM id="" action=/goform/formSetTermTime method=POST target="id_iframe">
.................................
<INPUT onClick="" type=submit value=设置终端时间 name=Modify>
</FORM>
<iframe id="id_iframe" name="id_iframe" style="display:none;"></iframe>
这在网页设计上叫提交到隐藏的iframe页面,实现form提交但不刷新的目的。
注意事项:iframe 的id与Form的target要一致。
iframe的style要设置成display:none
相关资料参考:可以百度一下以下关键字,不过这些GoAhead的版本都比3.4.3低,有些API的用法不太一样,大家看的时候可以选择性的看。