程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> Delphi >> Delphi+Word解決方案參考

Delphi+Word解決方案參考

編輯:Delphi
這是我做項目過程中自己做的幾個函數,見到大家都在問Word的問題。現在拿出來和大家共享。(希望有朋友可以進一步添加新的功能,或者做成包或者lib等,更方便大家使用。我自己是沒有時間啦,呵呵)
  
  使用前,先根據需要建立一個空的WORD文件作為模板,在模板文件中設置好各種格式和文本。另外,其中的PrnWordTable的參數是TDBGridEh類型的控件,取自Ehlib2.6
  
  其中用到的shFileCopy函數(用於復制文件)和guiInfo函數(用於顯示消息框)也是自己編寫的,代碼也附後。
  
   
  
  示范代碼如下:
  
   
  
  代碼完成的功能:
  
  1.         替換打印模板中的“#TITLE#”文本為“示范代碼1”
  
  2.         並且將DBGridEh1控件當前顯示的內容插入到文檔的末尾
  
  3.         在文檔末尾插入一個空行
  
  4.         在文檔末尾插入新的一行文本
  
  5.         將文檔中的空行去掉
  
   
  
    if PrnWordBegin('C:打印模板.DOC','C:目標文件1.DOC') then
  
    begin
  
      PrnWordReplace('#TITLE#','示范代碼1');
  
      PrnWordTable(DBGridEh1);
  
      PrnWordInsert('');
  
      PrnWordInsert('這是新的一行文本');
  
      PrnWordReplace('^p^p','^p',true);
  
      PrnWordSave;
  
    end;
  
   
  
  源代碼如下:
  
   
  
  //Word打印(聲明部分)
  
   
  
      wDoc,wApp:Variant;
  
      function PrnWordBegin(tempDoc,docName:String):boolean;
  
      function PrnWordReplace(docText,newText:String;bSimpleReplace:boolean=false):boolean;
  
      function PrnWordInsert(lineText:String;bNewLine:boolean=true):boolean;overload;
  
      function PrnWordInsert(var imgInsert:TImage;sBookMark:String=''):boolean;overload;
  
      function PrnWordInsert(var chartInsert:TChart;sBookMark:String=''):boolean;overload;
  
      function PrnWordTable(var dbG:TDBGridEh;sBookMark:String=''):boolean;
  
      procedure PrnWordSave;
  
      procedure PrnWordEnd;
  
   
  
  //Word打印(實現部分)
  
   
  
  {
  
  功能:基於模板文件tempDoc新建目標文件docName並打開文件
  
  }
  
  function PrnWordBegin(tempDoc,docName:String):boolean;
  
  begin
  
    result:=false;
  
    //復制模版
  
    if tempDoc<>'' then
  
      if not shFileCopy(tempDoc,docName) then exit;
  
    //連接Word
  
    try
  
      wApp:=CreateOleObject('Word.Application');
  
    except
  
      guiInfo('請先安裝 Microsoft Word 。');
  
      exit;
  
    end;
  
    try
  
      //打開
  
      if tempDoc='' then
  
      begin
  
        //創建新文檔
  
        wDoc:=wApp.document.Add;
  
        wDoc.SaveAs(docName);
  
      end else begin
  
        //打開模版
  
        wDoc:=wApp.document..Open(docName);
  
      end;
  
    except
  
      guiInfo('打開模版失敗,請檢查模版是否正確。');
  
      wApp.Quit;
  
      exit;
  
    end;
  
    wApp.Visible:=true;
  
    result:=true;
  
  end;
  
   
  
  {
  
  功能:使用newText替換docText內容
  
  bSimpleReplace:true時僅做簡單的替換,false時對新文本進行換行處理
  
  }
  
  function PrnWordReplace(docText,newText:String;bSimpleReplace:boolean=false):boolean;
  
  var i:Integer;
  
  begin
  
    if bSimpleReplace then
  
    begin
  
      //簡單處理,直接執行替換操作
  
    try
  
      wApp.Selection.Find.ClearFormatting;
  
      wApp.Selection.Find.Replacement.ClearFormatting;
  
      wApp.Selection.Find.Text := docText;
  
      wApp.Selection.Find.Replacement.Text :=newText;
  
      wApp.Selection.Find.Forward := True;
  
      wApp.Selection.Find.Wrap := wdFindContinue;
  
      wApp.Selection.Find.Format := False;
  
      wApp.Selection.Find.MatchCase := False;
  
      wApp.Selection.Find.MatchWholeWord := true;
  
      wApp.Selection.Find.MatchByte := True;
  
      wApp.Selection.Find.MatchWildcards := False;
  
      wApp.Selection.Find.MatchSoundsLike := False;
  
      wApp.Selection.Find.MatchAllWordForms := False;
  
      wApp.Selection.Find.Execute(Replace:=wdReplaceAll);
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
      exit;
  
    end;
  
    //自動分行
  
    reWord.Lines.Clear;
  
    reWord.Lines.Add(newText);
  
    try
  
      //定位到要替換的位置的後面
  
      wApp.Selection.Find.ClearFormatting;
  
      wApp.Selection.Find.Text := docText;
  
      wApp.Selection.Find.Replacement.Text := '';
  
      wApp.Selection.Find.Forward := True;
  
      wApp.Selection.Find.Wrap := wdFindContinue;
  
      wApp.Selection.Find.Format := False;
  
      wApp.Selection.Find.MatchCase := False;
  
      wApp.Selection.Find.MatchWholeWord := False;
  
      wApp.Selection.Find.MatchByte := True;
  
      wApp.Selection.Find.MatchWildcards := False;
  
      wApp.Selection.Find.MatchSoundsLike := False;
  
      wApp.Selection.Find.MatchAllWordForms := False;
  
      wApp.Selection.Find.Execute;
  
      wApp.Selection.MoveRight(wdCharacter,1);
  
      //開始逐行插入
  
      for i:=0 to reWord.Lines.Count-1 Do
  
      begin
  
        //插入當前行
  
        wApp.Selection.InsertAfter(reWord.Lines[i]);
  
        //除最後一行外,自動加入新行
  
        if i<reWord.Lines.Count-1 then
  
          wApp.Selection.InsertAfter(#13);
  
      end;
  
      //刪除替換位標
  
      wApp.Selection.Find.ClearFormatting;
  
      wApp.Selection.Find.Replacement.ClearFormatting;
  
      wApp.Selection.Find.Text := docText;
  
      wApp.Selection.Find.Replacement.Text := '';
  
      wApp.Selection.Find.Forward := True;
  
      wApp.Selection.Find.Wrap := wdFindContinue;
  
      wApp.Selection.Find.Format := False;
  
      wApp.Selection.Find.MatchCase := False;
  
      wApp.Selection.Find.MatchWholeWord := true;
  
      wApp.Selection.Find.MatchByte := True;
  
      wApp.Selection.Find.MatchWildcards := False;
  
      wApp.Selection.Find.MatchSoundsLike := False;
  
      wApp.Selection.Find.MatchAllWordForms := False;
  
      wApp.Selection.Find.Execute(Replace:=wdReplaceAll);
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
  end;
  
   
  
  {
  
  功能:打印TDBGridEh當前顯示的內容
  
  基於TDBGridEh控件的格式和內容,自動在文檔中的sBookMark書簽處生成Word表格
  
  目前能夠支持單元格對齊、多行標題(兩行)、底部合計等特性
  
  sBookMark:Word中要插入表格的書簽名稱
  
  }
  
  function PrnWordTable(var dbG:TDBGridEh;sBookMark:String=''):boolean;
  
  var iCol,iLine,i,j,k:Integer;
  
      wTable,wRange:Variant;
  
      iRangeEnd:longint;
  
      iGridLine,iTitleLine:Integer;
  
      getTextText:String;getTextDisplay:boolean;
  
      titleList:TStringList;titleSplit,titleCol:Integer;lastTitleSplit,SubTitle:Integer;lastTitle:String;
  
  begin
  
    result:=false;
  
    try
  
      //計算表格的列數(不包括隱藏的列)
  
      iTitleLine:=1;  //始終默認為1
  
      iCol:=0;
  
      for i:=0 to dbG.Columns.Count-1 Do
  
      begin
  
        if dbG.Columns[i].Visible then
  
        begin
  
          iCol:=iCol+1;
  
        end;
  
      end;
  
      //計算表格的行數(不包括隱藏的列)
  
      if dbG.DataSource.DataSet.Active then
  
        iLine:=dbG.DataSource.DataSet.RecordCount
  
      else
  
        iLine:=0;
  
      iGridLine:=iLine+iTitleLine+dbG.FooterRowCount;
  
      //定位插入點
  
      if sBookMark='' then
  
      begin
  
        //在文檔末尾
  
        iRangeEnd:=wDoc.Range.End-1;
  
        if iRangeEnd<0 then iRangeEnd:=0;
  
        wRange:=wDoc.Range(iRangeEnd,iRangeEnd);
  
      end else begin
  
        //在書簽處
  
        wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);
  
      end;
  
      wTable:=wDoc.Tables.Add(wRange,iGridLine,iCol);
  
      wTable.Columns.AutoFit;
  
      //標題行
  
      k:=1;
  
      for j:=1 to dbG.Columns.Count Do
  
      begin
  
        if dbG.Columns[j-1].Visible then
  
        begin
  
          if dbG.UseMultiTitle then
  
          begin
  
            titleList:=strSplit(dbG.Columns[j-1].Title.Caption,'|');
  
            wTable.Cell(1,k).Range.InsertAfter(titleList.Strings[0]);
  
          end else
  
            wTable.Cell(1,k).Range.InsertAfter(dbG.Columns[j-1].Title.Caption);
  
          //設置單元格對齊方式
  
          if dbG.Columns[j-1].Title.Alignment=taCenter then
  
            wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter
  
          else if dbG.Columns[j-1].Title.Alignment=taRightJustify then
  
            wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight
  
          else if dbG.Columns[j-1].Title.Alignment=taLeftJustify then
  
            wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;
  
          k:=k+1;
  
        end;
  
      end;
  
      //填寫每一行
  
      if iLine>0 then
  
      begin
  
        dbG.DataSource.dataset.DisableControls;
  
        dbG.DataSource.DataSet.First;
  
        for i:=1 to iLine Do
  
        begin
  
          k:=1;
  
          for j:=1 to dbG.Columns.Count Do
  
          begin
  
            if dbG.Columns[j-1].Visible then
  
            begin
  
              if dbG.Columns[j-1].FIEldName<>'' then //避免由於空列而出錯
  
              begin
  
                //如果該列有自己的格式化顯示函數,則調用顯示函數獲取顯示串
  
                getTextText:='';
  
                if Assigned(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FIEldName).OnGetText) then
  
                begin
  
                  dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FieldName).OnGetText(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FIEldName),getTextText,getTextDisplay);
  
                  wTable.Cell(i+iTitleLine,k).Range.InsertAfter(getTextText);
  
                end else begin
  
                  //使用數據庫內容顯示
  
                  wTable.Cell(i+iTitleLine,k).Range.InsertAfter(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FIEldName).AsString);
  
                end;
  
              end;
  
              //設置單元格對齊方式
  
              if dbG.Columns[j-1].Alignment=taCenter then
  
                wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter
  
              else if dbG.Columns[j-1].Alignment=taRightJustify then
  
                wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight
  
              else if dbG.Columns[j-1].Alignment=taLeftJustify then
  
                wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;
  
              k:=k+1;
  
            end;
  
          end;
  
          dbG.DataSource.DataSet.Next;
  
        end;
  
      end;
  
      //結尾行
  
      for i:=1 to dbG.FooterRowCount Do
  
      begin
  
        k:=1;
  
        for j:=1 to dbG.Columns.Count Do
  
        begin
  
          if dbG.Columns[j-1].Visible then
  
          begin
  
            wTable.Cell(iLine+1+i,k).Range.InsertAfter(dbG.GetFootervalue(i-1,dbG.Columns[j-1]));
  
            //設置單元格對齊方式
  
            if dbG.Columns[j-1].Footer.Alignment=taCenter then
  
              wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter
  
            else if dbG.Columns[j-1].Footer.Alignment=taRightJustify then
  
              wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight
  
            else if dbG.Columns[j-1].Footer.Alignment=taLeftJustify then
  
              wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;
  
            k:=k+1;
  
          end;
  
        end;
  
      end;
  
      //處理多行標題
  
      if dbG.UseMultiTitle then
  
      begin
  
        //先分割單元格,再逐個填入第二行
  
        k:=1;
  
        titleCol:=1;
  
        lastTitleSplit:=1;
  
        SubTitle:=0;
  
        lastTitle:='';
  
        for j:=1 to dbG.Columns.Count Do
  
        begin
  
          if dbG.Columns[j-1].Visible then
  
          begin
  
            titleList:=strSplit(dbG.Columns[j-1].Title.Caption,'|');
  
            if titleList.Count>1 then
  
            begin
  
              //處理第二行以上的內容
  
              wTable.Cell(1,k-SubTitle).Range.Cells.Split(titleList.Count,1,false);
  
              for titleSplit:=1 to titleList.Count-1 Do
  
              begin
  
                wTable.Cell(titleSplit+1,titleCol).Range.InsertAfter(titleList.Strings[titleSplit]);
  
              end;
  
              titleCol:=titleCol+1;
  
              //處理第一行合並
  
              if (lastTitleSplit=titleList.Count) and (lastTitle=titleList.Strings[0]) then
  
              begin
  
                //內容相同時,合並單元格
  
                wTable.Cell(1,k-SubTitle).Range.Copy;
  
                wRange:=wDoc.Range(wTable.Cell(1,k-SubTitle-1).Range.Start,wTable.Cell(1,k-SubTitle).Range.End);
  
                wRange.Cells.Merge;
  
                wRange.Paste;
  
                SubTitle:=SubTitle+1;
  
              end;
  
            end;
  
            lastTitle:=titleList.Strings[0];
  
            lastTitleSplit:=titleList.Count;
  
            titleList.Clear;titleList.Free;
  
            k:=k+1;
  
          end;
  
        end;
  
      end;
  
      //自動調整表格
  
      wTable.AutoFitBehavior(1);//根據內容自動調整表格wdAutoFitContent
  
      wTable.AutoFitBehavior(2);//根據窗口自動調整表格wdAutoFitWindow
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
    try
  
      dbG.DataSource.dataset.EnableControls;
  
    except
  
    end;
  
  end;
  
   
  
  {
  
  功能:在Word文件中插入文本(能夠自動進行換行處理)
  
  lineText:要插入的文本
  
  bNewLine:true時新起一行,false時在當前行插入
  
  }
  
  function PrnWordInsert(lineText:String;bNewLine:boolean=true):boolean;
  
  var i:Integer;
  
  begin
  
    try
  
      if bNewLine then
  
        wDoc.Range.InsertAfter(#13);
  
      //自動分行
  
      reWord.Lines.Clear;
  
      reWord.Lines.Add(lineText);
  
      //開始逐行插入
  
      for i:=0 to reWord.Lines.Count-1 Do
  
      begin
  
        //插入當前行
  
        wDoc.Range.InsertAfter(reWord.Lines[i]);
  
        //除最後一行外,自動加入新行
  
        if i<reWord.Lines.Count-1 then
  
          wDoc.Range.InsertAfter(#13);
  
      end;
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
  end;
  
   
  
  {
  
  功能:在Word文件的sBookMark書簽處插入TImage控件包含的圖片
  
  }
  
  function PrnWordInsert(var imgInsert:TImage;sBookMark:String=''):boolean;
  
  var wRange:Variant;iRangeEnd:Integer;
  
  begin
  
    try
  
      if sBookMark='' then
  
      begin
  
        //在文檔末尾
  
        iRangeEnd:=wDoc.Range.End-1;
  
        if iRangeEnd<0 then iRangeEnd:=0;
  
        wRange:=wDoc.Range(iRangeEnd,iRangeEnd);
  
      end else begin
  
        //在書簽處
  
        wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);
  
      end;
  
      if imgInsert.Picture.Graphic<>nil then
  
      begin
  
        Clipboard.Assign(imgInsert.Picture);
  
        wRange.Paste;
  
      end else begin
  
        wRange.InsertAfter('照片');
  
      end;
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
  end;
  
   
  
  {
  
  功能:在書簽sBookMark處插入TChart控件包含的圖表
  
  }
  
  function PrnWordInsert(var chartInsert:TChart;sBookMark:String=''):boolean;
  
  var wRange:Variant;iRangeEnd:Integer;
  
  begin
  
    try
  
      if sBookMark='' then
  
      begin
  
        //在文檔末尾
  
        iRangeEnd:=wDoc.Range.End-1;
  
        if iRangeEnd<0 then iRangeEnd:=0;
  
        wRange:=wDoc.Range(iRangeEnd,iRangeEnd);
  
      end else begin
  
        //在書簽處
  
        wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);
  
      end;
  
      chartInsert.CopyToClipboardBitmap;
  
      wRange.Paste;
  
      result:=true;
  
    except
  
      result:=false;
  
    end;
  
  end;
  
   
  
  {
  
  功能:保存Word文件
  
  }
  
  procedure PrnWordSave;
  
  begin
  
    try
  
      wDoc.Save;
  
    except
  
    end;
  
  end;
  
   
  
  {
  
  功能:關閉Word文件
  
  }
  
  procedure PrnWordEnd;
  
  begin
  
    try
  
      wDoc.Save;
  
      wDoc.Close;
  
      wApp.Quit;
  
    except
  
    end;
  
  end;
  
   
  
  附:shFileCopy源代碼
  
   
  
  {
  
  功能:安全的復制文件
  
  srcFile,destFile:源文件和目標文件
  
  bDelDest:如果目標文件已經存在,是否覆蓋
  
  返回值:true成功,false失敗
  
  }
  
  function shFileCopy(srcFile,destfile&:String;bDelDest:boolean=true):boolean;
  
  begin
  
    result:=false;
  
    if not FileExists(srcFile) then
  
    begin
  
      guiInfo ('源文件不存在,不能復制。'+#10#13+srcFile);
  
      exit;
  
    end;
  
    if srcFile=destFile then
  
    begin
  
      guiInfo ('源文件和目標文件相同,不能復制。');
  
      exit;
  
    end;
  
    if FileExists(destFile) then
  
    begin
  
      if not bDelDest then
  
      begin
  
        guiInfo ('目標文件已經存在,不能復制。'+#10#13+destFile);
  
        exit;
  
      end;
  
      FileSetAttr(destFile,FileGetAttr(destFile) and not $00000001);
  
      if not DeleteFile(PChar(destFile)) then
  
      begin
  
        guiInfo ('目標文件已經存在,並且不能被刪除,復制失敗。'+#10#13+destFile);
  
        exit;
  
      end;
  
    end;
  
    if not CopyFileTo(srcFile,destFile) then
  
    begin
  
      guiInfo ('發生未知的錯誤,復制文件失敗。');
  
      exit;
  
    end;
  
    //目標文件去掉只讀屬性
  
    FileSetAttr(destFile,FileGetAttr(destFile) and not $00000001);
  
    result:=true;
  
  end;
  
   
  
  附:guiInfo源代碼
  
   
  
  {
  
  功能:封裝了各種性質的提示框
  
  sMsg:要提示的消息
  
  }
  
  procedure guiInfo(sMsg:String);
  
  begin
  
    MessageDlg(sMsg,mtInformation,[mbOK],0);
  
  end;
  
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved