当前位置: 首页 > 图文教程 > 网络编程 > ASP.NET > 不同映射模式下的直线输出的效果问题

ASP.NET
Visual Basic .NET处理Excel表格全接触
ASP.NET缓存方法分析和实践示例
.NET Framework 新功能和增强的功能
ASP.NET MVC :MVC页面验证与授权
URL的传递字符串问题小研究
.NET程序员新方向 Ruby核心语法入门
Asp.net教程:设计IP地址屏蔽功能
asp.net中的观察者模式
ASP.NET网站程序防SQL注入式攻击方法
ASp.Net自定义验证码控件
ASP.Net开发新手常见问题备忘录
.net解决数据导出excel时的格式问题
让.Net 应用程序突破2G的内存访问限制
asp.net 获取客户端IP与mac
用ASP.NET创建网络相册 保存美好时光
创建基于ASP.NET的SMTP邮件系统
在ASP.Net 中实现将图片存入数据库
经典ASP.NET打印技术
详细解读.NET中的代码动态编译
如何在VC中实现淡入淡出的启动画面

ASP.NET 中的 不同映射模式下的直线输出的效果问题


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-09-23   浏览: 69 ::
收藏到网摘: n/a

在做打印预览时:发现有趣的现象。

1:打印预览时宽度设置为0.1mm的线与设置为0.2mm的线显示效果一样,无区别。

打印机输出正常。

2:打印预览时:相同宽度的线有些显示有些不显示。

打印机输出正常。

仔细查看了一下,实际上是映射模式下的显示问题,鉴于映射模式总是让人焦头

烂额,值得总结一下:

开始分析:

第一步;写一个MM_TEXT模式的例子。打印6条竖线,宽度由1递增。

以下为引用的内容:
  dc.SetMapMode(MM_TEXT);
  for(int ii=1;ii<=6;ii++)
  {
     CPen pen;
     pen.CreatePen(PS_SOLID,ii,RGB(255,0,0));
     CPen *pOldPen = dc.SelectObject(&pen);
     dc.MoveTo(xStart+xInterval*ii,0);
     dc.LineTo(xStart+xInterval*ii,xLineLen);
     dc.SelectObject(&pOldPen);
  } 

当然,会输出不同宽度的6条线。

很正常。映射模式是像素为单位。

第二步:模拟一下打印预览下的模式(0.1mm的方式)

以下为引用的内容:

  long lLogInch = 254;
  dc.SetMapMode(MM_ANISOTROPIC);
  dc.SetWindowExt(lLogInch, lLogInch); 
  dc.SetViewportExt(dc.GetDeviceCaps(LOGPIXELSX), dc.GetDeviceCaps(LOGPIXELSY)); 

映射:1个单位是0.1mm。

同样用上面的代码输出。

嘿:1,2,3线一样粗,4,5,6线一样粗。

实际打印机输出,粗细是递增的,非常正常。

Why?

第三步:估计是精度问题吧?

看看新的映射模式下:每根线的实际输出像素。

首先,通过long lX = dc.GetDeviceCaps(LOGPIXELSX);

取得当前显示器一个英寸是等于96个像素。

来,算一下:

以下为引用的内容:

   
  0.1毫米 ——》96/254 = 0.378
  0.2毫米 ——》 0.756
  0.3毫米 ——》 1.134
  0.4毫米 ——》 1.512
  0.5毫米 ——》 1.89
  0.6毫米 ——》 2.268

呵,很好解释了,四舍五入后,1,2,3都是1像素,4,5,6都是2像素。

实际打印机呢? lX = 600.


不用计算了吧,精度高,打印出来当然就OK了。

 

结论1:

由于显示器的像素点精度问题,在使用Pen进行绘制时,会进行四舍五入,

如果当前映射模式不合适,会导致应用端不同的输入在四舍五入后,得到一样的显示。解决办法:如Excel一般,不要提供太小的线宽,比如最小线宽必须是折算后的

一个像素。如:1/0.378 = 0.26 mm .

第四步:以上规则只适用于Pen的Line绘制方法,直线还可以通过FillRect或FillSolidRect方式。并且会出现开头讲的现像2。

来,再试试FillSolidRect会如何:

以下为引用的内容:
  long xStart = 100;
  long xLineLen = 100*10;
  long xInterval = 50;
  for(int ii=0;ii<=5;ii++)
  {
    CRect rc(xStart+xInterval*ii,0,xStart+xInterval*ii+ii+1,xLineLen);
    dc.FillSolidRect(rc,RGB(255,0,0));
  } 

结果:

只显示出后四条线。

为何?为啥不四舍五入了呢?

第五步:矩形可能会是先四舍五入再算宽度的?是吗?做个测试先。

打印时,加多一行测试:

以下为引用的内容:
TRACE(_T("\n%f,%f"),rc.left*0.378,rc.right*0.378);
 
输出如下:

以下为引用的内容:
    37.800000,38.178000
    56.700000,57.456000
    75.600000,76.734000
    94.500000,96.012000
    113.400000,115.290000
    132.300000,134.568000

假设一下,如果先四舍五入,然后再相减,得到宽度是:

0   0  1  1  2   3  

比较一下输出效果,果然如此。