大家来讨论发票打印的解决方案(300分)

L

ljzk

Unregistered / Unconfirmed
GUEST, unregistred user!
打印机:OKI5530SC,有切纸功能。
发票大小:宽216MM,长127MM,连续纸。
开发:WIN98+DELPHI6.0+ACCESS。
本发票为实时收取电费,要求打印好一张发票后,发票能够自动进
到撕纸位置(即发票页缝刚好走到撕纸位置)。撕下本发票后,打印下一张时发票退回
打印位置继续打印。以下依此类推。
本人用quickrep做了一个发票打印报表,自定义纸张为QuickRep1.PAGE.LENGTH=127.MM,
QuickRep1.PAGE.LENGTH=216MM,QuickRep1.PAGE.PAGESIZE=Custom;打印机设置为默认
自定义纸张216*127;问题是第一张发票打印完后,打印头停在最后一行上。我无法控制发票
向前走纸,自然也无法控制打印下一张发票时能退回到打印初始位置。
目前只有打印完一张发票后,用人工卷纸,撕下发票后,再将发票进到打印位置。
请有这方面经验的朋友,指教、交流。最好能贴出解决代码。
 
我也想知道!
不过可不可以先用回车使纸先跑多一点,然后再手工退回,这样可节一次手工!
 
试试看:
procedure Directprint(s:string);
type
PrnBuffRec=record
Bufflength:integer;
Buffer:array[0..255] of char;
end;
var Buff:prnBuffRec;
begin
StrPcopy(Buff.Buffer,s);
Buff.Bufflength:=Strlen(Buff.Buffer);
Escape(Printer.handle,Passthrough,0,@Buff,nil);
end;
DirectPrint(CHR(27)+退纸机器码);
 
使用打印机的自动切纸功能,
你想它出多长,它就出多长,而且下次打印时自动缩进多少。
 
关注!!
 
To:lin008:我们用的是发票,有编号,若是LQ打印机的话会出现切错现象,同时若走纸多了
就浪费发票,还有并没有脱离手工!
 
单张打印何必作那么精确,又不是连继打把纸的自定义高度设大些,这样
打完后不就出来多些了,既然你的打印机有切纸功能,按切纸键不就自动
进到切纸位置了,撕纸后,再按进纸退纸键,就回到了打印初始位置,又
可以打下一张了,我们也作收电费的程序,不过是用VFP,不过道理一样,
是不是?
 
用打印代码呀
在打印机的说明书上有
 
TO: lin008
怎样设置打印机的切纸长度?我试着调了一下,但它总是默认页长为11英吋,
不能控制页长(即切纸长度)。请给个详细的说明或方案。
 
TO::我爱PASCAL
你说的对,可以看作是单张发票来处理。但打印结束后,按切纸键时纸张会默认页长
为11英吋,而非进到你想要的位置(自定义的页长:127MM),不知你是如何设置打印机
或程序中如何设置?
 
自定义的页长:127MM并非切纸时的走纸长度,
我们也是用自定义纸张(在VFP中)长宽为发票长宽,打印机属性也为自定义的发票长宽,
至于走纸长度的长度,我们在程序中倒是无法控制,可能在打印机中调吧,是否非要走到
切纸口处,好像用户也没怎么追求,反正出来一截,手能撕就行。不知能满足你的要求否?
 
关注。。
我也遇到类似的问题,
我在写一个执照打印的程序,需要较对位置的。
QUICKREPORT和PRINT的画布哪个更好?
哪位有经验的高手给点意见。
 
我以前也遇到过这样的问题
没办法
只好人工做了
主要是控制打印机走纸如何做
 
用TPRINTER可能好些吧,没试过
 
关注。
有类似问题,打印机是oki 5530sc,但翻遍了说明书也没有找到切纸
命令码。
 
1 在程序中将quickreport纸张大小设置为实际发票的长度和宽度。
2 打印机的纸张设置中长和宽均设为最大值。非打印区域均为0。
3 连续走纸模式。
试试吧
 
我以前曾做过DOS下FOXPRO的打印程序,无论发票长度如何(只要不小于可打印长度)
是能够用打印机的控制命令( ESC码)实现以上所说的功能。不知在DELPHI中能不能用
TPRINTER对象+ESC码来实现?
请做过的朋友贴个例子(源代码)!谢谢。
 
我就是用quickreport作的发票打印,把打印机设为自动切纸,打印机的切纸长度就是你
定义的打印纸长度,我用的是LQ670。打了一张就出一张,撕了后再打就自动退纸。
 
复制一段代码参考一下"Delphi中票据凭证的精确打印

  一、概述
  在银行,税务,邮政等行业的实际工作中,经常涉及到在印刷好具有固定格式的汇款单,储蓄凭证,税票等单据上的确定位置打印输出相关的信息。在此类需求中,精确地定位单据并打印相关信息,是解决问题]的关键。一般情况下,开发者都是通过在打印机上通过重复的测试来达到实际需求。那么,有没有简单有效而又灵活的方法实现上述功能呢?
  二、基本思路
  分析上述单据的特征,可以发现:此类打印输出的信息一般比较简短,不涉及到文字过长的折行处理,另外,其打印输出的位置相对固定。因此,我们可以通过用尺子以毫米为单位,测量好每个输出信息位置的横向和纵向坐标,作为信息输出的位置。但由于不同打印机在实际输出效果上,总是存在理论和实际位置的偏差,因此,要求程序具有一定的灵活性,供最终用户根据需要,进行必要的位置调整。因此,可设置一打印配置文件,用于存储横坐标和纵坐标的偏移量,用于用户进行位置校正,从而提供了一定的灵活性。
  三、精确打印输出的程序实现
  1. 在Delphi中新建一个名为mprint.pas的单元文件并编写如下程序,单元引用中加入Printers略:
//取得字符的高度
function CharHeight: Word;
var
 Metrics: TTextMetric;
begin
 GetTextMetrics(Printer.Canvas.Handle, Metrics);
 Result := Metrics.tmHeight;
end;

file://取得字符的平均宽度
function AvgCharWidth: Word;
var
 Metrics: TTextMetric;
begin
 GetTextMetrics(Printer.Canvas.Handle, Metrics);
 Result := Metrics.tmAveCharWidth;
end;

file://取得纸张的物理尺寸---单位:点
function GetPhicalPaper: TPoint;
var
 PageSize : TPoint;
begin
 file://PageSize.X;
纸张物理宽度-单位:点
 file://PageSize.Y;
纸张物理高度-单位:点
 Escape(Printer.Handle, GETPHYSPAGESIZE, 0,nil,@PageSize);
 Result := PageSize;
end;

file://2.取得纸张的逻辑宽度--可打印区域
file://取得纸张的逻辑尺寸
function PaperLogicSize: TPoint;
var
 APoint: TPoint;
begin
 APoint.X := Printer.PageWidth;
 APoint.Y := Printer.PageHeight;
 Result := APoint;
end;

file://纸张水平对垂直方向的纵横比例
function HVLogincRatio: Extended;
var
 AP: TPoint;
begin
 Ap := PaperLogicSize;
 Result := Ap.y/Ap.X;
end;

file://取得纸张的横向偏移量-单位:点
function GetOffSetX: Integer;
begin
 Result := GetDeviceCaps(Printer.Handle, PhysicalOffSetX);
end;

file://取得纸张的纵向偏移量-单位:点
function GetOffSetY: Integer;
begin
 Result := GetDeviceCaps(Printer.Handle, PhysicalOffSetY);
end;

file://毫米单位转换为英寸单位
function MmToInch(Length: Extended): Extended;
begin
 Result := Length/25.4;
end;

file://英寸单位转换为毫米单位
function InchToMm(Length: Extended): Extended;
begin
 Result := Length*25.4;
end;

file://取得水平方向每英寸打印机的点数
function HPointsPerInch: Integer;
begin
 Result := GetDeviceCaps(Printer.Handle, LOGPIXELSX);
end;

file://取得纵向方向每英寸打印机的光栅数
function VPointsPerInch: Integer;
begin
 Result := GetDeviceCaps(Printer.Handle, LOGPIXELSY)
end;

file://横向点单位转换为毫米单位
function XPointToMm(Pos: Integer): Extended;
begin
 Result := Pos*25.4/HPointsPerInch;
end;

file://纵向点单位转换为毫米单位
function YPointToMm(Pos: Integer): Extended;
begin
 Result := Pos*25.4/VPointsPerInch;
end;

file://设置纸张高度-单位:mm
procedure SetPaperHeight(Value:integer);
var
 Device : array[0..255] of char;
 Driver : array[0..255] of char;
 Port : array[0..255] of char;
 hDMode : THandle;
 PDMode : PDEVMODE;
begin
file://自定义纸张最小高度127mm
if Value < 127 then
Value := 127;
 file://自定义纸张最大高度432mm
 if Value > 432 then
Value := 432;
  Printer.PrinterIndex := Printer.PrinterIndex;
  Printer.GetPrinter(Device, Driver, Port, hDMode);
  if hDMode <> 0 then
   begin
    pDMode := GlobalLock(hDMode);
    if pDMode <> nil then
    begin
     pDMode^.dmFields := pDMode^.dmFields or DM_PAPERSIZE or
               DM_PAPERLENGTH;
     pDMode^.dmPaperSize := DMPAPER_USER;
     pDMode^.dmPaperLength := Value * 10;
     pDMode^.dmFields := pDMode^.dmFields or DMBIN_MANUAL;
     pDMode^.dmDefaultSource := DMBIN_MANUAL;
     GlobalUnlock(hDMode);
    end;
   end;
   Printer.PrinterIndex := Printer.PrinterIndex;
end;

file://设置纸张宽度:单位--mm
Procedure SetPaperWidth(Value:integer);
var
 Device : array[0..255] of char;
 Driver : array[0..255] of char;
 Port : array[0..255] of char;
 hDMode : THandle;
 PDMode : PDEVMODE;
begin
file://自定义纸张最小宽度76mm
if Value < 76 then
Value := 76;
 file://自定义纸张最大宽度216mm
 if Value > 216 then
Value := 216;
  Printer.PrinterIndex := Printer.PrinterIndex;
  Printer.GetPrinter(Device, Driver, Port, hDMode);
  if hDMode <> 0 then
  begin
   pDMode := GlobalLock(hDMode);
   if pDMode <> nil then
   begin
    pDMode^.dmFields := pDMode^.dmFields or DM_PAPERSIZE or
              DM_PAPERWIDTH;
    pDMode^.dmPaperSize := DMPAPER_USER;
    file://将毫米单位转换为0.1mm单位
    pDMode^.dmPaperWidth := Value * 10;
    pDMode^.dmFields := pDMode^.dmFields or DMBIN_MANUAL;
    pDMode^.dmDefaultSource := DMBIN_MANUAL;
    GlobalUnlock(hDMode);
   end;
  end;
  Printer.PrinterIndex := Printer.PrinterIndex;
end;

file://在 (Xmm, Ymm)处按指定配置文件信息和字体输出字符串
procedure PrintText(X, Y: Extended;
Txt: string;
ConfigFileName: string;
FontSize: Integer=12);
var
 OrX, OrY: Extended;
 Px, Py: Integer;
 AP: TPoint;
 Fn: TStrings;
 FileName: string;
 OffSetX, OffSetY: Integer;
begin
file://打开配置文件,读出横向和纵向偏移量
try
 Fn := TStringList.Create;
 FileName := ExtractFilePath(Application.ExeName) + ConfigFileName;
 if FileExists(FileName) then
 begin
  Fn.LoadFromFile(FileName);
  file://横向偏移量
  OffSetX := StrToInt(Fn.Values['X']);
  file://纵向偏移量
  OffSetY := StrToInt(Fn.Values['Y']);
 end
else
begin
 file://如果没有配置文件,则生成
 Fn.Values['X'] := '0';
 Fn.Values['Y'] := '0';
 Fn.SaveToFile(FileName);
end;
finally
 Fn.Free;
end;
X := X + OffSetX;
Y := Y + OffSetY;
Px := Round(Round(X * HPointsPerInch * 10000/25.4) / 10000);
Py := Round(Round(Y * VPointsPerInch * 10000/25.4) / 10000);
Py := Py - GetOffSetY;
file://因为是绝对坐标, 因此, 不用换算成相对于Y轴坐标
Px := Px + 2 * AvgCharWidth;
Printer.Canvas.Font.Name := '宋体';
Printer.Canvas.Font.Size := FontSize;
file://Printer.Canvas.Font.Color := clGreen;
Printer.Canvas.TextOut(Px, Py, Txt);
end;

  2. 使用举例
  在主窗体中加入对mprint单元的引用,在一命令钮的OnClick事件中书写如下代码(用于在邮政汇款单上的相应方框内打印邮政编码843300):
Printer.begin
Doc;
PrintText(16, 14, '8', 'config.txt');
PrintText(26, 14, '4', 'config.txt');
PrintText(36, 14, '3', 'config.txt');
PrintText(46, 14, '3', 'config.txt');
PrintText(56, 14, '0', 'config.txt');
PrintText(66, 14, '0', 'config.txt');
Printer.EndDoc;

  观察结果,用尺子测量偏移量,在config.txt文件中修改X,Y的值即可。
  其它,设置打印机和纸张类型从略。
  四、结束语
  笔者通过该方法,实现了邮政汇款单,储蓄凭证,客户信封等单据的精确打印,取得了较为满意的效果。该程序在Windows98,Delphi5下调试通过。

"

 

Similar threads

D
回复
0
查看
2K
DelphiTeacher的专栏
D
D
回复
0
查看
2K
DelphiTeacher的专栏
D
D
回复
0
查看
1K
DelphiTeacher的专栏
D
D
回复
0
查看
1K
DelphiTeacher的专栏
D
顶部