ClosedXML - Excelファイルの操作

提供:MochiuWiki : SUSE, EC, PCB
2024年9月10日 (火) 17:48時点におけるWiki (トーク | 投稿記録)による版 (→‎条件付き書式の適用)
ナビゲーションに移動 検索に移動

概要

ClosedXMLは、Microsoft Excelファイルの作成、読み取り、編集を簡単に行うことができるように設計されている。

ClosedXMLの主な特徴は、その使いやすさにある。
開発者は、Excelファイルの操作をプログラム的に行う場合に、複雑なExcel Object Modelを直接扱う必要がない。
代わりに、ClosedXMLは直感的なAPIを提供しており、これにより開発者は効率的にExcelファイルを操作することができる。

このライブラリを使用すると、新しいワークブックの作成、既存のワークブックの開封、ワークシートの追加や削除、セルの書式設定、数式の追加、グラフや画像の挿入等が可能である。
また、大量のデータを扱う際のパフォーマンスも考慮されており、メモリ効率の良い操作が可能である。

ClosedXMLの大きなメリットの1つは、Microsoft Office製品がインストールされていないPCでも動作することである。
これは、サーバ環境でのExcelファイル操作を可能にして、多くの企業システムで重要な機能となっている。

セキュリティの観点からもClosedXMLは優れており、ファイルの操作がプログラム的に行われるため、マクロを含むExcelファイルによるセキュリティリスクを軽減できる。

ClosedXMLの使用例としては、財務レポートの自動生成、大量のデータの分析と視覚化、テンプレートベースの文書生成等が挙げられる。
これらのタスクは、ビジネスアプリケーションでよく必要とされるものである。

このライブラリは活発に開発が続けられており、コミュニティからの支援も豊富なため、バグ修正や新機能の追加が定期的に行われており、最新のExcel機能にも対応している。

ただし、ClosedXMLにも制限はある。
例えば、一部の高度なExcel機能はサポートされていない場合がある。
また、非常に大きなファイルを扱う場合には、メモリ使用量に注意が必要となる。

ClosedXMLはC#開発者にとって、Excelファイル操作を効率的に行うための強力なツールである。
その使いやすさ、豊富な機能、そして堅牢性により、多くのプロジェクトで重要な役割を果たしている。


Closed XMLとは

Open XML形式のExcelファイル(拡張子 : xlsx / xlsm)を読み書きするためのライブラリである。
MicrosoftのOpen XML SDKをラップし、Excelに特化することで使用しやすいライブラリになっている。

Closed XMLのソースコードは、GitHubで公開されている。
また、Closed XMLのライセンスは、MITライセンスである。

Closed XMLで可能なこと

  • Open XML形式のExcelファイルの作成と編集
  • セルデータの読み書き
    • 数値、文字列、日付等の各種データ型
    • 複雑な数式の挿入と評価
  • セルの書式設定
    • フォント
    • 罫線
    • 背景色
  • ワークシートの操作
    • 追加、削除、名前変更、コピー
    • 行や列の挿入、削除、非表示設定
  • Excelの主要関数のサポート
    • SUM, AVERAGE, VLOOKUP, IF等
  • 大量データの高速処理
  • テンプレートベースの文書生成
  • 基本的な条件付き書式の適用
  • ハイパーリンクの挿入と管理
  • 印刷設定の調整(ページ設定、印刷範囲など)


Closed XMLで実行できないこと

  • Excelアプリケーションの直接操作
  • マクロやVBAコードの実行・編集
  • 複雑なグラフや図表の新規作成
  • ピボットテーブルの新規作成や高度な編集
  • 一部の高度なExcel機能
    • 複雑な条件付き書式
    • スパークライン
    • 高度なデータ検証ルール
  • 外部データソースとの動的連携
  • リアルタイムデータ更新
  • Excelアドインの使用
  • 複雑な画像操作や図形の作成
  • コメントの詳細な管理や高度な編集


※注意
上記は、Closed XMLは継続的に更新されているため、一部の機能は将来のバージョンで追加される可能性がある。

サポートされているプラットフォーム

以下に示すプラットフォームサポートにより、Closed XMLは幅広い.NET環境で使用できる柔軟性の高いライブラリとなっている。
デスクトップアプリケーション、Webアプリケーション、モバイルアプリケーション、ゲーム開発まで、様々な種類のプロジェクトでExcelファイルの操作が可能である。

  • .NET Core 2.0以降 / .NET 5以降
  • .NET Framework 4.5以降(4.0ではなく4.5からのサポート)
  • .NET Standard 2.0以降


.NET Standard 2.0のサポートにより、以下のプラットフォームでも利用可能である。

  • ASP.NET Core 2.0以降
  • UWP(Universal Windows Platform)10.0.16299 (Fall Creators Update、1709) 以降
  • Xamarin.iOS 10.14以降
  • Xamarin.Android 8.0以降
  • Xamarin.Mac 3.8以降
  • Unity 2018.1以降 (ゲーム開発プラットフォームとしても利用可能)


※注意
プラットフォームのバージョンは、Closed XMLのバージョンによって変更される可能性がある。
最新の対応状況は、Closed XMLの公式ドキュメントやGithubリポジトリで確認することを推奨する。
また、一部の機能は、特定のプラットフォームでのみ利用可能な場合がある。

Closed XMLのドキュメントは、ClosedXMLのGithubで公開されている。

Closed XMLの詳細を知りたい場合は、公式ドキュメントを参照すること。


Closed XMLのインストール

RiderまたはVisual StudioからNuGetを使用して、Closed XMLをインストールする。

  • Riderの場合
    1. [ツール]メインメニュー - [Nuget] - [ソリューション の Nuget パッケージを管理] (または、[<プロジェクト名> の Nuget パッケージを管理])を選択する。
    2. メイン画面下部にある[パッケージ]タブからClosedXMLと入力して検索する。
    3. メイン画面下部の右にある[+]ボタンを押下して、Closed XMLをインストールする。


また、Closed XMLのGithubからソースコードをダウンロードして、ビルドおよびインストールすることもできる。

Closed XMLをインストールする時、Open XML SDK等のいくつかの必要なパッケージが同時にインストールされる。


Excelファイルの作成

以下の例では、Excelファイルおよびシートを新規作成している。

  • Open XML形式のExcelファイルの作成と編集
    new XLWorkbook()を使用して新しいワークブックを作成する。
    workbook.Worksheets.Add("サンプルシート")でワークシートを追加する。
    workbook.SaveAs("sample.xlsx")でファイルを保存する。
    既存のファイルを開く場合は、new XLWorkbook("サンプル.xlsx")を使用する。


 using System;
 using ClosedXML.Excel;
 
 const string ExcelFilePath = @"sample.xlsx";
 
 // Excelファイルの作成
 using (var workbook = new XLWorkbook())
 {
    // ワークシートの追加
    using (var worksheet = workbook.Worksheets.Add("サンプルシート"))
    {
       // 何らかの処理
 
       // ワークブックを保存する
       workbook.SaveAs(ExcelFilePath);
    }
 
    Console.WriteLine(@"Excelファイルを保存しました");
 
    // 保存したファイルを開いて編集
    using (var workbook = new XLWorkbook("サンプル.xlsx"))
    {
       var worksheet = workbook.Worksheet(1);
 
       // 何らかの処理
  
       // 変更を保存
       workbook.Save();
    }
 }



セルデータの読み書き

以下の例では、Open XML形式のExcelファイルにおいて、セルの値の設定、読み取り、数式の挿入、ファイルの保存と再編集等を行っている。

  • セルデータの読み書き
    • 書き込み
      worksheet.Cell("A1").Value = "文字列" のように、Cellメソッドを使用してセルを指定して、Valueプロパティに値を代入する。
    • 読み取り
      var cellValue = worksheet.Cell("A2").Value のように、Valueプロパティを使用してセルの値を取得する。


  • 数値、文字列、日付等の各種データ型
    • 文字列
      worksheet.Cell("A1").Value = "文字列"
    • 数値
      worksheet.Cell("A2").Value = 123
    • 日付
      worksheet.Cell("A3").Value = DateTime.Now


  • 複雑な数式の挿入と評価
    • 基本的な数式
      worksheet.Cell("B1").FormulaA1 = "=SUM(A1:A3)"
    • 条件付き数式
      worksheet.Cell("C1").FormulaA1 = "=IF(A2>100, \"高い\", \"低い\")"


また、大量のデータを扱う場合は、パフォーマンスを考慮したアプローチ (例: バッチ処理) を検討する必要がある。

 using System;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       // 新しいExcelワークブックを作成
       using (var workbook = new XLWorkbook())
       {
          var worksheet = workbook.Worksheets.Add("サンプルシート");
 
          // セルデータの書き込み
          worksheet.Cell("A1").Value = "文字列";
          worksheet.Cell("A2").Value = 123;           // 数値
          worksheet.Cell("A3").Value = DateTime.Now;  // 日付
 
          // 数式の挿入
          worksheet.Cell("B1").FormulaA1 = "=SUM(A1:A3)";
 
          // セルのスタイル設定
          var cell = worksheet.Cell("A1");
          cell.Style.Font.Bold = true;
          cell.Style.Fill.BackgroundColor = XLColor.LightBlue;
 
          // 複雑な数式の挿入
          worksheet.Cell("C1").FormulaA1 = "=IF(A2>100, \"高い\", \"低い\")";
 
          // 位置を指定してセルの値を読み取り
          var cellValue = worksheet.Cell("A2").Value;
          var cellA1 = worksheet.Cell("A1");
          var cellA2 = worksheet.Cell(2, "A");  // 行番号と列名でも指定可能
          var cellB1 = worksheet.Cell(3, 1);    // 行番号と列番号でも指定可能
 
          Console.WriteLine($"A2セルの値: {cellValue}");
          Console.WriteLine($"A1セルの値: {cellA1.GetValue<strring>()}");        // 出力 : A1 = "文字列"
          Console.WriteLine($"A2セルの値: {cellA2.GetValue<int>()}");            // 出力 : A2 = 123
          Console.WriteLine($"A3(値)セルの値: {cellA3.GetValue<DateTime>()}");   // 出力 : A3(値) = 30
          Console.WriteLine($"B1(数式)セルの値: {cellB1.FormulaA1}");            // 出力 : B1(数式) = SUM(A1:A3)
 
          // ファイルを保存
          workbook.SaveAs("サンプル.xlsx");
       }
 
       // 保存したファイルを開いて編集
       using (var workbook = new XLWorkbook("サンプル.xlsx"))
       {
          var worksheet = workbook.Worksheet(1);
 
          // 既存のデータを読み取り
          var a1Value = worksheet.Cell("A1").Value;
          Console.WriteLine($"A1セルの値: {a1Value}");
 
          // 新しいデータを追加
          worksheet.Cell("D1").Value = "新しいデータ";
 
          // 変更を保存
          workbook.Save();
       }
 
       Console.WriteLine("処理が完了しました。");
    }
 }



セルの書式設定

以下の例では、セルにフォント、色、罫線、背景色等の様々な書式設定を適用している。
これらの設定を組み合わせたり、より複雑な条件付き書式を適用することができる。

 using System;
 using System.Drawing;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       using (var workbook = new XLWorkbook())
       {
          var worksheet = workbook.Worksheets.Add("書式設定サンプル");
 
          // フォントの設定
          var cell1 = worksheet.Cell("A1");
          cell1.Value                = "フォント設定";
          cell1.Style.Font.FontName  = "Arial";
          cell1.Style.Font.FontSize  = 14;
          cell1.Style.Font.Bold      = true;
          cell1.Style.Font.Italic    = true;
          cell1.Style.Font.Underline = XLFontUnderlineValues.Single;
          cell1.Style.Font.FontColor = XLColor.Red;
 
          // 背景色の設定
          var cell2                        = worksheet.Cell("A2");
          cell2.Value                      = "背景色設定";
          cell2.Style.Fill.BackgroundColor = XLColor.LightBlue;
 
          // 罫線の設定
          var range = worksheet.Range("A3:C5");
          range.Style.Border.OutsideBorder      = XLBorderStyleValues.Thick;
          range.Style.Border.OutsideBorderColor = XLColor.Black;
          range.Style.Border.InsideBorder       = XLBorderStyleValues.Thin;
          range.Style.Border.InsideBorderColor  = XLColor.Gray;
 
          // セルの配置
          var cell3 = worksheet.Cell("A3");
          cell3.Value                      = "セル配置";
          cell3.Style.Alignment.Horizontal = XLAlignmentHorizontalValues.Center;
          cell3.Style.Alignment.Vertical   = XLAlignmentVerticalValues.Center;
 
          // 数値書式の設定
          var cell4 = worksheet.Cell("A6");
          cell4.Value                     = 12345.6789;
          cell4.Style.NumberFormat.Format = "#,##0.00";
 
          // 日付書式の設定
          var cell5 = worksheet.Cell("A7");
          cell5.Value                   = DateTime.Now;
          cell5.Style.DateFormat.Format = "yyyy年MM月dd日";
 
          // セルの幅と高さの設定
          worksheet.Column(1).Width = 20;  // セルの幅
          worksheet.Row(1).Height   = 30;  // セルの高さ
 
          // 条件付き書式の設定 (メソッドチェーンを使用しない場合)
          var range             = worksheet.Range("A8:A10");
          var conditionalFormat = range.AddConditionalFormat();
          conditionalFormat.WhenGreaterThan(50);
 
          var fillFormat = conditionalFormat.Fill;
          fillFormat.SetBackgroundColor(XLColor.Green);
 
          var fontFormat = conditionalFormat.Font;
          fontFormat.SetFontColor(XLColor.White);
 
          worksheet.Cell("A8").Value  = 30;
          worksheet.Cell("A9").Value  = 60;
          worksheet.Cell("A10").Value = 90;
 
          // 条件付き書式の設定 (メソッドチェーンを使用する場合)
          var conditionalFormat = worksheet.Range("A8:A10").AddConditionalFormat();
          conditionalFormat.WhenGreaterThan(50)
                           .Fill.SetBackgroundColor(XLColor.Green)
                           .Font.SetFontColor(XLColor.White);
 
          worksheet.Cell("A8").Value  = 30;
          worksheet.Cell("A9").Value  = 60;
          worksheet.Cell("A10").Value = 90;
 
          // ファイルを保存
          workbook.SaveAs("書式設定サンプル.xlsx");
       }
 
       Console.WriteLine("書式設定サンプルファイルを作成");
    }
 }



ワークシートの操作

以下の例では、ワークシートの追加、削除、名前変更、コピー、行や列の挿入、削除、非表示設定をしている。

また、エラーハンドリングのためのtry-catchブロックも含んでいる。
これにより、操作中に発生する可能性のある例外を捕捉して、適切なエラーメッセージを表示する。

 using System;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       using (var workbook = new XLWorkbook())
       {
          try
          {
             // ワークシートの追加
             var sheet1 = workbook.Worksheets.Add("Sheet1");
             var sheet2 = workbook.Worksheets.Add("Sheet2");
             Console.WriteLine("2つのワークシートを追加");
 
             // ワークシートの名前変更
             sheet1.Name = "新しい名前";
             Console.WriteLine("Sheet1の名前を 新しい名前 に変更");
 
             // ワークシートのコピー (既存のワークシートをコピー)
             var copiedSheet = sheet2.CopyTo("Sheet2のコピー");
             Console.WriteLine("Sheet2をコピー");
 
             // 行の挿入 (指定した行の上に新しい行を挿入)
             sheet1.Row(2).InsertRowsAbove(2);
             Console.WriteLine("2行目の上に2行挿入");
 
             // 列の挿入 (指定した列の前に新しい列を挿入)
             sheet1.Column(2).InsertColumnsBefore(2);
             Console.WriteLine("B列の前に2列挿入");
 
             // 行の削除
             sheet1.Row(5).Delete();
             Console.WriteLine("5行目を削除");
 
             // 列の削除
             sheet1.Column("D").Delete();
             Console.WriteLine("D列を削除");
 
             // 行の非表示設定
             sheet1.Row(3).Hide();
             Console.WriteLine("3行目を非表示");
 
             // 列の非表示設定
             sheet1.Column("C").Hide();
             Console.WriteLine("C列を非表示");
 
             // ワークシートの削除 (ワークシート全体を削除)
             workbook.Worksheet("Sheet2のコピー").Delete();
             Console.WriteLine("Sheet2のコピーを削除");
 
             // ファイルを保存
             workbook.SaveAs("ワークシート操作サンプル.xlsx");
             Console.WriteLine("ファイルを保存");
          }
          catch (Exception ex)
          {
             Console.WriteLine($"エラーが発生: {ex.Message}");
          }
       }
    }
 }



Excelの主要な関数

使用例

以下の例では、SUM、AVERAGE、VLOOKUP、IF等の関数を直接挿入して使用している。

関数の適用にはFormulaA1プロパティを使用して、通常のExcelで使用する形式で数式を記述する。
関数の結果を取得するには、セルのValueプロパティを使用する。

また、文字列を含む数式では、ダブルクォートをエスケープする必要がある。(例: \"高い\")

※注意
実際の使用では、これらの関数をより複雑な数式の一部として使用したり、大量のデータに対して適用したりすることができる。
また、Closed XMLは多くのExcel関数をサポートしているが、一部の高度な関数や最新の関数はサポートされていない可能性がある。
そのような場合は、代替の方法を検討する必要がある。

 using System;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       using (var workbook = new XLWorkbook())
       {
          var worksheet = workbook.Worksheets.Add("関数サンプル");
 
          try
          {
             // サンプルデータの入力
             worksheet.Cell("A1").Value = "商品";
             worksheet.Cell("B1").Value = "価格";
             worksheet.Cell("A2").Value = "りんご";
             worksheet.Cell("B2").Value = 100;
             worksheet.Cell("A3").Value = "バナナ";
             worksheet.Cell("B3").Value = 80;
             worksheet.Cell("A4").Value = "オレンジ";
             worksheet.Cell("B4").Value = 120;
 
             // SUM関数
             // A1形式の数式を設定して、B2からB4セルの値を合計する
             worksheet.Cell("B5").FormulaA1 = "=SUM(B2:B4)";
             Console.WriteLine($"合計金額: {worksheet.Cell("B5").Value}");
 
             // AVERAGE関数
             // B2からB4セルの値の平均を計算する
             worksheet.Cell("B6").FormulaA1 = "=AVERAGE(B2:B4)";
             Console.WriteLine($"平均価格: {worksheet.Cell("B6").Value}");
 
             // 検索用データの入力
             worksheet.Cell("D1").Value = "商品コード";
             worksheet.Cell("E1").Value = "商品名";
             worksheet.Cell("D2").Value = "A001";
             worksheet.Cell("E2").Value = "りんご";
             worksheet.Cell("D3").Value = "B001";
             worksheet.Cell("E3").Value = "バナナ";
             worksheet.Cell("D4").Value = "C001";
             worksheet.Cell("E4").Value = "オレンジ";
 
             // VLOOKUP関数
             // A7セルの値をキーとして、D2:E4の範囲から対応する値を検索する
             // 2は返す列のインデックス、FALSEは完全一致を意味する
             worksheet.Cell("A7").Value     = "B001";  // 検索キー
             worksheet.Cell("B7").FormulaA1 = "=VLOOKUP(A7,D2:E4,2,FALSE)";
             Console.WriteLine($"商品コードB001の商品名: {worksheet.Cell("B7").Value}");
 
             // IF関数
             // B2セルの値が100より大きい場合は"高い"、そうでない場合は"安い"を返す
             worksheet.Cell("C2").FormulaA1 = "=IF(B2>100,\"高い\",\"安い\")";
             worksheet.Cell("C3").FormulaA1 = "=IF(B3>100,\"高い\",\"安い\")";
             worksheet.Cell("C4").FormulaA1 = "=IF(B4>100,\"高い\",\"安い\")";
             Console.WriteLine($"りんごの評価: {worksheet.Cell("C2").Value}");
             Console.WriteLine($"バナナの評価: {worksheet.Cell("C3").Value}");
             Console.WriteLine($"オレンジの評価: {worksheet.Cell("C4").Value}");
 
             // ファイルを保存
             workbook.SaveAs("Excel関数サンプル.xlsx");
             Console.WriteLine("ファイルを保存");
          }
          catch (Exception ex)
          {
             Console.WriteLine($"エラーが発生: {ex.Message}");
          }
       }
    }
 }


関数がサポートされていない場合の代替方法

Closed XMLにおいてサポートされていない高度な関数や最新の関数を扱うための代替方法を、以下に示す。

  • C#でロジックを実装する
    Excelの関数と同等の機能をC#のコードで直接実装する。
    これにより、任意の複雑さの計算を行うことができる。
    例: SUMIFS関数の代わりに、LINQを使用して条件に合う値を合計する。

  • 数式を分解する
    複雑な関数を、サポートされているより単純な関数の組み合わせに分解する。
    これにより、同等の機能を実現できる場合がある。
    例: SUMPRODUCT関数を、複数のセルの乗算とSUM関数の組み合わせで実現する。

  • 名前付き範囲を利用する
    複雑な計算を名前付き範囲として定義して、その結果を参照することにより、計算を管理しやすくする。
    これにより、複雑な計算をより管理しやすい形に分割できる。

  • データ処理をExcelの外で行う
    複雑な計算や処理をC#で行い、結果のみをExcelに書き込む。
    これは、特に大量のデータを扱う場合に効果的である。

  • 動的配列数式の代替
    最新のExcelの動的配列数式がサポートされていない場合、結果を複数のセルに手動で展開する。

  • Power Query相当の処理をC#で実装する
    データの変換や結合等の操作をC#で実装して、結果をExcelに書き込む。

  • VBAマクロの代替
    VBAマクロで行っている処理をC#で実装して、Closed XMLを使用してExcelファイルを操作する。

  • 外部ライブラリの使用
    数学や統計の高度な計算を行う外部ライブラリを使用して、Excelの高度な関数を再現する。


代替方法の例 : C#でロジックを実装する

以下の例では、C#とLINQを使用して、ExcelのSUMIFS関数と同等の機能 (LINQを使用して、条件に合う値を合計) を実現している。

 // 実装例 : SUMIFS関数の代替
 // C#とLINQを使用
 
 public static double SumIfs(IXLRange dataRange, params Tuple<IXLRange, string>[] criteria)
 {
    var data = dataRange.CellsUsed().Select(c => c.GetDouble());
    foreach (var criterion in criteria)
    {
       var criteriaRange = criterion.Item1;
       var condition     = criterion.Item2;
       data = data.Where((value, index) => EvaluateCondition(criteriaRange.Cell(index + 1).GetString(), condition));
    }
 
    return data.Sum();
 }
 
 private static bool EvaluateCondition(string cellValue, string condition)
 {
    // 条件の評価ロジックを実装
    // 例: ">10", "=Test", "<5"などの条件をパースして評価
 }



条件付き書式の適用

条件付き書式は、データの傾向や例外を視覚的に把握するのに役立つ。

Closed XMLでは、AddConditionalFormatメソッドを使用して、各条件付き書式のルールを設定することができる。

以下の例では、条件付き書式をExcelシートに適用している。

  • サンプルデータを生成して、シートに挿入する。
  • 3つの異なる条件付き書式を適用する。
    • 値が50より大きい場合、セルを赤色に塗る。
    • 値が25以下の場合、セルを緑色に塗る。
    • データバーを適用して、値の大きさを視覚的に表現する。


 using System;
 using System.Linq;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       Console.WriteLine("条件付き書式の適用を開始...");
 
       try
       {
          using (var workbook = new XLWorkbook())
          {
             var worksheet = workbook.Worksheets.Add("条件付き書式");
 
             // サンプルデータを挿入
             var data = Enumerable.Range(1, 20).Select(i => new { Value = i * 5 }).ToList();
             worksheet.Cell(1, 1).Value = "Value";
             worksheet.Cell(2, 1).InsertData(data);
 
             // 条件付き書式を適用
             var range = worksheet.Range(2, 1, 21, 1);
 
             // 1. 値が50より大きい場合、セルを赤色に
             var rule1 = range.AddConditionalFormat();
             rule1.WhenGreaterThan(50).Fill.SetBackgroundColor(XLColor.Red);
 
             // 2. 値が25以下の場合、セルを緑色に
             var rule2 = range.AddConditionalFormat();
             rule2.WhenLessThanOrEqual(25).Fill.SetBackgroundColor(XLColor.Green);
 
             // 3. データバーを適用
             var rule3 = range.AddConditionalFormat();
             rule3.DataBar(XLColor.Blue);
 
             workbook.SaveAs("条件付き書式.xlsx");
          }
 
          Console.WriteLine("条件付き書式の適用が完了");
       }
       catch (Exception ex)
       {
          Console.WriteLine($"エラーが発生: {ex.Message}");
       }
    }
 }



テンプレートベースの文書生成

テンプレートベースの文書の生成は、レポートや請求書等、構造化された文書の生成に適している。

※注意
ただし、テンプレートファイルの構造が変更された場合、コードの修正が必要になる可能性があることに注意すること。

以下の例では、既存のテンプレートファイルを基に新しい文書を生成している。

  • 既存のテンプレートファイルを開いて、新しいワークブックにコピーする。
  • ReplaceTextメソッドを使用して、プレースホルダーを実際の値で置換する。
  • 動的なデータを挿入して、必要な計算を行う。


 using System;
 using System.Collections.Generic;
 using ClosedXML.Excel;
 
 class Program
 {
    static void Main(string[] args)
    {
       Console.WriteLine("テンプレートベースの文書生成を開始...");
 
       try
       {
          // テンプレートファイルを開く
          using (var template = new XLWorkbook("テンプレート.xlsx"))
          using (var result = new XLWorkbook())
          {
             var worksheetTemplate = template.Worksheet(1);
             var worksheetResult = result.Worksheets.Add("生成結果");
 
             // テンプレートをコピー
             worksheetTemplate.CopyTo(worksheetResult);
 
             // プレースホルダーを置換
             worksheetResult.Cells().ReplaceText("{{日付}}", DateTime.Now.ToString("yyyy/MM/dd"));
             worksheetResult.Cells().ReplaceText("{{会社名}}", "株式会社サンプル");
 
             // 動的データを挿入
             var data = new List<object[]>
             {
                new object[] { "商品A", 100, 5000 },
                new object[] { "商品B", 200, 3000 },
                new object[] { "商品C", 150, 4000 }
             };
 
             var dataRange = worksheetResult.Cell("A5").InsertData(data);
 
             // 合計を計算
             var lastRow = dataRange.LastRow().RowNumber();
             worksheetResult.Cell(lastRow + 1, 1).Value = "合計";
             worksheetResult.Cell(lastRow + 1, 3).FormulaA1 = $"=SUM(C5:C{lastRow})";
 
             result.SaveAs("生成結果.xlsx");
          }
 
          Console.WriteLine("テンプレートベースの文書生成が完了");
       }
       catch (Exception ex)
       {
          Console.WriteLine($"エラーが発生: {ex.Message}");
       }
    }
 }



コレクションやDataTableの使用

Closed XMLは、コレクションやLINQなどと相性が良い。
コレクションやDataTableオブジェクト(System.Data名前空間)を簡単にワークシートに挿入できる。

以下のサンプルコードは、コレクションを挿入している例である。
データを挿入するInsertDataメソッドは、挿入したセルの範囲を返すので、続けてその範囲のセルを処理しやすい。
また、その範囲から特定の行を抽出するには、ラムダ式で条件を指定できる。

以下のサンプルコードではコレクションを挿入したが、DataTableにデータを挿入するときも同様にInsertDataメソッドを使用する。

 const string ExcelFilePath = @"sample.xlsx";
 
 // Excelファイルに貼り付けたいデータ
 // 例として、ここでは今月の日付と曜日のコレクションを用意する
 List<(DateTime, string)> days = new List<(DateTime, string)>();
 var firstDayOfThisMonth = new DateTime(DateTime.Today.Year, DateTime.Today.Month, 1);
 int daysInThisMonth = DateTime.DaysInMonth(firstDayOfThisMonth.Year, firstDayOfThisMonth.Month);
 
 for (int i = 0; i < daysInThisMonth; i++)
 {
    var d = firstDayOfThisMonth.AddDays(i);
    days.Add((d, d.DayOfWeek.ToString()));
 }
 
 // Excelファイルを開く
 using (var workbook = new XLWorkbook(ExcelFilePath))
 {
    // ワークシートの追加
    using (var worksheet = workbook.Worksheets.Add("サンプルシート2"))
    {
       // タイトル行
       var cellA1 = worksheet.Cell("A1");
       cellA1.Value = "今月のカレンダー";
       cellA1.Style.Alignment.SetHorizontal(XLAlignmentHorizontalValues.Center)
                             .Font.SetFontName("Meiryo")
                             .Font.SetBold(true)
                             .Font.SetFontColor(XLColor.White)
                             .Fill.SetBackgroundColor(XLColor.FromArgb(0x00A2E8));
 
       // セルの連結
       worksheet.Range("A1:B1").Merge();
 
       // ワークシートに日付と曜日のコレクションを挿入
       var calendarRange = worksheet.Cell("A2").InsertData(days);
 
       // 日曜日の行を抽出して使用しているセルの背景を赤に変更
       var sundayRows = calendarRange.Rows(r => r.Cell(1).GetDateTime().DayOfWeek == 0);
       foreach (var r in sundayRows)
       {
          r.CellsUsed().Style.Fill.BackgroundColor = XLColor.Red;
       }
 
       // ワークブックの保存
       workbook.Save();
    }
 
    Console.WriteLine("ワークシートを追加しました。");
 }


上記で作成したExcelファイルを読み取る。
何らかのデータや数式が入力されているセルの範囲を取得するRangeUsedメソッドがあるので、セルの位置を指定せずに簡単にデータ範囲を取り出せる。
そのセル範囲からコレクションを生成するには、LINQのSelect拡張メソッドを使用する。
また、DataTableに変換するのも簡単である。

 const string ExcelFilePath = ".\\sample.xlsx";
 
 // Excelファイルを開く
 using (var workbook = new XLWorkbook(ExcelFilePath))
 {
    // ワークシートの取得
    using (var worksheet = workbook.Worksheet("サンプルシート2"))
    {
       // データや数式が入力されているセルの範囲
       var rangeUsed = worksheet.RangeUsed();
       Console.WriteLine($"使用中のセル範囲:{rangeUsed.FirstCell().Address}~{rangeUsed.LastCell().Address}");
 
       // 1列目がDateTime型の行のみを取得してコレクションに変換
       IEnumerable<(DateTime,string)> data = rangeUsed.Rows(r => r.Cell(1).DataType == XLDataType.DateTime)
                                             .Select(r => (r.Cell(1).GetDateTime(), r.Cell(2).GetString()));
 
       // 最初の7個を表示
       // 出力 : 10/1(Monday) 10/2(Tuesday) 10/3(Wednesday) 10/4(Thursday) 10/5(Friday) 10/6(Saturday) 10/7(Sunday)
       foreach (var (d, s) in data.Take(7))
       {
          Console.WriteLine($"{d:M/d}({s})");
       }
 
       // DataTableへの変換もサポート
       System.Data.DataTable dataTable = rangeUsed.CreateTable().AsNativeDataTable();
 
       // 8個目から7個を表示
       // 出力 : 10/8(Monday) 10/9(Tuesday) 10/10(Wednesday) 10/11(Thursday) 10/12(Friday) 10/13(Saturday) 10/14(Sunday)
       foreach(System.Data.DataRow r in dataTable.Select().Skip(7).Take(7))
       {
          Console.WriteLine($"{r[0]:M/d}({r[1]})");
       }
    }
 }