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

ASP.NET
asp.net GridView控件中模板列CheckBox全选、反选、取消
asp.net GridView 删除时弹出确认对话框(包括内容提示)
asp.net DropDownList 三级联动下拉菜单实现代码
asp DataTable添加列和行的三种方法
Asp.net 页面调用javascript变量的值
asp.net 长文章通过设定的行数分页
asp.net 定时间点执行任务的简易解决办法
asp.net 页面延时五秒,跳转到另外的页面
asp.net 动态输出透明gif图片
asp.net DataList与Repeater用法区别
asp.net Javascript获取CheckBoxList的value
asp.net程序在调式和发布之间图片路径问题的解决方法
asp.net下生成英文字符数字验证码的代码
asp.net 页面版文本框智能提示JSCode (升级版)
ASP.NET URL伪静态重写实现方法
ASP.NET 2.0 中Forms安全认证
asp.net 动态添加多个用户控件
asp.net Repeater显示父子表数据,无闪烁
asp.net 无法获取的内部内容,因为该内容不是文本 的解决方法
asp.net GridView排序简单实现

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


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-09-23   浏览: 95 ::
收藏到网摘: 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  

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