dotnet OpenXml SDK 形狀填充漸變色的主題色

来源:https://www.cnblogs.com/lindexi/archive/2020/05/21/12932707.html
-Advertisement-
Play Games

在 Office 文檔的一些有趣的設計,顏色和畫刷是可以繼承的,這個繼承包括了屬性的繼承。在形狀填充裡面使用的漸變色是可以一部分屬性放在主題裡面,主要找到主題裡面的畫刷,替換掉形狀自己定義的內容,才是形狀的畫刷 ...


在 Office 文檔的一些有趣的設計,顏色和畫刷是可以繼承的,這個繼承包括了屬性的繼承。在形狀填充裡面使用的漸變色是可以一部分屬性放在主題裡面,主要找到主題裡面的畫刷,替換掉形狀自己定義的內容,才是形狀的畫刷

我拿到一份有趣的課件,從這份課件的表現上,可以找到在一個 Shape 元素裡面的 ShapeProperties 定義了 GradientFill 屬性

// OpenXmlElementList element
var gradientFill = element.First<GradientFill>();

而這個 gradientFill 的 GradientStopList 的值是空

if (gradientFill.GradientStopList != null)
{

}

通過 Office Open XML - DrawingML - Shapes - Gradient Fill 可以知道,這個 GradientStopList 是用來定義漸變的顏色的,也就是這個值是空那麼就是漸變將會丟失顏色,也就是形狀填充丟失漸變色

從文檔上看,形狀屬性定義裡面沒有 gsLst 的值

<p:spPr>
	<a:xfrm>
		<a:off x="611560" y="1059582"/>
		<a:ext cx="2120518" cy="645160"/>
	</a:xfrm>
	<a:prstGeom prst="rect">
		<a:avLst/>
	</a:prstGeom>
	<a:gradFill flip="none" rotWithShape="1">
		<a:lin ang="10800000" scaled="1"/>
		<a:tileRect/>
	</a:gradFill>
</p:spPr>

也就是 a:gradFill 裡面不包含 a:gsLst 的值,也就是沒有 a:gs 定義顏色

這個問題是 OpenXML a:gradFill without a:gsLst 漸變色裡面沒有 GradientStopList 的值

而在這個形狀一般都可以看到 ShapeStyle 的值,這個值裡面有一個屬性是 FillReference 表示樣式裡面的填充

<p:style>
	<a:fillRef idx="2">
		<a:schemeClr val="accent3"/>
	</a:fillRef>
</p:style>

而這個樣式要的屬性是 idx 屬性,表示屬於主題的哪個樣式

那麼在 OpenXML SDK 裡面如何獲取主題,如果是在 Slide 頁面的元素可以通過下麵的代碼獲取主題

// Slide slide
var slidePart = slide.SlidePart;
FormatScheme formatScheme = slidePart?.ThemeOverridePart?.ThemeOverride?.FormatScheme;

當前,如果 Slide 拿不到就從 SlideLayoutPart 拿,再拿不到就從 SlideMasterPart 拿

然後是通過 FillReference 的 idx 從 FormatScheme 的 FillStyleList 拿到指定的元素,註意這裡的 idx 使用的是從 1 開始的下標。但是 FillStyleList 的數組是 C# 的數組,下標是從 0 開始

// FillReference reference, FormatScheme formatScheme
    if (reference.Index != null && formatScheme != null)
    {
        var index = (int) reference.Index.Value;

        var openXmlElementList = formatScheme.FillStyleList?.ChildElements;

        if (openXmlElementList != null)
        {
            return GetThemeElement(index, openXmlElementList);
        }
    }

而 GetThemeElement 方法的實現如下

        private static OpenXmlElement GetThemeElement(int index, OpenXmlElementList elements)
        {
            if (index > 0 && elements != null && elements.Count >= index)
            {
                //GetItem是0 base的數組,所以需要減去1
                var xmlElement = elements.GetItem(index - 1);

                return xmlElement;
            }

            return null;
        }

此時就能拿到 OpenXmlElement 返回值了,這個 OpenXmlElement 是某個填充顏色。在 OpenXML SDK 裡面沒有給填充顏色一個基類,這部分的設計不是很好

在 PPT 文檔裡面對應的是 ppt\theme\themex.xml 文件裡面的 a:fillStyleLst 的值

<a:fmtScheme name="Office">
	<a:fillStyleLst>
		<a:solidFill>
			<a:schemeClr val="phClr"/>
		</a:solidFill>
		<a:gradFill rotWithShape="1">
			<a:gsLst>
				<a:gs pos="0">
					<a:schemeClr val="phClr">
						<a:tint val="50000"/>
						<a:satMod val="300000"/>
					</a:schemeClr>
				</a:gs>
				<a:gs pos="35000">
					<a:schemeClr val="phClr">
						<a:tint val="37000"/>
						<a:satMod val="300000"/>
					</a:schemeClr>
				</a:gs>
				<a:gs pos="100000">
					<a:schemeClr val="phClr">
						<a:tint val="15000"/>
						<a:satMod val="350000"/>
					</a:schemeClr>
				</a:gs>
			</a:gsLst>
			<a:lin ang="16200000" scaled="1"/>
		</a:gradFill>
		<!-- 忽略代碼 -->

上面文檔裡面用的是 idx 是 2 也就是對應 a:fillStyleLst 的第二項,也就是 a:gradFill 漸變的值

此時的主題的 a:gradFilla:gsLst 將會被形狀的填充用到,如果形狀的填充的顏色也是漸變色,如果這個漸變色沒有設置 a:gsLst 的值,那麼將會採用主題裡面的 a:gsLst 的值。如果形狀自己定義了就使用形狀定義的

請看下圖就知道如何獲取

大部分的存在繼承關係和聯繫的都在 OpenXML SDK 裡面寫出來關係了,只有這些比較邊角的功能需要自己實現

而漸變色的各個屬性的行為請看 Office Open XML - DrawingML - Shapes - Gradient Fill

官方文檔請看 GradientFill Class (DocumentFormat.OpenXml.Drawing)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 Step1:基礎數據準備(通過爬蟲獲取到),以下是從第一期03年雙色球開獎號到今天的所有數據整理,截止目前一共2549期,balls.txt 文件內容如下 : 備註:想要現成 ...
  • 給大家分享這幾個月整理出來的Java程式員面試筆試真題,順便給大家分享一本電子書《Java程式員面試筆試寶典》需要的可以掃描文末的二維碼即可,話不多說給大家看看部分截圖。 ...
  • 這份資源我自己歷經三年才整理歸類出來,現在免費分享給大家; 面試題有:螞蟻金服、拼多多、阿裡雲、百度、唯品會、攜程、豐巢科技、樂信、軟通動力、OPPO、銀盛支付、中國平安等初,中級,高級Java面試題集合。 面試題以及答案,已經整理成PDF電子書形式打包在網盤; 面試題領取微信掃一掃,加好友請備註“ ...
  • 在秒殺,搶購等併發場景下,可能會出現超賣的現象,在 PHP 語言中並沒有原生提供併發的解決方案,因此就需要藉助其他方式來實現併發控制。 列出常見的解決方案有: 使用隊列,額外起一個進程處理隊列,併發請求都放到隊列中,由額外進程串列處理,併發問題就不存在了,但是要額外進程支持以及處理延遲嚴重,本文不先 ...
  • 6.32(游戲:贏取雙骰子賭博游戲的機會)修改編程練習題6.30使該程式運行10000次,然後顯示贏得游戲的次數 6.32(Game: chance of winning at craps)Revise Exercise 6.30 to run it 15,000 times and display ...
  • 異常有可能導致某些資源沒有釋放到,這時可以使用using或者finally來釋放資源。 有些異常的可能會間歇性的發生,不影響業務運行的,可以視為正常的邏輯,那在異常可能發生前,應該使用判斷邏輯。 if (conn.State != ConnectionState.Closed) { conn.Clo ...
  • SunnyUI.Net 開發日誌:ListBox 增加跟隨滑鼠滑過高亮,這可是在別人兩年的基礎上再改進的,哈哈!~ ...
  • 什麼是Pub Sub 發佈訂閱是一種設計模式,它允許應用程式組件之間進行鬆散耦合。 其實訂閱發佈設計中主要是發佈者生成事件通道,用於在不瞭解任何訂閱者存在的情況下通知訂閱者。 當然委托EventHandlers和Event關鍵字在此事件處理機制中擔任著重要的角色。下麵我們來看看如何使用它們。 Pub ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...