1. 引言

        几年前,笔者在做某项目时,需要根据开票方的实际信息(含企业名称,社会信用代码,印章编号)绘制某椭圆形的专用章,加盖到PDF版式文件上,并使用开票方的证书信息进行签名,以防范版式文件伪造、抵赖、冒充和篡改等风险。

        至今,依然清晰地记得,对于专用章图片的生成,笔者当时曾经翻阅很多的相关资料,参考过诸多的代码,但所绘制的图片都不太理想。最后,笔者重温解析几何学的知识,经过数日披星戴月,坚持不懈地努力,终于可以绘制出理想的专用章高清PNG图片。 

        今日,笔者稍得闲暇,愿将笔者摸索的专用章绘制相关的核心算法(该算法使用其它圆形或椭圆形印章),核心的代码进行分享。希望,对遇到类似问题的朋友起到抛砖引玉的作用,同时,由于笔者认知和技术能力有限,难免会有不当或错误之处,欢迎各位朋友不吝赐教。

2. 专用章规范

        专用章的规范很简单,笔者摘录如下:

        一、形状为椭圆形,尺寸为40×30(mm);

        二、边宽1mm;

        三、中间为信用代码,18位阿拉伯数字字高3.7mm,字宽1.3mm,18位阿拉伯

数字总宽度26mm(字体为Arial);

        四、上方环排中文文字高为4.2mm,环排角度(夹角)210-260度,字与

边线内侧的距离0.5mm(字体为仿宋体);

        五、下横排“**专用章”文字字高4.6mm,字宽3mm,延章中心线到下

横排字顶端距离4.2mm(字体为仿宋体);

        六、专用章下横排号码字高2.2mm,字宽1.7mm,延章中心线到下横排号

码顶端距离10mm(字体为Arial),不需编号时可省去此横排号码。

        在常见的公章中以圆形公章和椭圆形公章最为常见。从几何学的角度来说圆是椭圆的一种特殊形式。无论是圆形印章还是椭圆形印章的绘制算法,有两个要点需要掌握。第一个是环绕印章边缘文字的均匀分布的问题,另外一个是环绕印章边缘文字的旋转问题。如果,这两个问题能得心应手,恰到好处的处理,则可以绘制精美的印章。

        然而,通过对专用章规范的解读会发现,在规范中对文字的大小的规定,似乎让人有点为难,因为字宽和字高的比例和字体比例不一致,为此必须做特殊的处理才能满足规范。因此专用章除要关注以上两点之外,还要关注对于字体的缩放操作。

        下面,笔者对以上三点所涉及的核心算法做简要分析。

3. 专用章绘制核心算法分析

3.1 环绕印章边缘文字的均匀分布

        在圆形印章和椭圆形印章中,环绕印章边缘的文字都是以圆心(对于椭圆来说是中心)为中心,环绕印章边缘均与分布,这样从视觉上来看才是舒服和完美的。所谓的均匀分布,是指相邻的文字之间的弧长相等。对于圆形印章来说做到这一点相对来说比较容易,因为只要控制相邻的文字之间的圆心角相等即可。但对于椭圆形印章来说,圆心角相等则对应的弧长未必相等,所以椭圆形印章环绕印章边缘文字的均与分布,相对来说计算比较复杂。

        熟悉椭圆性质的朋友都知道,椭圆的周长和弧长没有精确的计算公式,只能借鉴积分进行近似的计算。如图-1所示,使用平面解析几何学知识,可以根据椭圆的圆心角θ,计算出对应的舷长ℓ。如果圆心角θ足够小,我们则可以认为圆心角θ对应的弧长等于弦长ℓ。关于椭圆的弦长计算公式的推导,笔者不再展开赘述,有兴趣的朋友可以查阅相关资料,或者和笔者做进一步的沟通交流。使用上述理论,我可以将椭圆的弧分割成若干段,然后对求出每一段弧所对应的弦长,延后求和,便可近似的得到椭圆的弧长。我们分割弧度随对应的圆心角θ越小,则计算结果越精确。

java画椭圆印章算法 椭圆印章制作_几何学

图-1 椭圆弧长的计算

        环绕椭圆印章边缘均匀分布文字,首先需要计算出起始弦和结束弦之间的弧长,然后进行等分。行之有效的方法是,使θ足够小,分割弧,计算出对应的弦长,在内存中建立θ和弦长对应的二维表,然后再计算出弧上的等分点即可。如图-2所示,笔者使用上述理论,将X轴上方的椭圆弧,进行十等分的效果,左边使用θ=1度进行计算,右边使用θ=0.0025度进行计算所的效果。通过对比,可以证明如果θ越小,等分效果越精确。 

java画椭圆印章算法 椭圆印章制作_缩放_02

 图-2θ越小计算椭圆弧长越精确效果对比

3.2 环绕印章边缘文字的旋转

        在计算机中常规的字体打印,字体是垂直于字体基线的。在圆形印章和椭圆形印章中,环绕印章边缘文字的基线,需要平行于文字输出点的切线,这样才显得整洁和美观,如图-3所示。为了满座该原则,在进行环绕印章边缘文字的旋转操作时,首先,需要计算出文字输出点切线的角度。然后,旋转坐标系平行椭圆切线。最后,文字输出点根据坐标系旋转角度进行坐标位置变换,输出文本。

java画椭圆印章算法 椭圆印章制作_栅格_03

 图-3 旋转文字基线平行于输出点切线效果图

3.3 通过仿射变换对字体大小进行缩放

        由于专用章规范中所规定的字高和字宽的比例,和字体本身的比例不一致,如果通过绘制文本(drawstring)的方式来实现,显然是不行的。有经验的朋友会自然而然地会想到,是否可以通过仿射变换来进行处理呢?

        所谓仿射变换就是“线性变换”加“平移”,其满足以下三点,1.变换前是直线,变换后依然是直线;2. 直线比例保持不变;3.变换前是原点,变换后依然是原点。仿射变换只需构造两行三列的仿射变换矩阵即可实现图形的缩放(scale),旋转(rotate),切变(share),平移(translate)。显然,仿射变换是可以达到我们的目标的。有关仿射变换更多的细节,笔者不再展开叙述,有兴趣的朋友可查阅相关的资料,或者和笔者做进一步的沟通交流。

        然而,仿射变换是对图像进行处理的,不可以对文字直接处理。如果要对文字进行处理,需要首先对文字进行栅格化,然后再做仿射变换。

        为了满足专用章对字高和字宽的限定,我们需要对文本进行栅格化处理,再构造构造用于缩放(scale)计算的仿射矩阵,最后对栅格化后的文本进行仿射变换即可。需要特别注意的是,进行仿射变换后,图像(栅格化后的文本)会产生位移,需要应用特殊的处理以解决问题。

        如果对以上三点的核心算法能够掌握,想必绘制符合规范要求的专用章应该不在话下。下面将笔者开发的专用章绘制工具及核心代码分享给各位朋友。

4. 使用GDI+绘制专用章核心代码分享

        下面将设计算法的核心代码进行分享,是该代码绘制的椭圆形印章的效果如下图所示。

java画椭圆印章算法 椭圆印章制作_缩放_04

 图-4 使用GDI+绘制的椭圆形印章效果图

{************************************************************************
函数:drawStamp
功能:绘制专用章
参数:graph           输入参数,GDI+ Graphics对象
      corpName        输入参数,公司名称
	  nsrsbh          输入参数,信用代码
	  stampNo         输入参数,印章编号
	  scale           输入参数,缩放比例
	  isSave          输入参数,是否保存图片文件
作者:海之边  QQ-3094353627
 ************************************************************************}
procedure drawStamp(graph: TGPGraphics; corpName: WideString;
  nsrsbh: WideString; stampNo: WideString; scale: Single; isSave: Boolean);
var iPntCnt, iWordCnt, idx: Integer;
    fOffsetX, fOffSetY, fWid, fHigh, fAInter, fBInter, fStartAngle, fEndAngle: Single;
    fTagent: Single;
    cBackColor: Cardinal;
    wsCur: WideString;
    sTag: string;
    pPnts, pEqualPnts, pPntCur: PGPPointF;
    pPntTypes: PByte;
    rPntStart, rPntCenter, rPntOrigin, rCurPnt: TGPPointF;
    oPen: TGPPen;
    oBrush: TGPSolidBrush;
    oPath, oPathNew: TGPGraphicsPath;
    oFontFamilyArial, oFontFamilySong: TGPFontFamily;
    oTxtFmt: TGPStringFormat;
    oMatrix: TGPMatrix;
    s: TStatus;
begin
  oPen := nil;
  oBrush := nil;
  oPath := nil;
  oPathNew := nil;
  oFontFamilyArial := nil;
  oTxtFmt := nil;
  pPnts := nil;
  pPntTypes := nil;
  oMatrix := nil;
  pEqualPnts := nil;
  oFontFamilySong := nil;

  try
    graph.SetSmoothingMode(SmoothingModeAntiAlias);
    graph.SetTextRenderingHint(TextRenderingHintAntiAlias);
    graph.SetInterpolationMode(InterpolationModeHighQualityBicubic);
    graph.SetPageUnit(UnitMillimeter);
    graph.SetPageScale(scale);

    //擦除背景
    if not isSave then
      cBackColor := MakeColor(255, 255, 255)
    else
      cBackColor := MakeColor(0, 255, 255, 255);
    graph.Clear(cBackColor);

    //2. 绘制边框
    if isSave then
    begin
      fOffsetX := 0.0;
      fOffsetX := 0.0;
    end
    else
    begin
      fOffsetX := 5.0;
      fOffSetY := 5.0;
    end;
    fWid := 40.0;
    fHigh := 30.0;

    oPen := TGPPen.Create(MakeColor(255, 0, 0), 1);
    rPntStart.X := fOffsetX + 0.5;
    rPntStart.Y := fOffSetY + 0.5;
    graph.DrawEllipse(oPen, rPntStart.X, rPntStart.Y, fWid - 1, fHigh - 1);

    //2. 平移坐标原点到椭圆中心
    rPntCenter.X := fOffsetX + fWid / 2;
    rPntCenter.Y := fOffSetY + fHigh / 2;
    graph.TranslateTransform(rPntCenter.X, rPntCenter.Y);

    //3. 绘制税号(社会信用代码)
    //3.1. 初始化
    oBrush := TGPSolidBrush.Create(MakeColor(255, 0, 0));
    rPntOrigin.X := 0;
    rPntOrigin.Y := 0;
    oFontFamilyArial := TGPFontFamily.Create('Arial');

    oTxtFmt := TGPStringFormat.Create;
    oTxtFmt.SetAlignment(StringAlignmentCenter);
    oTxtFmt.SetLineAlignment(StringAlignmentCenter);

    //3.2 创建文本Path并栅格化
    oPath := TGPGraphicsPath.Create;
    oPath.AddString(nsrsbh, Length(nsrsbh), oFontFamilyArial, FontStyleRegular,
      2.273, rPntOrigin, oTxtFmt);
    iPntCnt := oPath.GetPointCount;
    pPnts := GetMemory(iPntCnt * SizeOf(TGpPointF));
    oPath.GetPathPoints(pPnts, iPntCnt);
    pPntTypes := GetMemory(iPntCnt);
    oPath.GetPathTypes(pPntTypes, iPntCnt);

    //3.3 进行缩放仿射变换
	scalAffine(1, 1.3121, pPnts, iPntCnt);
	oPathNew := TGPGraphicsPath.Create(pPnts, pPntTypes, iPntCnt);

    //3.4 绘制税号
    graph.FillPath(oBrush, oPathNew);

    //4. 绘制公司名称
    //4.1 设置公司名称外对齐轮廓椭圆的长轴和短轴
    fAInter := 16.4;
    fBInter := 11.4;

    //4.2 设置公司名称在椭圆上的绘制范围
    iWordCnt := Length(corpName);
    if iWordCnt >= 23 then
    begin
      fStartAngle := 140.0;
      fEndAngle := 400.0;
    end
    else if iWordCnt > 20 then
    begin
      fStartAngle := 145.0;
      fEndAngle := 390.0;
    end
    else
    begin
      fStartAngle := 165.0;
      fEndAngle := 375.0;
    end;
    oFontFamilySong := TGPFontFamily.Create('仿宋');

    //4.3 平分弧
    pEqualPnts := self.splitEllipseArc(fAInter, fBInter, fStartAngle,
      fEndAngle, iWordCnt);

    //4.5 逐字绘制
    pPntCur := pEqualPnts;
    for idx := 1 to iWordCnt do
    begin
      wsCur := corpName[idx];
      //4.5.1 计算切线角度
      fTagent := calcEllipseTangentLineDegree(fAInter, fBInter, pPntCur^.X,
        pPntCur^.Y);

      //4.5.2 平移坐标系原点到等点
      graph.TranslateTransform(pPntCur^.X, pPntCur^.Y);

      //4.5.3 创建Path
      if Assigned(oPath) then
        FreeAndNil(oPath);
      if Assigned(oPathNew) then
        FreeAndNil(oPathNew);
      if Assigned(oMatrix) then
        FreeAndNil(oMatrix);
      if Assigned(pPnts) then
        FreeMemory(pPnts);
      if Assigned(pPntTypes) then
        FreeMemory(pPntTypes);

      oPath := TGPGraphicsPath.Create;
      rPntOrigin.X := 0.0;
      rPntOrigin.Y := 0.0;
      oPath.AddString(wsCur, Length(wsCur), oFontFamilySong, FontStyleRegular,
        3.3158, rPntOrigin, oTxtFmt);

      iPntCnt := oPath.GetPointCount;
      pPnts := GetMemory(iPntCnt * SizeOf(TGpPointF));
      oPath.GetPathPoints(pPnts, iPntCnt);
      pPntTypes := GetMemory(iPntCnt);
      oPath.GetPathTypes(pPntTypes, iPntCnt);

      //4.5.4 旋转仿射变换
	  rotateAffine(fTagent * 180.0 / Pi, pPnts, iPntCnt);
      oPathNew := TGPGraphicsPath.Create(pPnts, pPntTypes, iPntCnt);

      graph.FillPath(oBrush, oPathNew);

      //4.5.5 恢复坐标系
      graph.TranslateTransform(pPntCur^.X * -1, pPntCur^.Y * -1);

      Inc(pPntCur);
    end;

    //5. 绘制"XX专用章"
    //5.1 初始化
    wsCur := 'XX专用章';
    if Assigned(oPath) then
      FreeAndNil(oPath);
    if Assigned(oPathNew) then
      FreeAndNil(oPathNew);
    if Assigned(oMatrix) then
      FreeAndNil(oMatrix);
    if Assigned(pPnts) then
      FreeMemory(pPnts);
    if Assigned(pPntTypes) then
      FreeMemory(pPntTypes);

    //5.2 平移坐标系原点
    graph.TranslateTransform(0, 7.5);

    //5.3 输出文本并栅格化
    oPath := TGPGraphicsPath.Create;
    rPntOrigin.X := 0;
    rPntOrigin.Y := 0;
    oPath.AddString(wsCur, Length(wsCur), oFontFamilySong, FontStyleRegular,
      2.201, rPntOrigin, oTxtFmt);

    iPntCnt := oPath.GetPointCount;
    pPnts := GetMemory(iPntCnt * SizeOf(TGpPointF));
    oPath.GetPathPoints(pPnts, iPntCnt);
    pPntTypes := GetMemory(iPntCnt);
    oPath.GetPathTypes(pPntTypes, iPntCnt);

    //5.4 仿射变换
	scalAffine(1, 1.6487, pPnts, iPntCnt);
	oPathNew := TGPGraphicsPath.Create(pPnts, pPntTypes, iPntCnt);

    //5.5 输出
    graph.FillPath(oBrush, oPathNew);

    //5.6 还原坐标系原点
    graph.TranslateTransform(0, -7.5);

    //6. 绘制印章编号
    if stampNo = '' then
      Exit;

    //6.1 初始化
    wsCur := Format('(%s)', [stampNo]);
    if Assigned(oPath) then
      FreeAndNil(oPath);
    if Assigned(oPathNew) then
      FreeAndNil(oPathNew);
    if Assigned(oMatrix) then
      FreeAndNil(oMatrix);
    if Assigned(pPnts) then
      FreeMemory(pPnts);
    if Assigned(pPntTypes) then
      FreeMemory(pPntTypes);

    //6.2 平移坐标系原点
    graph.TranslateTransform(0, 11.1);

    //6.3 输出文本并栅格化
    oPath := TGPGraphicsPath.Create;
    rPntOrigin.X := 0;
    rPntOrigin.Y := 0;
    oPath.AddString(wsCur, Length(wsCur), oFontFamilyArial, FontStyleRegular,
      1.8730, rPntOrigin, oTxtFmt);

    iPntCnt := oPath.GetPointCount;
    pPnts := GetMemory(iPntCnt * SizeOf(TGpPointF));
    oPath.GetPathPoints(pPnts, iPntCnt);
    pPntTypes := GetMemory(iPntCnt);
    oPath.GetPathTypes(pPntTypes, iPntCnt);

    //6.4 仿射变换
	scalAffine(1, 1.9442, pPnts, iPntCnt);
	oPathNew := TGPGraphicsPath.Create(pPnts, pPntTypes, iPntCnt);

    //6.5 输出
    graph.FillPath(oBrush, oPathNew);

    //6.6 还原坐标系
    graph.TranslateTransform(0, -11.1);
  finally
    if Assigned(oPen) then
      FreeAndNil(oPen);

    if Assigned(oBrush) then
      FreeAndNil(oBrush);

    if Assigned(oPath) then
      FreeAndNil(oPath);

    if Assigned(oPathNew) then
      FreeAndNil(oPathNew);

    if Assigned(oFontFamilyArial) then
      FreeAndNil(oFontFamilyArial);

    if Assigned(oTxtFmt) then
      FreeAndNil(oTxtFmt);

    if Assigned(pPnts) then
      FreeMemory(pPnts);

    if Assigned(pPntTypes) then
      FreeMemory(pPntTypes);

    if Assigned(oMatrix) then
      FreeAndNil(oMatrix);

    if Assigned(pEqualPnts) then
      FreeMemory(pEqualPnts);

    if Assigned(oFontFamilySong) then
      FreeAndNil(oFontFamilySong);
  end;
end;

5. 后记

        笔者期望能通过这篇文章,对有相同业务背景,或遇到相关技术障碍的朋友,能起到抛砖引玉的作用。由于篇幅有限,诸多的细节不能一一展开叙述,如有需要,笔者也坦诚地期盼能和各位朋友做进一步的沟通交流,相互学习,共同受益。