基于HTTP可供浏览器调用的本地打印程序
之前给公司做打印都是用ActiveX控件,只支持IE浏览器,最近需要支持谷歌,又不想去学谷歌插件编写,于是就用本地启动一个http服务器来供浏览器调用(写成windows服务更好),同事用了都说好(笑)!为了方便大家使用,重新简单的封装了一下,源码下载:点我下载
源码简单的封装了一个webserver(基于httplistener,可以使用Nancy框架代替),可以大致了解HTTP服务器处理的流程:APACHE,NINGX等服务器主要负责响应浏览器HTTP(基于SOCKET)的请求,并将请求转交给JAVA,C#,PHP等语言处理(应该就是启动了一个该语言的虚拟机解析执行对应的代码),再将处理结果返回给浏览器
namespace LocalPrint.Web
{
//本地微型HTTP服务器,可以理解网络请求处理的服务器处理流程
class WebServer
{
HttpListener httpListener;
public string Err;
Dictionary<string, IHttpHandler> handlerMap = new Dictionary<string, IHttpHandler>();
//启动本地HTTP服务器,相当于APACHE,NGINX之类的WEB服务器,接受浏览器发送过来的SOCKET请求
public bool RunWeb(string url)
{
try
{
httpListener = new HttpListener();
httpListener.Prefixes.Add(url);
httpListener.Start();
var th = new Thread(Process);
th.IsBackground = true;
th.Start();
return true;
}
catch (Exception ex)
{
Err = "启动本地服务器出现问题:" + ex.Message;
return false;
}
}
//设置相应路由和相应的处理类
public bool AddHandler(string url,IHttpHandler httpHandler)
{
try
{
handlerMap.Add(url, httpHandler);
return true;
}
catch (Exception ex)
{
Err = "添加路径出错:" + ex.Message;
return false;
}
}
//HTTP数据处理,这一部分相当于C#,JAVA和PHP和其他语言的功能,WEB服务器将请求转发给相应的语言处理,并把结果返回给浏览器
void Process()
{
for (; ; )
{
var cnx = httpListener.GetContext();//获取浏览器请求上下文,串行处理,也可以改成并行
var req = cnx.Request;
var rep = cnx.Response;
rep.ContentEncoding = Encoding.UTF8;
rep.Headers.Add("Access-Control-Allow-Origin", "*");//允许浏览器跨域!非常重要
rep.StatusCode = ;
var ret = "pag not found";
//rep.ContentType = "text";//返回内容,这里为text,ajax里面的请求datatype也需要设置为text或者html,不然会为null
//这一部分其实就是大部分MVC网络框架里的路由部分!这里简单的发送原始文本给handler处理
foreach (var kv in handlerMap)
{
if(System.Text.RegularExpressions.Regex.IsMatch(req.RawUrl,kv.Key))//正则匹配
{
var data = "";
if (req.HttpMethod == "GET")
data = req.RawUrl;//不做任何处理,直接将原始的http请求转发到handler。。。
else
using (var r = new StreamReader(req.InputStream, Encoding.UTF8))
{
data = r.ReadToEnd();
}
ret = kv.Value.Handler(data);
rep.StatusCode = ;//ok
break;
}
}
//返回处理结果给浏览器
using (var w = new StreamWriter(rep.OutputStream, Encoding.UTF8))
{
w.WriteLine(ret);
}
}
} }
}
void Process()
{
for(; ; )
{
var cnx = httpListener.GetContext();
var req = cnx.Request;
var rep = cnx.Response;
rep.Headers.Add("Access-Control-Allow-Origin", "*");
var txt = "";
if (req.HttpMethod == "POST")
{
var fp = "";
using (var r = new StreamReader(req.InputStream, Encoding.UTF8))
{
fp = r.ReadToEnd();
}
var printer = req.QueryString["printer"];
switch (req.Url.LocalPath)
{
case "/printZPP":
printComp1.Print(fp, printer);
break;
case "/printTJBB":
printComp1.PrintBB(fp, printer);
break;
case "/printQD":
printComp1.PrintQd(fp, printer);
break;
case "/printAll":
printComp1.UniversalPrint(fp, printer);
break;
default:
//rep.StatusCode = 404;
txt = "";
break;
}
}
else
{
switch (req.Url.LocalPath)
{
case "/GetPrintNames":
txt = printComp1.GetPrinterNames();
break;
case "/GetPrinter":
{
var id = req.QueryString["id"];
txt = printComp1.GetLocalPrinter(id);
}
break;
case "/SetPrinter":
{
var id = req.QueryString["id"];
var printer = req.QueryString["printer"];
printComp1.SetLocalPrinter(id, printer);
}
break;
default:
txt = printComp1.GetPrinterNames();
//rep.StatusCode = 404;
txt = "";
break;
}
} using (var w = new StreamWriter(rep.OutputStream, Encoding.UTF8))
{
w.WriteLine(txt);
}
}
}
HTTP处理接口,PrintHandler派生此接口处理打印任务,也可以派生接口执行其他本地任务,只是简单的处理get和post文本,没有具体的参数解析,需要类自行解析。。。
//类似于MVC框架的ACTION或者CONTROLER,可派生此接口实现其他操作
public interface IHttpHandler
{
string Handler(string txt);//简单的处理文本消息
}
IPrint打印接口,派生此接口执行具体打印任务,打印格式这块可以封装出一整套的处理代码(常用的标签,表格打印和格式设置),有空再处理了
//打印接口,需要打印格式,派生此接口即可
public interface IPrint
{
bool Print(Graphics g);//返回值为是否还有打印需要打印的页内容,即是否打印结束
}
启动本地服务器后,浏览器直接访问或ajax跨域访问本地服务器即可调用本地处理服务!
$.ajax({
url : "http://localhost:23333/print",
type : "POST",
contentType: "application/json;charset=utf-8",
data : "{'name':'大萝卜卜','age':32,'sex':true}",
dataType : "text",
success : function(result) {
if (result == "ok") {
alert("打印成功!");
} else {
alert("打印出错:"+result )
}
},
error:function(msg){
aler('Error:'+msg);
}
})
本地测试
var web = new WebClient();
var txt = textBox1.Text;
byte[] bytearray = Encoding.UTF8.GetBytes(txt);
txt = Convert.ToBase64String(bytearray);
txt = System.Web.HttpUtility.UrlEncode(txt, Encoding.UTF8);//必须经过url编码
web.Encoding = Encoding.UTF8;
web.Headers.Add("ContentType", "application/x-www-form-urlencoded"); try
{
var ret = web.UploadString("http://127.0.0.1:1234", "POST", "args=" + txt);
textBox2.Text = Encoding.UTF8.GetString(Convert.FromBase64String(ret)); }
catch (WebException ex)
{
var rsp = ex.Response as HttpWebResponse;
if(rsp !=null)
{
var code = (int)rsp.StatusCode;
textBox2.Text = code + "\r\n";
var stream = rsp.GetResponseStream();
using (var s = new StreamReader(stream))
{
textBox2.Text += s.ReadToEnd();
}
}
else
{
textBox2.Text = ex.Message; }
}
为了发表博客,花了2个小时,把代码整理,简单的封装成接口,平常写代码哪有这么费事,分分钟写完交差了事(意大利面条,笑),写文章不易,点个赞吧。
基于HTTP可供浏览器调用的本地打印程序的更多相关文章
- 学习笔记:URL Protocol在浏览器中打开本地应用程序
看到阿里的网站上可以通过点击卖家的旺旺图标从而调用本地的阿里旺旺程序,而且还可以传递当前浏览者需要咨询的商品.这是怎么实现的呢?是通过URLProtocol来完成. 原理还没有太清楚,即在系统里注册一 ...
- chrome浏览器插件启动本地应用程序
chrome浏览器插件启动本地应用程序 2014-04-20 00:04:30| 分类: 浏览器插件|举报|字号 订阅 下载LOFTER我的照片书 | chrome的插件开发这里就 ...
- 一个实现浏览器网页与本地程序之间进行双向调用的轻量级、强兼容、可扩展的插件开发平台—PluginOK中间件
通过PluginOK中间件插件平台(原名本网通WebRunLocal)可实现在网页中的JavaScript脚本无障碍访问本地电脑的硬件.调用本地系统的API及相关组件,同时可彻底解决ActiveX组件 ...
- ubuntu下浏览器调用本地应用程序
ubunut下浏览器调用本地应用程序需要desktop文件和scheme协议的支持,和windows 的url protocol类似,只是注册协议的方式不一样. 首先是desktop文件,里面需要加入 ...
- mac 下基于firebreath 开发多浏览器支持的浏览器插件
mac 下基于firebreath 开发多浏览器支持的浏览器插件 首先要区分什么是浏览器扩展和浏览器插件;插件可以像本地程序一样做的更多 一. 关于 firebreath http://www.fir ...
- 可以供MFC调用的,QT实现的DLL(使用qt-solutions的qtwinmigrate实现)
MFC和QT的消息循环机制不同,所以,要让QT写的DLL可以供MFC调用,要做一点特殊的处理 #include <qmfcapp.h> #include <qwinwidget.h& ...
- 可以供MFC调用的,QT实现的DLL(qtwinmigrate实现)
MFC和QT的消息循环机制不同,所以,要让QT写的DLL可以供MFC调用,要做一点特殊的处理 #include <qmfcapp.h> #include <qwinwidget.h& ...
- asp.net core系列 58 IS4 基于浏览器的JavaScript客户端应用程序
一. 概述 本篇探讨使用"基于浏览器的JavaScript客户端应用程序".与上篇实现功能一样,只不过这篇使用JavaScript作为客户端程序,而非core mvc的后台代码Ht ...
- python selenium中如何测试360等基于chrome内核的浏览器
转自:https://blog.csdn.net/five3/article/details/50013159 直接上代码,注意是基于chrome内核的浏览器,基于ie的请替换其中的chrome方法为 ...
随机推荐
- 商品描述里包含了英文双引号,ERP无法同步菜品信息
1. 2.因菜品描述里包含英文双引号,破坏了json格式,导致json格式错乱,ERP无法解析,所以无法同步数据.
- python之模块使用
1.入口 """ 模块测试入口 """ import show_message as sm # 导入方式一 sm.show(sm.__nam ...
- Greenplum hostname和address不一致导致配置文件无法加载
最近又遇到了几个坑,逐一记录分析下. 1.主机名hostname和address不一致 在又一次部署压测环境交由测试组进行压测时,同事修改了pg_hba.conf文件重新加载配置文件时报错.(找不到l ...
- SQL Server的实例恢复解析
同Oracle一样,SQL Server在非一致性关闭的时候也会进行实例恢复(Instance Recovery),本文根据stack overflow的文章介绍一些SQL Server实例恢复的知识 ...
- C语言 大小写字母转换
//凯鲁嘎吉 - 博客园 http://www.cnblogs.com/kailugaji/ 方法1: #include<stdio.h> #include<stdlib.h> ...
- win21api、win32gui、win32con三个模块操作系统窗口时一些小技巧
下面这段脚本是操作一个浏览器上弹窗,打开文件窗口,由于脚本 执行速度快,当时未添加第2行的延时时,脚本顺利的执行成功,但弹的窗却没有进行操作,建议后续如果脚本执行到打开弹窗时,延时个几秒再去操作所弹窗 ...
- 【C编程基础】make命令和makefile文件
1.关于程序的编译和链接 一般来说,无论是C.C++首先要把源文件编译成中间目标文件即 Object File(windows为.obj文件,unix为.o文件),这个动作叫做编译(compile). ...
- ORC Creation Best Practices
Short Description: ORC Creation Best Practices with examples and references. Article Synopsis. ORC i ...
- 设计模式のObserver Pattern(观察者模式)----行为模式
一.问题产生背景 又被称为订阅发布模式. 最初流传最广的一个面试题:有一只猫咪,猫咪叫了一声,老鼠跑了,老人惊醒了,男主人骂,小偷吓得不敢动了....这就产生一个问题的模型,当对象间存在一对多关系时, ...
- LDAP概念
1.1.LDAP目录结构 此图为树形目录结构,我将此跳过去了,因为这个是按照“国家这种结构来划分的”.如果你喜欢这样看更好,如下还有一种: 树也可以根据互联网域名组主.这种命名方式正越来越受欢迎, ...