之前给公司做打印都是用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可供浏览器调用的本地打印程序的更多相关文章

  1. 学习笔记:URL Protocol在浏览器中打开本地应用程序

    看到阿里的网站上可以通过点击卖家的旺旺图标从而调用本地的阿里旺旺程序,而且还可以传递当前浏览者需要咨询的商品.这是怎么实现的呢?是通过URLProtocol来完成. 原理还没有太清楚,即在系统里注册一 ...

  2. chrome浏览器插件启动本地应用程序

    chrome浏览器插件启动本地应用程序 2014-04-20 00:04:30|  分类: 浏览器插件|举报|字号 订阅     下载LOFTER我的照片书  |     chrome的插件开发这里就 ...

  3. 一个实现浏览器网页与本地程序之间进行双向调用的轻量级、强兼容、可扩展的插件开发平台—PluginOK中间件

    通过PluginOK中间件插件平台(原名本网通WebRunLocal)可实现在网页中的JavaScript脚本无障碍访问本地电脑的硬件.调用本地系统的API及相关组件,同时可彻底解决ActiveX组件 ...

  4. ubuntu下浏览器调用本地应用程序

    ubunut下浏览器调用本地应用程序需要desktop文件和scheme协议的支持,和windows 的url protocol类似,只是注册协议的方式不一样. 首先是desktop文件,里面需要加入 ...

  5. mac 下基于firebreath 开发多浏览器支持的浏览器插件

    mac 下基于firebreath 开发多浏览器支持的浏览器插件 首先要区分什么是浏览器扩展和浏览器插件;插件可以像本地程序一样做的更多 一. 关于 firebreath http://www.fir ...

  6. 可以供MFC调用的,QT实现的DLL(使用qt-solutions的qtwinmigrate实现)

    MFC和QT的消息循环机制不同,所以,要让QT写的DLL可以供MFC调用,要做一点特殊的处理 #include <qmfcapp.h> #include <qwinwidget.h& ...

  7. 可以供MFC调用的,QT实现的DLL(qtwinmigrate实现)

    MFC和QT的消息循环机制不同,所以,要让QT写的DLL可以供MFC调用,要做一点特殊的处理 #include <qmfcapp.h> #include <qwinwidget.h& ...

  8. asp.net core系列 58 IS4 基于浏览器的JavaScript客户端应用程序

    一. 概述 本篇探讨使用"基于浏览器的JavaScript客户端应用程序".与上篇实现功能一样,只不过这篇使用JavaScript作为客户端程序,而非core mvc的后台代码Ht ...

  9. python selenium中如何测试360等基于chrome内核的浏览器

    转自:https://blog.csdn.net/five3/article/details/50013159 直接上代码,注意是基于chrome内核的浏览器,基于ie的请替换其中的chrome方法为 ...

随机推荐

  1. OAF--基础

    OAF是WEB界面,FORM是由JDK将FORM里面的东西插入到HTML UI里的: OAF由 Oracle Business Components for JAVA(BC4J)框架作为其模型部分,完 ...

  2. sqlserver备份

    /// <summary> /// sqlserver备份 /// </summary> public class SqlserverBack : IBack { privat ...

  3. TCP/IP协议族(笔记)

    1.HTTP HTTPS DCHP ICMP  SMTP IMAP MIME POP PPTP 协议族 tcp/ip 是基于tcp和ip这两个最初的协议之上的不同的通信协议的大集合 TCP/IP不是一 ...

  4. C# -- Lambda 表达式的使用

    C# -- Lambda 表达式的使用 Lambda 表达式是作为对象处理的代码块(表达式或语句块). 它可作为参数传递给方法,也可通过方法调用返回. Lambda 表达式是可以表示为委托的代码,或者 ...

  5. 持续集成-Jenkins常用插件安装

    1. 更新站点修改 由于之前说过,安装Jenkins后首次访问时由于其他原因[具体未知]会产生离线问题.网上找了个遍还是不能解决,所以只能跳过常用插件安装这步.进入Jenkins后再安装这些插件. 在 ...

  6. Unity基础6 Shadow Map 阴影实现

    这篇实现来的有点墨迹,前前后后折腾零碎的时间折腾了半个月才才实现一个基本的shadow map流程,只能说是对原理理解更深刻一些,但离实际应用估计还需要做很多优化.这篇文章大致分析下shadow ma ...

  7. 【Linux基础】VI 编辑器基本使用方法

    vi编辑器是所有Unix及Linux系统下标准的编辑器.对Unix及Linux系统的任何版本,vi编辑器是完全相同的,它是Linux中最基本的文本编辑器. 第一章vi的三种模式 第二章vi文本编辑器 ...

  8. JavaScript的基本包装类型_String类型

    String类型概述 String在底层字符串是以字符数组的形式保存的 var str = "Hello"; // 在底层其实就是['H','e','l','l','o'] 字符串 ...

  9. 第2章 Java并发机制的底层实现原理

    2.2 synchronized的实现原理与应用 当一个线程A执行字节码时遇到monitorenter指令时,会首先检查该指令关联的Object的对象头中的Mark Word状态. 2.2.1 如果是 ...

  10. 爬取伯乐在线文章(二)通过xpath提取源文件中需要的内容

    爬取说明 以单个页面为例,如:http://blog.jobbole.com/110287/ 我们可以提取标题.日期.多少个评论.正文内容等 Xpath介绍 1. xpath简介 (1) xpath使 ...