惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Google DeepMind News
Google DeepMind News
Stack Overflow Blog
Stack Overflow Blog
Hugging Face - Blog
Hugging Face - Blog
博客园_首页
T
The Blog of Author Tim Ferriss
博客园 - 叶小钗
N
Netflix TechBlog - Medium
腾讯CDC
C
Check Point Blog
P
Proofpoint News Feed
Engineering at Meta
Engineering at Meta
GbyAI
GbyAI
S
SegmentFault 最新的问题
F
Fortinet All Blogs
美团技术团队
U
Unit 42
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
博客园 - 司徒正美
F
Full Disclosure
Recorded Future
Recorded Future
D
DataBreaches.Net
博客园 - 【当耐特】
Martin Fowler
Martin Fowler
J
Java Code Geeks
I
InfoQ
Y
Y Combinator Blog
A
About on SuperTechFans
AI
AI
爱范儿
爱范儿
Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
Forbes - Security
Forbes - Security
W
WeLiveSecurity
M
MIT News - Artificial intelligence
雷峰网
雷峰网
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Simon Willison's Weblog
Simon Willison's Weblog
Schneier on Security
Schneier on Security
The GitHub Blog
The GitHub Blog
Security Archives - TechRepublic
Security Archives - TechRepublic
aimingoo的专栏
aimingoo的专栏
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
G
GRAHAM CLULEY
Know Your Adversary
Know Your Adversary
Latest news
Latest news
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
D
Docker
Recent Commits to openclaw:main
Recent Commits to openclaw:main
量子位
V2EX - 技术
V2EX - 技术
Project Zero
Project Zero

博客园 - 反骨少年

cursor使用 wpf 进度条 https://docs.ultralytics.com/zh/tasks/detect/#export 博文阅读密码验证 - 博客园 博文阅读密码验证 - 博客园 博文阅读密码验证 - 博客园 博文阅读密码验证 - 博客园 vue 前端导出excel vue+.netCore 下载导出文件 SVN提交过滤忽略的文件 博文阅读密码验证 - 博客园 OpenTelemetry 博文阅读密码验证 - 博客园 vue3+ElementPlus 后台布局搭建 IdentityServer4 canal管理 数据同步 Canal数据同步Kafka Xcode的思想,以及能让我学习的规范 阿里云的CICD
通过outxml 复制sheet ,不完整
反骨少年 · 2025-04-23 · via 博客园 - 反骨少年
    internal class Program
    {
        static void Main(string[] args)
        {
            //CopySheet(@"D:\Works\2025-03-24 8D\2025-03-24\test.xlsx", "Sheet1", "8D表格 Copy");
            //CopySheet(@"D:\Works\2025-03-24 8D\2025-03-24\KP_TEMPLATE.xlsx", "8D表格", "8D表格 Copy");
            CopySheet(@"D:\Works\2025-03-24 8D\2025-03-24\开平依利安达-8D不合格 VCAR_for panasonic (003) .xlsx", "8D表格", "8D表格 Copy");
            Console.WriteLine("Hello World!");
        }

        static void CopySheet(string file, string sheetName, string newSheetName)
        {
            MemoryStream stream = new MemoryStream();
            byte[] buff = File.ReadAllBytes(file);
            stream.Write(buff, 0, buff.Length);

            SpreadsheetDocument doc = SpreadsheetDocument.Open(stream, true);
            Sheet sheetRelation = null;
            uint maxSheetId = 0;
            uint sourceSheetIndex = 0;
            uint maxSheetCount = 0;
            foreach (OpenXmlElement s in doc.WorkbookPart.Workbook.Sheets)
            {
                if (s is Sheet st)
                {
                    if (st.Name?.Value == sheetName)
                    {
                        sheetRelation = st;
                        sourceSheetIndex = maxSheetCount;
                    }

                    //记录最大id
                    if (st.SheetId?.Value != null)
                    {
                        if (st.SheetId.Value > maxSheetId)
                            maxSheetId = st.SheetId.Value;
                    }
                    maxSheetCount++;
                }
            }

            if (sheetRelation == null)
                throw new Exception("sheet not found");

            WorksheetPart sheetPart = doc.WorkbookPart.GetPartById(sheetRelation.Id) as WorksheetPart;
            Worksheet sheet = sheetPart.Worksheet;

            WorksheetPart newSheetPart = doc.WorkbookPart.AddNewPart<WorksheetPart>();
            newSheetPart.Worksheet = new Worksheet(sheet.OuterXml);
            SheetView sheetView = (SheetView?)newSheetPart.Worksheet.SheetViews?.Elements().FirstOrDefault();
            if (sheetView != null && sheetView.TabSelected != null)
            {
                sheetView.TabSelected = false;
            }

            var listNewParts = new List<OpenXmlPart>();
            foreach (IdPartPair idPart in sheetPart.Parts)
            {
                OpenXmlPart openxmlPart = null;
                if (idPart.OpenXmlPart is EmbeddedObjectPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<EmbeddedObjectPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                }
                else if (idPart.OpenXmlPart is VmlDrawingPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<VmlDrawingPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                    listNewParts.Add(openxmlPart);
                    foreach (IdPartPair vmlChildPart in idPart.OpenXmlPart.Parts)
                    {
                        openxmlPart.CreateRelationshipToPart(vmlChildPart.OpenXmlPart, vmlChildPart.RelationshipId);
                    }
                }
                else if (idPart.OpenXmlPart is DrawingsPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<DrawingsPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                    foreach (IdPartPair drawingChildPart in idPart.OpenXmlPart.Parts)
                    {
                        openxmlPart.CreateRelationshipToPart(drawingChildPart.OpenXmlPart, drawingChildPart.RelationshipId);
                    }
                    listNewParts.Add(openxmlPart);
                }
                else if (idPart.OpenXmlPart is SpreadsheetPrinterSettingsPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<SpreadsheetPrinterSettingsPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                }
                else if (idPart.OpenXmlPart is WorksheetThreadedCommentsPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<WorksheetThreadedCommentsPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                }
                else if (idPart.OpenXmlPart is WorksheetCommentsPart)
                {
                    openxmlPart = newSheetPart.AddNewPart<WorksheetCommentsPart>(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                }
                // else if (idPart.OpenXmlPart is ImagePart)
                // {
                //     openxmlPart = newSheetPart.AddNewPart(idPart.OpenXmlPart.ContentType, idPart.RelationshipId);
                // }

                if (openxmlPart != null)
                    openxmlPart.FeedData(idPart.OpenXmlPart.GetStream());
                else
                    openxmlPart = idPart.OpenXmlPart;
                newSheetPart.CreateRelationshipToPart(openxmlPart, idPart.RelationshipId);
            }

            ResetOleId(sheet, newSheetPart.Worksheet, listNewParts);
            if (newSheetPart.Parts.Any(d => d.OpenXmlPart is WorksheetThreadedCommentsPart) &&
                newSheetPart.Parts.Any(d => d.OpenXmlPart is WorksheetCommentsPart))
            {
                ResetCommentId(newSheetPart.Parts.First(d => d.OpenXmlPart is WorksheetCommentsPart).OpenXmlPart as WorksheetCommentsPart,
                    newSheetPart.Parts.First(d => d.OpenXmlPart is WorksheetThreadedCommentsPart).OpenXmlPart as WorksheetThreadedCommentsPart);
            }

            Sheet newSheetRelation = new Sheet();
            newSheetRelation.SheetId = (uint)(maxSheetId + 1);
            newSheetRelation.Id = doc.WorkbookPart.GetIdOfPart(newSheetPart);
            newSheetRelation.Name = newSheetName;
            doc.WorkbookPart.Workbook.Sheets.AppendChild(newSheetRelation);

            doc.Save();

            stream.Seek(0, SeekOrigin.Begin);
            buff = new byte[stream.Length];
            stream.Read(buff, 0, buff.Length);
            stream.Close();
            File.WriteAllBytes($@"{Path.GetDirectoryName(file)}\{Path.GetFileNameWithoutExtension(file)}-Copy{Path.GetExtension(file)}", buff);
        }

        static void ResetOleId(Worksheet sheetOld, Worksheet sheetNew, List<OpenXmlPart> newParts)
        {
            uint maxOleId = 0;
            OleObjects oles = sheetOld.Elements<OleObjects>().FirstOrDefault();
            if (oles != null)
            {
                var ids = oles.Elements().Select(d => d.Descendants<OleObject>().FirstOrDefault()?.ShapeId?.Value).Where(d => d != null).Select(d => d.Value);
                if (ids.Any())
                {
                    maxOleId = ids.Max();
                }
            }

            if (maxOleId < 1)
                return;

            var dic = new Dictionary<uint, uint>();
            foreach (AlternateContent alternateContent in sheetNew.Elements<DocumentFormat.OpenXml.Spreadsheet.OleObjects>().Last()?.Elements<AlternateContent>())
            {
                foreach (OleObject oleObject in alternateContent.Descendants<OleObject>())
                {
                    if (oleObject.ShapeId == null)
                        continue;

                    if (dic.ContainsKey(oleObject.ShapeId.Value))
                    {
                        oleObject.ShapeId = dic[oleObject.ShapeId.Value];
                    }
                    else if (oleObject.ShapeId <= maxOleId)
                    {
                        maxOleId = maxOleId * 2;
                        dic.Add(oleObject.ShapeId.Value, maxOleId);
                        oleObject.ShapeId = maxOleId;
                    }
                }
            }

            foreach (OpenXmlPart part in newParts)
            {
                Stream stream = part.GetStream();
                string str = GetStreamString(stream);
                foreach (KeyValuePair<uint, uint> item in dic)
                {
                    str = str.Replace($"s{item.Key}", $"s{item.Value}");
                    str = str.Replace($"id=\"{item.Key}\"", $"id=\"{item.Value}\"");
                }

                stream.Seek(0, SeekOrigin.Begin);
                stream.Write(Encoding.UTF8.GetBytes(str));
                stream.Close();
            }
        }

        static void ResetCommentId(WorksheetCommentsPart commentsPart, WorksheetThreadedCommentsPart threadedCommentsPart)
        {
            if (commentsPart.Comments.CommentList == null)
                return;

            var dic = new Dictionary<string, string>();
            foreach (Comment comment in commentsPart.Comments.CommentList.Elements())
            {
                try
                {
                    OpenXmlAttribute att = comment.GetAttribute("uid", "http://schemas.microsoft.com/office/spreadsheetml/2014/revision");
                    string id = att.Value;
                    string newId = Guid.NewGuid().ToString("B").ToUpper();

                    OpenXmlAttribute attNew = new OpenXmlAttribute("xr", "uid", "http://schemas.microsoft.com/office/spreadsheetml/2014/revision", newId);
                    comment.RemoveAttribute("uid", "http://schemas.microsoft.com/office/spreadsheetml/2014/revision");
                    comment.SetAttribute(attNew);
                    dic.Add(id, newId);
                }
                catch (Exception e)
                {
                    continue;
                }
            }

            foreach (ThreadedComment comment in threadedCommentsPart.ThreadedComments.Elements())
            {
                try
                {
                    if (comment.Id == null)
                        continue;

                    if (dic.ContainsKey(comment.Id.Value))
                        comment.Id = dic[comment.Id.Value];
                }
                catch (Exception e)
                {
                    continue;
                }
            }
        }


        static string GetStreamString(Stream stream)
        {
            stream.Seek(0, SeekOrigin.Begin);
            byte[] buff = new byte[stream.Length];
            stream.Read(buff, 0, buff.Length);
            return Encoding.UTF8.GetString(buff);

        }
    }