NPOI 在指定單元格導入導出圖片

NPOI 在指定單元格導入導出圖片

Intro

我維護了一個 NPOI 的擴展(WeihanLi.Npoi),主要用來導入導出 Excel 數據,最近有網友提出了導入 Excel 的時候解析圖片的需求,因而就有了本文的探索git

導入Excel 時解析圖片

xlsxlsx 的 API 稍有不一樣,詳細能夠直接參考如下代碼,實現代碼以下:github

public static Dictionary<CellPosition, IPictureData> GetPicturesAndPosition(this ISheet sheet)
{
    var dictionary = new Dictionary<CellPosition, IPictureData>();
    if (sheet.Workbook is HSSFWorkbook)
    {
        foreach (var shape in ((HSSFPatriarch)sheet.DrawingPatriarch).Children)
        {
            if (shape is HSSFPicture picture)
            {
                var position = new CellPosition(picture.ClientAnchor.Row1, picture.ClientAnchor.Col1);
                dictionary[position] = picture.PictureData;
            }
        }
    }
    else if (sheet.Workbook is XSSFWorkbook)
    {
        foreach (var shape in ((XSSFDrawing)sheet.DrawingPatriarch).GetShapes())
        {
            if (shape is XSSFPicture picture)
            {
                var position = new CellPosition(picture.ClientAnchor.Row1, picture.ClientAnchor.Col1);
                dictionary[position] = picture.PictureData;
            }
        }
    }
    return dictionary;
}

CellPosition 是一個自定義的結構體,表示當前單元格的位置,源碼以下:c#

public readonly struct CellPosition : IEquatable<CellPosition>
{
    public CellPosition(int row, int col)
    {
        Row = row;
        Column = col;
    }

    public int Row { get; }
    public int Column { get; }

    public bool Equals(CellPosition other)
    {
        return Row == other.Row && Column == other.Column;
    }

    public override bool Equals(object? obj) => obj is CellPosition other && Equals(other);

    public override int GetHashCode() => $"{Row}_{Column}".GetHashCode();
}

根據上面的代碼,咱們就能夠獲取到獲取到全部的圖片以及圖片的所在位置,這樣根據單元格位置去找圖片信息的時候就會很方便了數組

導出 Excel 時設置圖片

實現代碼以下:async

public static bool TryAddPicture(this ISheet sheet, int row, int col, byte[] pictureBytes, PictureType pictureType = PictureType.PNG)
{
    if (sheet is null)
    {
        throw new ArgumentNullException(nameof(sheet));
    }

    try
    {
        var pictureIndex = sheet.Workbook.AddPicture(pictureBytes, pictureType);

        var clientAnchor = sheet.Workbook.GetCreationHelper().CreateClientAnchor();
        clientAnchor.Row1 = row;
        clientAnchor.Col1 = col;

        var picture = (sheet.DrawingPatriarch ?? sheet.CreateDrawingPatriarch())
            .CreatePicture(clientAnchor, pictureIndex);
        picture.Resize();
        return true;
    }
    catch (Exception e)
    {
        Debug.WriteLine(e);
    }

    return false;
}

經過上面的代碼咱們就能夠在指定的單元格設置圖片,目前沒有支持單元格合併操做,有須要本身進行修改ide

WeihanLi.Npoi

WeihanLi.Npoi1.15.0 版本中增長了圖片導入導出的支持,使用示例能夠參考下面的單元測試:單元測試

[Theory]
[ExcelFormatData]
public async Task ImageImportExportTest(ExcelFormat excelFormat)
{
    using var httpClient = new HttpClient();
    var imageBytes = await httpClient.GetByteArrayAsync("https://weihanli.xyz/assets/avator.jpg");
    var list = Enumerable.Range(1, 5)
        .Select(x => new ImageTest() { Id = x, Image = imageBytes })
        .ToList();
    var excelBytes = list.ToExcelBytes(excelFormat);
    var importResult = ExcelHelper.ToEntityList<ImageTest>(excelBytes, excelFormat);
    Assert.NotNull(importResult);
    Assert.Equal(list.Count, importResult.Count);
    for (var i = 0; i < list.Count; i++)
    {
        Assert.NotNull(importResult[i]);
        var result = importResult[i]!;
        Assert.Equal(list[i].Id, result.Id);
        Assert.NotNull(result.Image);
        Assert.True(list[i].Image.SequenceEqual(result.Image));
    }
}

private class ImageTest
{
    public int Id { get; set; }

    public byte[] Image { get; set; } = null!;
}

導入時會自動將 byte[] 類型的屬性嘗試獲取對應的單元格位置的圖片,若是在對應的位置找到了圖片就可以讀取到圖片的字節數組信息映射到 model 裏的字節數組屬性測試

除此以外,還支持直接導入的時候將圖片信息 Map 到 IPictureData 屬性上,可是以爲仍是字節數組更通用一些,若是對此感興趣能夠參考項目源碼以及單元測試this

More

感謝 @ZeguangZhang94 童鞋提出的需求和幫忙測試~excel

若是你也有相似的讀取指定單元格的圖片或者在指定單元格插入圖片的需求,能夠試一下上面的方法,但願對你有幫助,能夠直接引用個人類庫或者直接拷貝源碼到本身的項目裏使用

References

相關文章
相關標籤/搜索