C# 通过调用Microsoft Office编辑Word文件_c#处理office文件-程序员宅基地

技术标签: C#  c#  

1. 添加引用

在这里插入图片描述
实际程序集看当前电脑安装的软件不一样而有所差异。

2.具体操作

//获取应用路径
string appPath = System.AppDomain.CurrentDomain.SetupInformation.ApplicationBase;
//获取模板路径
string templatePath = appPath + "Template\\Report.docx";

//声明Word工作簿
Word.Application wordApp = new Word.Application();
Word.Document wordDoc =  null;
            
//获取当前客户的项目记录
DataTable dtRecordItem = PGxMySqlDB.GetRecordItemListTable(customer.ID);

//文档操作时是否可见
wordApp.Visible = false;

for (int i = 0; i < dtRecordItem.Rows.Count; i++)
{
    
//获取主项目记录ID
Int32 recordItemID = Convert.ToInt32(dtRecordItem.Rows[i]["ID"],FormatInfo.Number);
string itemSN = dtRecordItem.Rows[i]["ItemSN"].ToString();  //获取项目编号
Item item = PGxMySqlDB.GetItem(itemSN);     //读取项目信息

try
{
    

    #region customer info

    //打开模板文件
    wordDoc = wordApp.Documents.Open(templatePath);
    //获取文件中的第一个表格,表格索引从1开始
    Word.Table baseInfoTable = wordDoc.Tables[1];

    //设置医院名称,将文本域第前四个字符替换掉
    wordDoc.Range(0, 4).Text = customer.Hospital;

    baseInfoTable.Cell(1, 2).Range.Text = customer.Name;                //客户名称
    baseInfoTable.Cell(1, 4).Range.Text = customer.Department;          //科室
    baseInfoTable.Cell(1, 6).Range.Text = customer.BedNumber;           //床号
    baseInfoTable.Cell(1, 8).Range.Text = customer.BarCode;             //条码

    if (customer.Sex == Gender.Female)                                  //性别
    {
    
        baseInfoTable.Cell(2, 2).Range.Text = Strings.Female;
    }
    else
    {
    
        baseInfoTable.Cell(2, 2).Range.Text = Strings.Male;
    }
    baseInfoTable.Cell(2, 4).Range.Text = customer.OutpatientOrInpatientNumber;         //门诊住院号
    baseInfoTable.Cell(2, 6).Range.Text = customer.Doctor;                              //医生
    baseInfoTable.Cell(2, 8).Range.Text = customer.DoctorPhone;                         //医生电话

    baseInfoTable.Cell(3, 2).Range.Text = customer.Age.ToString(FormatInfo.Number);     //年龄
    baseInfoTable.Cell(3, 4).Range.Text = customer.SampleType;                          //样本类型
    baseInfoTable.Cell(3, 6).Range.Text = customer.InspentionDepartment;                //送检单位

    baseInfoTable.Cell(4, 2).Range.Text = customer.SampleCondition;                     //样本情况
    baseInfoTable.Cell(4, 4).Range.Text = customer.Diagnosis;                           //临床诊断

    baseInfoTable.Cell(5, 2).Range.Text = customer.SamplingTime.Value.Date.ToString("dd/MM/yyyy", FormatInfo.DateTime);             //采样日期
    baseInfoTable.Cell(5, 4).Range.Text = customer.ReceiptTime.Value.Date.ToString("dd/MM/yyyy", FormatInfo.DateTime);              //接收日期
    baseInfoTable.Cell(5, 6).Range.Text = customer.Notes;                               //备注

    #endregion customer info <= END

    object replaceAll = Word.WdReplace.wdReplaceAll;   //替换所有文本
    object missing = Type.Missing;  //空操作

    /*----------------更新主项目信息---------------------------*/

    wordApp.Selection.Find.Replacement.ClearFormatting();
    wordApp.Selection.Find.ClearFormatting();
    wordApp.Selection.Find.Text = "{ItemName}";
    wordApp.Selection.Find.Replacement.Text = item.SN + " " + item.ChineseName + "(" + item.EnglishName + ")";
    //执行替换,replace模式需要设置才能正常替换
    wordApp.Selection.Find.Execute(missing, missing, missing, missing,
                                    missing, missing, missing, missing,
                                    missing, missing, replaceAll, missing,
                                    missing, missing, missing);   

    /*----------------更新子项目检测结果---------------------------*/

    string subItemName = "";

    DataTable dtReportData = PGxMySqlDB.GetReportSubItemDataListTable(recordItemID); //获取报告数据

    for (int j = 0; j < dtReportData.Rows.Count; j++)
    {
    
        wordDoc.Tables[2].Rows.Add();
        wordDoc.Tables[2].Cell(wordDoc.Tables[2].Rows.Count, 1).Range.Text = dtReportData.Rows[i]["SubitemName"].ToString();
        wordDoc.Tables[2].Cell(wordDoc.Tables[2].Rows.Count, 2).Range.Text = dtReportData.Rows[i]["MainGenotype"].ToString();
        wordDoc.Tables[2].Cell(wordDoc.Tables[2].Rows.Count, 3).Range.Text = "1A";
        wordDoc.Tables[2].Cell(wordDoc.Tables[2].Rows.Count, 4).Range.Text = dtReportData.Rows[i]["TestResult"].ToString();
        subItemName += dtReportData.Rows[i]["SubitemName"].ToString() + ";";  //子项目名称
    }

    /*----------------更新子项目(检测位点)---------------------------*/

    wordApp.Selection.Find.ClearFormatting();
    wordApp.Selection.Find.Text = "{Subitem}";
    wordApp.Selection.Find.Replacement.ClearFormatting();
    wordApp.Selection.Find.Replacement.Text = subItemName;
    wordApp.Selection.Find.Execute(missing, missing, missing, missing,
                                    missing, missing, missing, missing,
                                    missing, missing, replaceAll, missing,
                                    missing, missing, missing);   //执行替换

    string filePath = Settings.Default.ReportSavePath + "\\" + customer.Hospital + "_" + customer.BarCode + "(" + customer.Name + ")_" + item.SN + "(" + item.ChineseName + ")_" + DateTime.Now.ToString("yyyyMMdd HHmm", FormatInfo.DateTime);

    wordDoc.SaveAs2(filePath + ".docx");

    wordDoc.ExportAsFixedFormat(filePath + ".pdf", Word.WdExportFormat.wdExportFormatPDF);

    PGxMySqlDB.UpdateCustomerTestStatus(customer.ID, Status.Printed);   //客户状态更新为已打印
    Logs.AddRecordEdit(customer.ID, Strings.Print); //增加打印信息

}
catch (Exception err)
{
    
    throw;
}
finally
{
    
    wordDoc.Close(); //关闭文件
}

}//for (int i = 0; i < dtRecordItem.Rows.Count; i++)

wordApp.Quit();   //退出应用

3.在页眉插入照片

            string logoFile = HospitalLogo.GetLogoFilePth();
            if (wordApp == null || string.IsNullOrEmpty(logoFile)) return;

            wordApp.ActiveWindow.View.SeekView = Word.WdSeekView.wdSeekCurrentPageHeader;   //编辑页眉
            wordApp.Selection.HeaderFooter.LinkToPrevious = false;

            Object _Range = wordApp.Selection.Range;  //要向Word文档中插入图片的位置
            Object isLinkToFile = false; // 定义该插入的图片是否为外部链接默认,这里貌似设置为bool类型更清晰一些  
            Object isSaveWithDcm = true; //定义要插入的图片是否随Word文档一起保存
            //插入照片,注意InlineShapes对应不应调用Docement的,否则会添加到正文中。
            wordApp.Selection.InlineShapes.AddPicture(logoFile,isLinkToFile,isSaveWithDcm,_Range);
            wordApp.Selection.HeaderFooter.Range.Paragraphs.Format.Borders.Enable = 0;      //无边框
            wordApp.Selection.HeaderFooter.Range.Paragraphs.Format.Alignment = Word.WdParagraphAlignment.wdAlignParagraphLeft;  //段落左对其

            //获取照片对象,注意InlineShapes对应不应调用Docement的,否则会读取错误。
            //处理图片时需要单独获取图片后再进行编辑。
            Word.Shape logo = wordApp.Selection.HeaderFooter.Range.InlineShapes[1].ConvertToShape();  
            logo.WrapFormat.Type = Word.WdWrapType.wdWrapNone;      //衬于文字下方
            logo.LockAspectRatio = Microsoft.Office.Core.MsoTriState.msoCTrue;  //锁定长宽比
            logo.Height = 40;   //照片缩小,固定高度为40磅
            logo.Select();
            logo.Top = -30;
            logo.Left = 0;

			wordApp.Selection.SetRange(0,0);    //取消选择

            wordApp.ActiveWindow.View.SeekView = Word.WdSeekView.wdSeekMainDocument;   //编辑主文档

4.在主页面指定位置插入图片

            string logoFile = HospitalLogo.GetLogoFilePth();
            if (wordDoc == null || string.IsNullOrEmpty(logoFile)) return;

            Object _Range = wordDoc.Range(0,0);  //要向Word文档中插入图片的位置
            Object isLinkToFile = false; // 定义该插入的图片是否为外部链接默认,这里貌似设置为bool类型更清晰一些  
            Object isSaveWithDcm = true; //定义要插入的图片是否随Word文档一起保存
            //插入照片,注意InlineShapes对应不应调用Docement的,否则会添加到正文中。
            wordDoc.InlineShapes.AddPicture(logoFile, isLinkToFile, isSaveWithDcm, _Range);

            //调用图片,调用图片建议采用rang范围获取才能准确获取
            Word.Shape logo = wordDoc.Range(0,1).InlineShapes[1].ConvertToShape();	

            logo.WrapFormat.Type = Word.WdWrapType.wdWrapNone;      //衬于文字下方
            logo.LockAspectRatio = Microsoft.Office.Core.MsoTriState.msoCTrue;  //锁定长宽比
            logo.Height = 100;   //照片缩小,固定高度为100磅
            logo.Select();
            logo.Top = -45;
            logo.Left = 0;

注意
上文获取图片是实验后可以准确获取的方式,原来采用直接document最后一张图片获取的方式,若文件后面还有其他图片会获取错误。

Word.Shape logo = wordDoc.InlineShapes[wordDoc.InlineShapes.Count].ConvertToShape();

实际使用过程中发现,word插入图片后图片的顺序会动态变化。

在这里插入图片描述
原来图片A是wordDoc.InlineShapes[1],图片B的位置的wordDoc.InlineShapes[2];
在这里插入图片描述
在图片A和B之间插入图片C,图片的顺序跟随变更:
图片A的位置是wordDoc.InlineShapes[1];
图片B的位置是wordDoc.InlineShapes[3];
图片C的位置是wordDoc.InlineShapes[2];

5.设置表格自动调整格式

table.AutoFitBehavior(Word.WdAutoFitBehavior.wdAutoFitWindow);

样式属性如下:
wdAutoFitContent 1 根据表格中包含的内容自动调整表格的大小。
wdAutoFitFixed 0 将表格设置为固定大小而与内容无关,因此不会自动调整表格大小。
wdAutoFitWindow 2 根据活动窗口的宽度自动调整表格大小。

踩坑记录:

5.1 AutoFitBehavior失效,无法自动调整表格。

实际测试过程中发现应用了方法后,无法自动调整的问题,但是同时在其他电脑又可以正常操作,或者将App.Visible设置为true之后就可以正常,没有找到具体的问题,后面将office卸载重装还是没有解决,最后修改操作逻辑,把就表格删除,然后重新添加表格才解决问题

	Word.Table table = wordDoc.Tables[1]; //获取文件中的第一个表格,表格索引从1开始
	Word.Range tableRange = table.Range;
	table.Rows.Delete();
	table = wordDoc.Tables.Add(tableRange, 5, 6, Word.WdDefaultTableBehavior.wdWord9TableBehavior, Word.WdAutoFitBehavior.wdAutoFitWindow);
	
	//修改边框及字体
	table.Borders.OutsideLineStyle = Word.WdLineStyle.wdLineStyleNone;
	table.Borders.InsideLineStyle = Word.WdLineStyle.wdLineStyleNone;
	table.Borders[Word.WdBorderType.wdBorderTop].LineStyle = Word.WdLineStyle.wdLineStyleSingle;
	table.Borders[Word.WdBorderType.wdBorderBottom].LineStyle = Word.WdLineStyle.wdLineStyleSingle;
	table.Borders[Word.WdBorderType.wdBorderHorizontal].LineStyle = Word.WdLineStyle.wdLineStyleSingle;
	table.Range.Font.Size = 12;     //14磅字体
	table.Range.Font.Bold = 0;      //字体不加粗

6.设置列自动试应列宽

            table.Columns[5].AutoFit();
            table.Columns[7].AutoFit();

使用AutoFit()方法设置对应列自动调整列宽,注意添加文本后再设置才能生效。

7.设置段落-允许西文在单词中间换行

word有时候为了格式会进行段落设置,允许单词中间换行,可以最大程度的保持表格、段落的基本样式,不至于过度挤压文本内容。
在这里插入图片描述
刚开始在网上搜了半天,都是使用word打开然后段落设置,勾选,这一套标准操作,但是我需要使用软件代码实现,找了很久都没有找到解决方案。
既然找不到解决方案,那就自己弄解决方案。
首先我查找了下微软的官方文档,找到段落样式说明:ParagraphFormat 接口
找了半天,发现了一个WordWrap属性,用于设置允许西文中间断字换行。
在这里插入图片描述
结果点进去发现就短短几行字,说明属性值设置为True,但是属性的数据类型是个Int32类型。工程师直接VBA操作文档直接抄过来了,改都懒得改。
在这里插入图片描述
既然有属性值可以使用,那我就试试了,先从1开始,结果直接报错,2,3,4,5一样全部报错。崩溃了。
为了找到属性值,我用word设置了允许断行和不允许断行,然后调试,直接查看具体的属性值差异。最终发现:

  • 不允许断行:WordWrap属性为-1
  • 允许断行: WordWrap属性为0
    然后代码将该属性设置为0.
    结果还是设置不了
            //允许西文在单词中间换行
            table.Range.Paragraphs.WordWrap = 0;
            table.Range.Paragraphs[1].WordWrap = 0;
            table.Cell(1, 4).Range.Paragraphs[1].WordWrap = 0;

通过各种参数设置,以及先选中,然后再用选中样式选择段落设置进行修改。还是不起作用。
一度以为这是个假的属性。
后面查看代码,根据之前使用的一起经验,猜测会不会跟属性的调用时间和位置有影响。
我尝试这把代码换位置调用,多次尝试之后发现,需要在自动匹配的方法调用之前设置才能生效。最终调整如下:

            //允许西文在单词中间换行,需要放置于AutoFit相关函数前面才能生效。
            table.Range.Paragraphs.WordWrap = 0;

            //自动匹配宽度
            table.AutoFitBehavior(Word.WdAutoFitBehavior.wdAutoFitContent);
            //自动调整-根据窗体
            table.AutoFitBehavior(Word.WdAutoFitBehavior.wdAutoFitWindow);
            table.Columns[1].SetWidth(40, Word.WdRulerStyle.wdAdjustFirstColumn);
            table.Columns[2].Width = 60;
            table.Columns[3].SetWidth(60, Word.WdRulerStyle.wdAdjustFirstColumn);
            table.Columns[4].Width = 100;
            table.Columns[5].SetWidth(75, Word.WdRulerStyle.wdAdjustFirstColumn);
            table.Columns[6].Width = 80;

            //居中
            table.Rows.Alignment = Word.WdRowAlignment.wdAlignRowCenter;
            //自动调整高度
            table.Rows.HeightRule = Word.WdRowHeightRule.wdRowHeightAuto;

            //合并最后后一行单元格
            table.Cell(5, 2).Merge(table.Cell(5, 4));
            table.Cell(5, 2).Range.ParagraphFormat.Alignment = Word.WdParagraphAlignment.wdAlignParagraphLeft;  //左对齐

实现效果如下:
在这里插入图片描述

8.设置字体颜色及自定义字体颜色

8.1 使用系统枚举设置颜色:

range.Font.Color = Word.WdColor.wdColorBlue;

具体颜色详见官网:https://learn.microsoft.com/zh-cn/dotnet/api/microsoft.office.interop.word.wdcolor?view=word-pia

8.2 设置成自定义的颜色

如果要设置自定义的颜色,需要将颜色转换成对应的int值,然后再强制转换成WdColor类型。

int intColor = Color.R + Color.G * 0xFF + Color.B * 0xFFFF;	//需要注意RGB对应的乘数值。
range.Font.Color = (Word.WdColor)intColor;

9.设置单元格文本居中对齐

Word.Table.Range.Cells.VerticalAlignment = Word.WdCellVerticalAlignment.wdCellAlignVerticalCenter;
wordDoc.Tables[2].Cell(wordDoc.Tables[2].Rows.Count, 1).VerticalAlignment = Word.WdCellVerticalAlignment.wdCellAlignVerticalCenter;

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u010839204/article/details/108277869

智能推荐

蓝凌EIS智慧协同平台saveImg接口存在任意文件上传漏洞_蓝凌eis智慧协同平台文件上传漏洞-程序员宅基地

文章浏览阅读979次。蓝凌智慧协同平台eis集合了非常丰富的模块,满足组织企业在知识、协同、项目管理系统建设等需求。_蓝凌eis智慧协同平台文件上传漏洞

LLaVA-1.5-程序员宅基地

文章浏览阅读193次。与InstructBLIP或Qwen-VL在数亿甚至数十几亿的图像文本配对数据上训练的、专门设计的视觉重新采样器相比,LLaVA用的是最简单的LMM架构设计,只需要在600K个图像-文本对上,训练一个简单的完全连接映射层即可。结果表明,LLaVA-1.5不仅可以使用更少的预训练和指令微调数据,而且还可以利用最简单的架构、学术计算和公共数据集来实现最佳的性能——在12个基准中的11个上取得了SOTA。为了解决这个问题,研究人员建议在VQA问题的末尾,添加一个可以明确输出格式的提示,进而让模型生成简短回答。

ORACLE基本数据类型总结_oracle 数值类型最大值-程序员宅基地

文章浏览阅读442次。2013-08-17 21:04 by 潇湘隐者, 100246 阅读, 5 评论, 收藏, 编辑 ORACLE基本数据类型(亦叫内置数据类型 built-in datatypes)可以按类型分为:字符串类型、数字类型、日期类型、LOB类型、LONG RAW& RAW类型、ROWID & UROWID类型。在讲叙字符串类型前,先要讲一下编码。字符串类型的数据可依编码方式分成_oracle 数值类型最大值

10种机器学习算法_决策树和mlp-程序员宅基地

文章浏览阅读315次。作为数据科学家的实践者,我们必须了解一些通用机器学习的基础知识算法,这将帮助我们解决所遇到的新领域问题。本文对通用机器学习算法进行了简要的阐述,并列举了它们的相关资源,从而帮助你能够快速掌握其中的奥妙。▌1.主成分分析(PCA)/ SVDPCA是一种无监督的方法,用于对由向量组成的数据集的全局属性进行理解。本文分析了数据点的协方差矩阵,以了解哪些维度(大部分情况)/数据点(少数情况)更为重要,即它..._决策树和mlp

桥接模式的实现-程序员宅基地

文章浏览阅读148次。在这个示例中,我们使用std::shared_ptr来管理Implementor对象的生命周期,确保在不再需要时自动释放资源。通过智能指针的使用,我们避免了手动管理内存的复杂性,提高了代码的可靠性和可维护性。希望这个示例能帮助你理解如何使用智能指针来实现桥接模式。当使用智能指针来实现桥接模式时,我们可以利用std::shared_ptr或std::unique_ptr来管理对象的生命周期,确保资源的正确释放。

制造业敏感文件外发不安全?一招解锁更高效的加密方式!-程序员宅基地

文章浏览阅读440次,点赞11次,收藏8次。云盒子在制造业上有丰富的部署经验,在面向制造类企业的重要文件,可以通过审计、授权、文件加密进行多重保护,使得图纸文件、专利技术、采购订单等敏感数据等到有效保护,做到无处可泄,同时安全可靠,也不会对日常工作效率有影响 ,实现真正有效的企业文件保护的目的,达到既防止机密文件外泄和扩散,又支持内部知识积累和文件共享的目的。云盒子的加密方式是通过将本地文件数据上传到云盘进行统一加密存储,而不是对设备加密,通过【本地加密】+【云加密】双重组合下,不管用什么设备打开文件都受到管控,使管理者管理起来能够更高效。

随便推点

计算几何讲义——计算几何中的欧拉定理-程序员宅基地

文章浏览阅读188次。在处理计算几何的问题中,有时候我们会将其看成图论中的graph图,结合我们在图论中学习过的欧拉定理,我们可以通过图形的节点数(v)和边数(e)得到不是那么好求的面数f。 平面图中的欧拉定理: 定理:设G为任意的连通的平面图,则v-e+f=2,v是G的顶点数,e是G的边数,f是G的面数。证明:其实有点类似几何学中的欧拉公式的证明方法,这里采用归纳证明的方法。对m..._怎么证明平面图欧拉定理

c语言中各种括号的作用,C语言中各种类型指针的特性与用法介绍-程序员宅基地

文章浏览阅读750次。C语言中各种类型指针的特性与用法介绍本文主要介绍了C语言中各种类型指针的特性与用法,有需要的朋友可以参考一下!想了解更多相关信息请持续关注我们应届毕业生考试网!指针为什么要区分类型:在同一种编译器环境下,一个指针变量所占用的内存空间是固定的。比如,在16位编译器环境 下,任何一个指针变量都只占用8个字节,并不会随所指向变量的类型而改变。虽然所有的指针都只占8个字节,但不同类型的变量却占不同的字节数..._c语言带括号指针

缅甸文字库 缅甸语字库 缅甸字库算法_0x103c-程序员宅基地

文章浏览阅读9.5k次。字库交流 QQ:2229691219 缅甸语比较特殊、缅甸语有官方和民间之分,二者不同的是编码机制不同,因此这2种缅甸语的字串翻译、处理引擎、字库都是不同的。我们这里只讨论官方语言。 缅文、泰文等婆罗米系文字大多是元音附标文字,一般辅音字母自带默认元音可以发音,真正拼写词句时元音像标点符号一样附标在辅音上下左右的相应位置。由于每个元音位于辅音的具体位置是有自己的规则的,当只书写..._0x103c

Python+django+vue校园二手闲置物品拍卖系统pycharm毕业设计项目推荐_基于python+django+vue实现的校园二手交易平台-程序员宅基地

文章浏览阅读200次。在校园,随着学生数量的增多,存在许多生活和学习物品,许多学习用品经过一学期学习之后往往被闲置,一些出于一时喜欢而购买的物品使用机会少而被闲置,还有一些物品以低廉的价格卖给资源回收站,造成巨大的资源浪费。校园闲置物品拍卖系统使用python技术,MySQL数据库进行开发,系统后台使用django框架进行开发,具有低耦合、高内聚的特点,其中校园用户通过人脸识别的方法增加系统安全性,在闲置物品推荐中,使用协同过滤算法进行商品推荐。系统的开发,帮助高校有效的对闲置物品进行管理,提高了闲置物品销售的效率。_基于python+django+vue实现的校园二手交易平台

【推荐系统论文精读系列】(十)--Wide&Deep Learning for Recommender Systems_引用《wide & deep learning for recommender systems》-程序员宅基地

文章浏览阅读1.1k次,点赞3次,收藏3次。文章目录Wide & Deep Learning for Recommender Systems一、摘要二、介绍三、推荐系统综述四、Wide&Deep学习4.1 Wide部分4.2 Deep部分4.3 联合训练 Wide&Deep ModelPreferenceWide & Deep Learning for Recommender Systems一、摘要具有非线性特征转化能力的广义线性模型被广泛用于大规模的分类和回归问题,对于那些输入数据是极度稀疏的情况下。通过使用交_引用《wide & deep learning for recommender systems》

c++ sleep函数_Linux 多线程应用中如何编写安全的信号处理函数-程序员宅基地

文章浏览阅读171次。关于代码的可重入性,设计开发人员一般只考虑到线程安全,异步信号处理函数的安全却往往被忽略。本文首先介绍如何编写安全的异步信号处理函数;然后举例说明在多线程应用中如何构建模型让异步信号在指定的线程中以同步的方式处理。Linux 多线程应用中编写安全的信号处理函数在开发多线程应用时,开发人员一般都会考虑线程安全,会使用 pthread_mutex 去保护全局变量。如果应用中使用了信号,而且信号的产生不..._linux c++ sleep 不被中断

推荐文章

热门文章

相关标签