6 Commits

Author SHA1 Message Date
melekhin
5302edfb8f borders 2026-06-19 16:35:17 +07:00
melekhin
e373d4108a many debugs
All checks were successful
Publish NuGet packages / publish (push) Successful in 28s
2026-06-19 15:06:40 +07:00
melekhin
08b39b7bfe Sheet.TryMergeBy...()
All checks were successful
Publish NuGet packages / publish (push) Successful in 23s
2026-06-17 10:39:00 +07:00
melekhin
c9ef2a796e Remove ICellText
All checks were successful
Publish NuGet packages / publish (push) Successful in 27s
2026-06-17 09:33:21 +07:00
melekhin
eccb12b83c Text() debug
All checks were successful
Publish NuGet packages / publish (push) Successful in 25s
2026-06-16 15:52:08 +07:00
melekhin
804fc84563 ICellText rename
All checks were successful
Publish NuGet packages / publish (push) Successful in 26s
2026-06-16 12:03:56 +07:00
25 changed files with 2135 additions and 1132 deletions

View File

@@ -2,8 +2,8 @@
internal static class CellAddressHelper
{
private const string letters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
private static readonly uint[] _powers = [1, 26, 676];
const string letters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
static readonly uint[] _powers = [1, 26, 676];
public static uint ColumnLetterToIndex(string col)
{

View File

@@ -18,6 +18,30 @@ public readonly struct CellAlign : IEquatable<CellAlign>
/// <summary>Уменьшать размер шрифта, чтобы текст поместился в ячейку.</summary>
public bool? ShrinkToFit { get; init; }
internal bool TryMerge(CellAlign other, out CellAlign result)
{
// Если other не содержит новых значений, возвращаем this
if (other.Horizontal == Horizontal &&
other.Vertical == Vertical &&
other.WrapText == WrapText &&
other.ShrinkToFit == ShrinkToFit)
{
result = default;
return false;
}
result = new CellAlign
{
Horizontal = other.Horizontal ?? Horizontal,
Vertical = other.Vertical ?? Vertical,
WrapText = other.WrapText ?? WrapText,
ShrinkToFit = other.ShrinkToFit ?? ShrinkToFit
};
return true;
}
/// <summary>Преобразует горизонтальное выравнивание в тип Open XML.</summary>
public bool TryGetExcelHorizontalAlignment(out HorizontalAlignmentValues value)
{
@@ -104,7 +128,7 @@ public readonly struct CellAlign : IEquatable<CellAlign>
return result;
}
private static CellAlignHorizontal MapHorizontalFromExcel(HorizontalAlignmentValues value)
static CellAlignHorizontal MapHorizontalFromExcel(HorizontalAlignmentValues value)
{
if (value == HorizontalAlignmentValues.Left)
{
@@ -137,7 +161,7 @@ public readonly struct CellAlign : IEquatable<CellAlign>
else throw new NotSupportedException($"Unsupported horizontal alignment: {value}");
}
private static CellAlignVertical MapVerticalFromExcel(VerticalAlignmentValues value)
static CellAlignVertical MapVerticalFromExcel(VerticalAlignmentValues value)
{
if (value == VerticalAlignmentValues.Top)
{

View File

@@ -1,11 +1,53 @@
namespace QWERTYkez.ExcelProcessor;
/// <summary>
/// Определяет границы ячейки: верхнюю, нижнюю, левую, правую и диагональные.
/// Каждая граница может иметь стиль и цвет.
/// </summary>
public readonly struct CellBorder : IEquatable<CellBorder>
{
public static CellBorder BottomThin { get; } = new() { BottomBorder = BorderSide.BlackThin };
public static CellBorder TopThin { get; } = new() { TopBorder = BorderSide.BlackThin };
public static CellBorder LeftThin { get; } = new() { LeftBorder = BorderSide.BlackThin };
public static CellBorder RightThin { get; } = new() { RightBorder = BorderSide.BlackThin };
public static CellBorder AllThin { get; } = new()
{
BottomBorder = BorderSide.BlackThin,
TopBorder = BorderSide.BlackThin,
LeftBorder = BorderSide.BlackThin,
RightBorder = BorderSide.BlackThin
};
public static CellBorder BottomMedium { get; } = new() { BottomBorder = BorderSide.BlackMedium };
public static CellBorder TopMedium { get; } = new() { TopBorder = BorderSide.BlackMedium };
public static CellBorder LeftMedium { get; } = new() { LeftBorder = BorderSide.BlackMedium };
public static CellBorder RightMedium { get; } = new() { RightBorder = BorderSide.BlackMedium };
public static CellBorder AllMedium { get; } = new()
{
BottomBorder = BorderSide.BlackMedium,
TopBorder = BorderSide.BlackMedium,
LeftBorder = BorderSide.BlackMedium,
RightBorder = BorderSide.BlackMedium
};
public static CellBorder BottomThick { get; } = new() { BottomBorder = BorderSide.BlackThick };
public static CellBorder TopThick { get; } = new() { TopBorder = BorderSide.BlackThick };
public static CellBorder LeftThick { get; } = new() { LeftBorder = BorderSide.BlackThick };
public static CellBorder RightThick { get; } = new() { RightBorder = BorderSide.BlackThick };
public static CellBorder AllThick { get; } = new()
{
BottomBorder = BorderSide.BlackThick,
TopBorder = BorderSide.BlackThick,
LeftBorder = BorderSide.BlackThick,
RightBorder = BorderSide.BlackThick
};
/// <summary>Верхняя граница.</summary>
public BorderSide? TopBorder { get; init; }
@@ -24,6 +66,33 @@ public readonly struct CellBorder : IEquatable<CellBorder>
/// <summary>Диагональная граница «из левого нижнего в правый верхний» (//).</summary>
public BorderSide? DiagonalRight { get; init; }
public bool TryMerge(CellBorder other, out CellBorder result)
{
if (other.TopBorder == TopBorder &&
other.BottomBorder == BottomBorder &&
other.LeftBorder == LeftBorder &&
other.RightBorder == RightBorder &&
other.DiagonalLeft == DiagonalLeft &&
other.DiagonalRight == DiagonalRight)
{
result = default;
return false;
}
result = new CellBorder
{
TopBorder = other.TopBorder ?? TopBorder,
BottomBorder = other.BottomBorder ?? BottomBorder,
LeftBorder = other.LeftBorder ?? LeftBorder,
RightBorder = other.RightBorder ?? RightBorder,
DiagonalLeft = other.DiagonalLeft ?? DiagonalLeft,
DiagonalRight = other.DiagonalRight ?? DiagonalRight
};
return true;
}
/// <summary>Создаёт элемент Border для Open XML.</summary>
public Border? ToBorder()
{
@@ -112,11 +181,23 @@ public readonly struct CellBorder : IEquatable<CellBorder>
/// <summary>Стиль и цвет границы.</summary>
public readonly struct BorderSide : IEquatable<BorderSide>
{
/// <summary>Тонкая черная линия</summary>
public static BorderSide BlackThin { get; } = new() { Color = System.Drawing.Color.Black, Style = BorderStyle.Thin };
/// <summary>Толстая черная линия</summary>
public static BorderSide BlackThick { get; } = new() { Color = System.Drawing.Color.Black, Style = BorderStyle.Thick };
/// <summary>Средняя черная линия</summary>
public static BorderSide BlackMedium { get; } = new() { Color = System.Drawing.Color.Black, Style = BorderStyle.Medium };
/// <summary>Стиль линии границы.</summary>
public BorderStyle? Style { get; init; }
/// <summary>Цвет границы.</summary>
public ExColor? Color { get; init; }
public System.Drawing.Color? Color { get; init; }
internal T ToBorderElement<T>() where T : BorderPropertiesType, new()
{
@@ -141,9 +222,9 @@ public readonly struct BorderSide : IEquatable<BorderSide>
_ => throw new NotImplementedException(),
};
}
if (Color.HasValue && Color.Value.Color.HasValue)
if (Color.HasValue)
{
var c = Color.Value.Color.Value;
var c = Color.Value;
element.Color = new Color { Rgb = $"{c.R:X2}{c.G:X2}{c.B:X2}" };
}
return element;
@@ -162,17 +243,18 @@ public readonly struct BorderSide : IEquatable<BorderSide>
}
if (borderElement.Color?.Rgb?.Value is { } rgb && rgb.Length >= 6)
{
var color = System.Drawing.Color.FromArgb(
Convert.ToByte(rgb.Substring(0, 2), 16),
Convert.ToByte(rgb.Substring(2, 2), 16),
Convert.ToByte(rgb.Substring(4, 2), 16)
);
result = result with { Color = new ExColor(color) };
result = result with
{
Color = System.Drawing.Color.FromArgb(
Convert.ToByte(rgb.Substring(0, 2), 16),
Convert.ToByte(rgb.Substring(2, 2), 16),
Convert.ToByte(rgb.Substring(4, 2), 16))
};
}
return result;
}
private static BorderStyle MapBorderStyleFromExcel(BorderStyleValues value)
static BorderStyle MapBorderStyleFromExcel(BorderStyleValues value)
{
if (value == BorderStyleValues.Thin)
{
@@ -278,4 +360,25 @@ public enum BorderStyle
MediumDashDotDot,
/// <summary> Наклонная штрих-пунктирная (для диагональных) </summary>
SlantDashDot,
}
/// <summary>
/// Определяет, какие границы диапазона следует применить.
/// </summary>
public enum BorderTarget
{
/// <summary>Все границы (внешние и внутренние) полная сетка.</summary>
All,
/// <summary>Только внешние границы диапазона.</summary>
Outside,
/// <summary>Только внутренние границы (между ячейками).</summary>
Inside,
/// <summary>Только верхняя граница диапазона.</summary>
Top,
/// <summary>Только нижняя граница диапазона.</summary>
Bottom,
/// <summary>Только левая граница диапазона.</summary>
Left,
/// <summary>Только правая граница диапазона.</summary>
Right
}

View File

@@ -6,15 +6,13 @@
public readonly struct CellFill : IEquatable<CellFill>
{
/// <summary>Цвет фона.</summary>
public ExColor? BackgroundColor { get; init; }
public System.Drawing.Color? BackgroundColor { get; init; }
/// <summary>Создаёт элемент Fill для Open XML.</summary>
public Fill? ToFill()
{
if (!BackgroundColor.HasValue || !BackgroundColor.Value.Color.HasValue)
return null;
if (BackgroundColor is not { } c) return null;
var c = BackgroundColor.Value.Color.Value;
var fill = new Fill
{
PatternFill = new PatternFill
@@ -37,7 +35,7 @@ public readonly struct CellFill : IEquatable<CellFill>
Convert.ToByte(rgb.Substring(2, 2), 16),
Convert.ToByte(rgb.Substring(4, 2), 16)
);
return new CellFill { BackgroundColor = new ExColor(color) };
return new CellFill { BackgroundColor = color };
}
public override bool Equals(object? obj) => obj is CellFill other && Equals(other);

View File

@@ -13,7 +13,7 @@ public readonly struct CellFont : IEquatable<CellFont>
public string? FontFamily { get; init; }
/// <summary>Цвет текста.</summary>
public ExColor? FontColor { get; init; }
public System.Drawing.Color? FontColor { get; init; }
/// <summary>Жирное начертание.</summary>
public bool? IsBold { get; init; }
@@ -27,6 +27,35 @@ public readonly struct CellFont : IEquatable<CellFont>
/// <summary>Зачёркивание.</summary>
public bool? IsStrike { get; init; }
internal bool TryMerge(CellFont other, out CellFont result)
{
if (other.FontSize == FontSize &&
other.FontFamily == FontFamily &&
other.FontColor == FontColor &&
other.IsBold == IsBold &&
other.IsItalic == IsItalic &&
other.IsUnderline == IsUnderline &&
other.IsStrike == IsStrike)
{
result = default;
return false;
}
result = new CellFont
{
FontSize = other.FontSize ?? FontSize,
FontFamily = other.FontFamily ?? FontFamily,
FontColor = other.FontColor ?? FontColor,
IsBold = other.IsBold ?? IsBold,
IsItalic = other.IsItalic ?? IsItalic,
IsUnderline = other.IsUnderline ?? IsUnderline,
IsStrike = other.IsStrike ?? IsStrike
};
return true;
}
/// <summary>Создаёт элемент Font для Open XML.</summary>
public Font? ToFont()
{
@@ -39,11 +68,8 @@ public readonly struct CellFont : IEquatable<CellFont>
font.FontSize = new FontSize { Val = FontSize.Value };
if (FontFamily is not null)
font.FontName = new FontName { Val = FontFamily };
if (FontColor.HasValue && FontColor.Value.Color.HasValue)
{
var c = FontColor.Value.Color.Value;
if (FontColor is { } c)
font.Color = new Color { Rgb = $"{c.R:X2}{c.G:X2}{c.B:X2}" };
}
if (IsBold == true) font.Bold = new Bold();
if (IsItalic == true) font.Italic = new Italic();
if (IsUnderline == true) font.Underline = new Underline();
@@ -75,7 +101,7 @@ public readonly struct CellFont : IEquatable<CellFont>
Convert.ToByte(rgb.Substring(2, 2), 16),
Convert.ToByte(rgb.Substring(4, 2), 16)
);
result = result with { FontColor = new ExColor(color) };
result = result with { FontColor = color };
}
return result;
}

View File

@@ -0,0 +1,153 @@
namespace QWERTYkez.ExcelProcessor;
/// <summary>
/// Неизменяемый набор всех параметров оформления ячейки.
/// </summary>
public sealed record CellStyle
{
public CellAlign? Align { get; init; }
public CellFont? Font { get; init; }
public CellFill? Fill { get; init; }
public CellBorder? Border { get; init; }
public NumberFormatPattern? NumberFormat { get; init; }
public bool IsEmpty() =>
Align == null && Font == null && Fill == null && Border == null && NumberFormat == null;
internal bool TryMerge(NumberFormatPattern format, out CellStyle result)
{
result = this;
if (format is not null)
{
if (!Equals(NumberFormat, format))
{
result = result with { NumberFormat = format };
return true;
}
}
return false;
}
internal bool TryMerge(CellAlign align, out CellStyle result)
{
result = this;
var current = Align ?? new CellAlign();
if (current.TryMerge(align, out align))
{
result = result with { Align = align };
return true;
}
return false;
}
internal bool TryMerge(CellBorder border, out CellStyle result)
{
result = this;
var current = Border ?? new CellBorder();
if (current.TryMerge(border, out border))
{
result = result with { Border = border };
return true;
}
return false;
}
internal bool TryMerge(CellFill fill, out CellStyle result)
{
result = this;
var current = Fill ?? new CellFill();
if (!current.Equals(fill))
{
result = result with { Fill = fill };
return true;
}
return false;
}
internal bool TryMerge(CellFont font, out CellStyle result)
{
result = this;
var current = Font ?? new CellFont();
if (!current.Equals(font))
{
result = result with { Font = font };
return true;
}
return false;
}
internal CellStyle Merge(CellStyle other)
{
if (other == null) return this;
return new CellStyle
{
Align = other.Align ?? Align,
Font = other.Font ?? Font,
Fill = other.Fill ?? Fill,
Border = other.Border ?? Border,
NumberFormat = other.NumberFormat ?? NumberFormat
};
}
internal bool TryMerge(CellStyle other, out CellStyle result)
{
result = this;
bool changed = false;
// Объединяем Align, если есть
if (other.Align is not null)
{
var currentAlign = Align ?? new CellAlign();
if (currentAlign.TryMerge(other.Align.Value, out var newAlign))
{
result = result with { Align = newAlign };
changed = true;
}
}
// Аналогично для Font
if (other.Font is not null)
{
var currentFont = Font ?? new CellFont();
if (currentFont.TryMerge(other.Font.Value, out var newFont))
{
result = result with { Font = newFont };
changed = true;
}
}
// Для Fill просто заменяем, если он задан (нет nullable-полей)
if (other.Fill is not null)
{
if (!Fill.Equals(other.Fill))
{
result = result with { Fill = other.Fill };
changed = true;
}
}
// Border
if (other.Border is not null)
{
var currentBorder = Border ?? new CellBorder();
if (currentBorder.TryMerge(other.Border.Value, out var newBorder))
{
result = result with { Border = newBorder };
changed = true;
}
}
// NumberFormat
if (other.NumberFormat is not null)
{
if (!Equals(NumberFormat, other.NumberFormat))
{
result = result with { NumberFormat = other.NumberFormat };
changed = true;
}
}
return changed;
}
}

View File

@@ -6,15 +6,15 @@
/// </summary>
public readonly struct ColumnWidth
{
private readonly double _rawValue;
private readonly UnitType _unit;
readonly double _rawValue;
readonly UnitType _unit;
private enum UnitType { Characters, Points, Centimeters, Millimeters }
enum UnitType { Characters, Points, Centimeters, Millimeters }
/// <summary>Коэффициент перевода символов в пункты по умолчанию (используется, если нет калибровочной таблицы).</summary>
public static double DefaultPointsPerChar { get; set; } = 5.65;
private ColumnWidth(double value, UnitType unit)
ColumnWidth(double value, UnitType unit)
{
_rawValue = value;
_unit = unit;

View File

@@ -1,82 +0,0 @@
namespace QWERTYkez.ExcelProcessor;
public readonly struct ExColor(System.Drawing.Color? Color)
{
private readonly System.Drawing.Color? color = Color;
/// <summary>Проверяет, является ли цвет автоматическим (т.е. Color == null).</summary>
public bool IsAuto => color is null;
public System.Drawing.Color? Color => color;
public static ExColor FromArgb(byte r, byte g, byte b) => new(System.Drawing.Color.FromArgb(r, g, b));
public static ExColor FromName(string knownColor) => new(System.Drawing.Color.FromName(knownColor));
public static implicit operator ExColor(Color exColor) => FromExcel(exColor);
public static implicit operator Color(ExColor color) => color.ToExcel();
public static ExColor FromExcel(Color excelColor)
{
if (excelColor == null)
return new ExColor(null);
// Если цвет автоматический
if (excelColor.Auto != null && excelColor.Auto.Value)
return new ExColor(null);
// Если задан RGB
if (excelColor.Rgb?.Value is { } rgb && rgb.Length > 0)
{
if (rgb.Length == 6) // RRGGBB
{
byte r = Convert.ToByte(rgb.Substring(0, 2), 16);
byte g = Convert.ToByte(rgb.Substring(2, 2), 16);
byte b = Convert.ToByte(rgb.Substring(4, 2), 16);
return new ExColor(System.Drawing.Color.FromArgb(r, g, b));
}
else if (rgb.Length == 8) // AARRGGBB (альфа игнорируется)
{
byte r = Convert.ToByte(rgb.Substring(2, 2), 16);
byte g = Convert.ToByte(rgb.Substring(4, 2), 16);
byte b = Convert.ToByte(rgb.Substring(6, 2), 16);
return new ExColor(System.Drawing.Color.FromArgb(r, g, b));
}
}
// По умолчанию — автоматический цвет
return new ExColor(null);
}
public Color ToExcel()
{
var excelColor = new Color();
{
if (color.HasValue)
{
excelColor.Rgb = $"{color.Value.R:X2}{color.Value.G:X2}{color.Value.B:X2}";
}
else excelColor.Auto = true;
}
return excelColor;
}
public static ExColor FromRgb(string rgb)
{
if (string.IsNullOrEmpty(rgb)) return new ExColor(null);
if (rgb.Length == 6)
{
byte r = Convert.ToByte(rgb.Substring(0, 2), 16);
byte g = Convert.ToByte(rgb.Substring(2, 2), 16);
byte b = Convert.ToByte(rgb.Substring(4, 2), 16);
return new ExColor(System.Drawing.Color.FromArgb(r, g, b));
}
else if (rgb.Length == 8)
{
byte r = Convert.ToByte(rgb.Substring(2, 2), 16);
byte g = Convert.ToByte(rgb.Substring(4, 2), 16);
byte b = Convert.ToByte(rgb.Substring(6, 2), 16);
return new ExColor(System.Drawing.Color.FromArgb(r, g, b));
}
return new ExColor(null);
}
}

File diff suppressed because it is too large Load Diff

View File

@@ -1,270 +0,0 @@
namespace QWERTYkez.ExcelProcessor;
/// <summary>
/// Внутренняя реализация <see cref="ICellText"/> для работы с богатым текстом ячейки.
/// Хранит коллекцию фрагментов <see cref="ExcelRun"/>.
/// Минимизирует аллокации, не использует рефлексию.
/// </summary>
internal sealed class ExcelCellText : ICellText
{
private List<IRun>? _runs;
/// <inheritdoc />
public int Count => _runs?.Count ?? 0;
/// <inheritdoc />
public IEnumerable<IRun> GetRuns()
{
if (_runs is null)
return [];
// Возвращаем сам список, чтобы избежать копирования.
// Вызывающий не должен модифицировать коллекцию.
return _runs;
}
/// <inheritdoc />
public IRun? GetRunAt(int index)
{
if (_runs is null || index < 0 || index >= _runs.Count)
return null;
return _runs[index];
}
/// <inheritdoc />
public bool TryGetRunAt(int index, out IRun run)
{
run = GetRunAt(index)!;
return run != null;
}
/// <inheritdoc />
public IRun? First()
{
if (_runs is null || _runs.Count == 0)
return null;
return _runs[0];
}
/// <inheritdoc />
public bool TryGetFirst(out IRun run)
{
run = First()!;
return run != null;
}
/// <inheritdoc />
public IRun? Last()
{
if (_runs is null || _runs.Count == 0)
return null;
return _runs[_runs.Count - 1];
}
/// <inheritdoc />
public bool TryGetLast(out IRun run)
{
run = Last()!;
return run != null;
}
/// <inheritdoc />
public bool TryRemoveRun(IRun run)
{
if (run is null || _runs is null)
return false;
return _runs.Remove(run);
}
/// <inheritdoc />
public bool TryRemoveRun(int index)
{
if (_runs is null || index < 0 || index >= _runs.Count)
return false;
_runs.RemoveAt(index);
return true;
}
/// <inheritdoc />
public bool TryRemoveRun(int index, out IRun? removed)
{
removed = GetRunAt(index);
if (removed is null)
return false;
return TryRemoveRun(index);
}
/// <inheritdoc />
public ICellText AddBreak()
{
// Добавляем символ переноса строки в последний существующий Run
if (_runs != null && _runs.Count > 0)
{
var lastRun = _runs[_runs.Count - 1];
lastRun.Text += "\n";
}
else
{
// Если нет ни одного Run, создаём новый с символом переноса
AddRun("\n", null);
}
return this;
}
/// <inheritdoc />
public ICellText AddRun(string text, RunFormat? format = null)
{
if (string.IsNullOrEmpty(text))
return this;
_runs ??= [];
var run = new ExcelRun { Text = text, Format = format };
_runs.Add(run);
return this;
}
/// <inheritdoc />
public ICellText AddRunBreak(string text, RunFormat? format = null)
{
AddRun(text, format);
AddBreak();
return this;
}
/// <inheritdoc />
public ICellText AddSubRun(string text, RunFormat? format = null)
{
var subFormat = format is { } fmt
? new RunFormat
{
IsBold = fmt.IsBold,
IsItalic = fmt.IsItalic,
Underline = fmt.Underline,
IsStrike = fmt.IsStrike,
Color = fmt.Color,
FontSize = fmt.FontSize,
FontFamily = fmt.FontFamily,
Vertical = VerticalTextRunAlignment.Subscript
}
: new RunFormat { Vertical = VerticalTextRunAlignment.Subscript };
return AddRun(text, subFormat);
}
/// <inheritdoc />
public ICellText AddSupRun(string text, RunFormat? format = null)
{
var supFormat = format is { } fmt
? new RunFormat
{
IsBold = fmt.IsBold,
IsItalic = fmt.IsItalic,
Underline = fmt.Underline,
IsStrike = fmt.IsStrike,
Color = fmt.Color,
FontSize = fmt.FontSize,
FontFamily = fmt.FontFamily,
Vertical = VerticalTextRunAlignment.Subscript
}
: new RunFormat { Vertical = VerticalTextRunAlignment.Superscript };
return AddRun(text, supFormat);
}
/// <inheritdoc />
public bool TryInsertRun(int index, string text, RunFormat? format = null)
{
if (index < 0 || string.IsNullOrEmpty(text))
return false;
_runs ??= [];
if (index > _runs.Count)
return false;
var run = new ExcelRun { Text = text, Format = format };
_runs.Insert(index, run);
return true;
}
/// <inheritdoc />
public bool TryInsertRunBreak(int index, string text, RunFormat? format = null)
{
if (!TryInsertRun(index, text, format))
return false;
// После вставленного run добавляем break на следующей позиции
AddBreak();
// Сдвигаем? Просто добавляем break в конец неверно. Break должен быть сразу после вставленного.
// Но AddBreak добавляет в конец. Нужно вставить break на index+1.
return TryInsertRun(index + 1, "\n", null);
}
/// <inheritdoc />
public bool TryInsertSubRun(int index, string text, RunFormat? format = null)
{
var subFormat = format is { } fmt
? new RunFormat
{
IsBold = fmt.IsBold,
IsItalic = fmt.IsItalic,
Underline = fmt.Underline,
IsStrike = fmt.IsStrike,
Color = fmt.Color,
FontSize = fmt.FontSize,
FontFamily = fmt.FontFamily,
Vertical = VerticalTextRunAlignment.Subscript
}
: new RunFormat { Vertical = VerticalTextRunAlignment.Subscript };
return TryInsertRun(index, text, subFormat);
}
/// <inheritdoc />
public bool TryInsertSupRun(int index, string text, RunFormat? format = null)
{
var supFormat = format is { } fmt
? new RunFormat
{
IsBold = fmt.IsBold,
IsItalic = fmt.IsItalic,
Underline = fmt.Underline,
IsStrike = fmt.IsStrike,
Color = fmt.Color,
FontSize = fmt.FontSize,
FontFamily = fmt.FontFamily,
Vertical = VerticalTextRunAlignment.Subscript
}
: new RunFormat { Vertical = VerticalTextRunAlignment.Superscript };
return TryInsertRun(index, text, supFormat);
}
/// <inheritdoc />
public void ApplyFormatToAllRuns(RunFormat format)
{
if (_runs is null || _runs.Count == 0)
return;
foreach (var run in _runs)
{
if (run is ExcelRun xRun)
{
// Объединение форматов: ненулевые свойства overlay заменяют значения в base.
var baseFmt = xRun.Format ?? new RunFormat();
xRun.Format = MergeRunFormat(baseFmt, format);
}
}
}
/// <inheritdoc />
public void Clear()
{
_runs?.Clear();
_runs = null;
}
private static RunFormat MergeRunFormat(RunFormat baseFmt, RunFormat overlay)
{
return new RunFormat
{
IsBold = overlay.IsBold ?? baseFmt.IsBold,
IsItalic = overlay.IsItalic ?? baseFmt.IsItalic,
Underline = overlay.Underline ?? baseFmt.Underline,
IsStrike = overlay.IsStrike ?? baseFmt.IsStrike,
Color = overlay.Color ?? baseFmt.Color,
FontSize = overlay.FontSize ?? baseFmt.FontSize,
FontFamily = overlay.FontFamily ?? baseFmt.FontFamily,
Vertical = overlay.Vertical ?? baseFmt.Vertical
};
}
}

View File

@@ -16,7 +16,161 @@ internal sealed class ExcelColumn : IColumn
_colIndex = colIndex;
}
public uint Index => _colIndex;
private CellStyle? GetColumnStyle(Column columnElement)
{
if (columnElement.Style?.Value is not uint styleIndex)
return null;
return _writer.GetCellStyle(styleIndex);
}
private void ApplyStyleToColumn(CellStyle style)
{
var columnElement = GetOrCreateColumnElement();
int styleIndex = _writer.GetOrCreateStyleId(style);
columnElement.Style = (uint)styleIndex;
// Принудительно устанавливаем customStyle через атрибут
if (!columnElement.ExtendedAttributes.Any(attr => attr.LocalName == "customStyle" && attr.Value == "1"))
columnElement.SetAttribute(new OpenXmlAttribute("customStyle", "", "1"));
}
internal static Column? GetColumnElementInternal(ExcelSheet sheet, uint colIndex)
{
var cols = sheet.Worksheet.GetFirstChild<Columns>();
if (cols == null) return null;
foreach (Column col in cols.Elements<Column>())
{
if (col.Min?.Value is { } min && col.Max?.Value is { } max && min <= colIndex && max >= colIndex)
return col;
}
return null;
}
/// <inheritdoc />
public IColumn Set(NumberFormatPattern format)
{
if (format == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(format, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IColumn Set(CellAlign align)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(align, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IColumn Set(CellBorder border)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(border, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IColumn Set(CellFill fill)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(fill, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IColumn Set(CellFont font)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(font, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IColumn Set(CellStyle style)
{
if (style == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var columnElement = GetOrCreateColumnElement();
var currentStyle = GetColumnStyle(columnElement) ?? new CellStyle();
if (currentStyle.TryMerge(style, out var newStyle))
{
ApplyStyleToColumn(newStyle);
ApplyStyleToColumnCells(newStyle);
}
return this;
}
}
private void ApplyStyleToColumnCells(CellStyle style)
{
var sheetData = _sheet.GetSheetData();
foreach (var rowElement in sheetData.Elements<Row>())
{
var cellElement = FindCellInRow(rowElement, _colIndex);
if (cellElement != null)
{
var cell = new ExcelCell(_writer, _sheet, rowElement.RowIndex!.Value, _colIndex);
cell.Set(style);
}
}
}
public uint Index => _colIndex;
public string IndexLetter => NumberToColumnLetter(_colIndex);
@@ -64,23 +218,7 @@ internal sealed class ExcelColumn : IColumn
}
}
// Вспомогательные внутренние методы (перенести существующую логику)
private static Column? GetColumnElementInternal(ExcelSheet sheet, uint colIndex)
{
var cols = sheet.Worksheet.GetFirstChild<Columns>();
if (cols == null) return null;
foreach (Column col in cols.Elements<Column>())
{
if (col?.Min?.Value is { } min
&& col?.Max?.Value is { } max
&& min <= colIndex
&& max >= colIndex)
return col;
}
return null;
}
private static Column GetOrCreateColumnElementInternal(ExcelSheet sheet, uint colIndex)
static Column GetOrCreateColumnElementInternal(ExcelSheet sheet, uint colIndex)
{
var existing = GetColumnElementInternal(sheet, colIndex);
if (existing != null) return existing;
@@ -127,62 +265,46 @@ internal sealed class ExcelColumn : IColumn
}
public ColumnWidth Width
{
get
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
get
{
var column = GetColumnElement();
if (column is Column col && col.Width?.Value is { } val && col.CustomWidth?.Value == true)
return ColumnWidth.FromCharacters(val);
return ColumnWidth.FromCharacters(8.43); // стандартная ширина
}
}
set
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var col = GetOrCreateColumnElement();
double widthInChars;
if (value.IsCharacterUnit)
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
// Если ширина задана в символах, используем напрямую
widthInChars = value.CharacterValue;
}
else
{
double targetCm = value.GetTargetCentimeters();
widthInChars = _writer.TryGetCalibrateCoeff(targetCm, out var closestCw)
? closestCw : targetCm * (ColumnWidth.DefaultPointsPerChar / 28.3464566929);
}
col.Width = widthInChars;
col.CustomWidth = true;
}
}
}
public IColumn SetNumberFormat(NumberFormatPattern format)
{
if (format == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int fmtId = GetOrCreateNumberFormatId(format);
// Применяем ко всем ячейкам столбца
var sheetData = _sheet.GetSheetData();
foreach (var row in sheetData.Elements<Row>())
{
var cell = FindCellInRow(row, _colIndex);
cell?.StyleIndex = (uint)fmtId;
var column = GetColumnElement();
if (column is Column col && col.Width?.Value is { } val && col.CustomWidth?.Value == true)
return ColumnWidth.FromCharacters(val);
return ColumnWidth.FromCharacters(8.43); // стандартная ширина
}
}
set
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var col = GetOrCreateColumnElement();
double widthInChars;
if (value.IsCharacterUnit)
{
// Если ширина задана в символах, используем напрямую
widthInChars = value.CharacterValue;
}
else
{
double targetCm = value.GetTargetCentimeters();
widthInChars = _writer.TryGetCalibrateCoeff(targetCm, out var closestCw)
? closestCw : targetCm * (ColumnWidth.DefaultPointsPerChar / 28.3464566929);
}
col.Width = widthInChars;
col.CustomWidth = true;
}
}
return this;
}
public ICell Cell(uint row)
{
return new ExcelCell(_writer, _sheet, row, _colIndex);
@@ -207,7 +329,7 @@ internal sealed class ExcelColumn : IColumn
public IColumn Cell(uint row, string value, NumberFormatPattern? format = null)
{
Cell(row).Set(value, format); return this;
Cell(row).SetFormula(value, format); return this;
}
public IColumn Cell(uint row, DateTime value, NumberFormatPattern? format = null)
@@ -305,7 +427,7 @@ internal sealed class ExcelColumn : IColumn
// Вспомогательные методы
private Column? GetColumnElement()
Column? GetColumnElement()
{
var cols = _sheet.Worksheet.GetFirstChild<Columns>();
if (cols == null) return null;
@@ -320,7 +442,7 @@ internal sealed class ExcelColumn : IColumn
return null;
}
private Column GetOrCreateColumnElement()
Column GetOrCreateColumnElement()
{
var existing = GetColumnElement();
if (existing != null) return existing;
@@ -344,13 +466,13 @@ internal sealed class ExcelColumn : IColumn
return newCol;
}
private void DeleteColumnElement()
void DeleteColumnElement()
{
var col = GetColumnElement();
col?.Remove();
}
private void CopyColumnData(uint sourceCol, uint targetCol)
void CopyColumnData(uint sourceCol, uint targetCol)
{
if (sourceCol == targetCol) return;
var sheetData = _sheet.GetSheetData();
@@ -389,7 +511,7 @@ internal sealed class ExcelColumn : IColumn
}
}
private void ClearColumnData(uint col)
void ClearColumnData(uint col)
{
var sheetData = _sheet.GetSheetData();
foreach (var row in sheetData.Elements<Row>().ToList())
@@ -399,7 +521,7 @@ internal sealed class ExcelColumn : IColumn
}
}
private Cell? FindCellInRow(Row row, uint colIndex)
Cell? FindCellInRow(Row row, uint colIndex)
{
foreach (var cell in row.Elements<Cell>())
{
@@ -410,7 +532,7 @@ internal sealed class ExcelColumn : IColumn
return null;
}
private void InsertCellInRow(Row row, Cell cell, uint colIndex)
void InsertCellInRow(Row row, Cell cell, uint colIndex)
{
string newRef = NumberToColumnLetter(colIndex) + (row.RowIndex?.Value ?? 1).ToString();
cell.CellReference = newRef;
@@ -430,7 +552,7 @@ internal sealed class ExcelColumn : IColumn
row.Append(cell);
}
private Column? GetColumnElementForIndex(uint col)
Column? GetColumnElementForIndex(uint col)
{
var cols = _sheet.Worksheet.GetFirstChild<Columns>();
if (cols == null) return null;
@@ -445,7 +567,7 @@ internal sealed class ExcelColumn : IColumn
return null;
}
private Column GetOrCreateColumnElementForIndex(uint col)
Column GetOrCreateColumnElementForIndex(uint col)
{
var existing = GetColumnElementForIndex(col);
if (existing != null) return existing;
@@ -468,12 +590,12 @@ internal sealed class ExcelColumn : IColumn
return newCol;
}
private int GetOrCreateNumberFormatId(NumberFormatPattern format)
int GetOrCreateNumberFormatId(NumberFormatPattern format)
{
return _writer.GetOrCreateCellFormatId(numberFormat: format);
}
private static bool TryParseCellReference(string reference, out uint row, out uint col)
static bool TryParseCellReference(string reference, out uint row, out uint col)
{
row = 0; col = 0;
if (string.IsNullOrEmpty(reference)) return false;
@@ -487,7 +609,7 @@ internal sealed class ExcelColumn : IColumn
return true;
}
private static string NumberToColumnLetter(uint col)
static string NumberToColumnLetter(uint col)
{
if (col == 0) throw new ArgumentException("Column number must be > 0");
string result = "";

View File

@@ -5,10 +5,10 @@
/// </summary>
internal sealed class ExcelRange : IRange
{
private readonly ExcelWriter _writer;
private readonly ExcelSheet _sheet;
private uint _rowStart, _rowEnd;
private uint _colStart, _colEnd;
internal readonly ExcelWriter _writer;
internal readonly ExcelSheet _sheet;
internal uint _rowStart, _rowEnd;
internal uint _colStart, _colEnd;
internal ExcelRange(ExcelWriter writer, ExcelSheet sheet, uint rowStart, uint colStart, uint rowEnd, uint colEnd)
{
@@ -55,17 +55,90 @@ internal sealed class ExcelRange : IRange
void ForEachCell(Action<ICell> action)
{
for (uint r = _rowStart; r <= _rowEnd; r++)
for (uint c = _colStart; c <= _colEnd; c++)
{
var cell = new ExcelCell(_writer, _sheet, r, c);
action(cell);
}
}
public IRange Set(NumberFormatPattern format)
{
if (format == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(format));
return this;
}
}
public IRange Set(CellAlign align)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(align));
return this;
}
}
public IRange Set(CellBorder border)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(border));
return this;
}
}
public IRange Set(CellFill fill)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(fill));
return this;
}
}
public IRange Set(CellFont font)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(font));
return this;
}
}
public IRange Set(CellStyle style)
{
if (style == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
ForEachCell(cell => cell.Set(style));
return this;
}
}
/// <summary>
/// Перемещает текущий диапазон в новую позицию (как "вырезать-вставить").
/// Исходный диапазон очищается, а текущий объект IRange перемещается на новое место.
/// </summary>
/// <param name="newRow">Номер строки для нового верхнего левого угла.</param>
/// <param name="newCol">Номер столбца для нового верхнего левого угла.</param>
/// <returns>Тот же объект IRange с новыми координатами.</returns>
public IRange MoveTo(uint newRow, uint newCol)
/// <summary>
/// Перемещает текущий диапазон в новую позицию (как "вырезать-вставить").
/// Исходный диапазон очищается, а текущий объект IRange перемещается на новое место.
/// </summary>
/// <param name="newRow">Номер строки для нового верхнего левого угла.</param>
/// <param name="newCol">Номер столбца для нового верхнего левого угла.</param>
/// <returns>Тот же объект IRange с новыми координатами.</returns>
public IRange MoveTo(uint newRow, uint newCol)
{
if (newRow == _rowStart && newCol == _colStart) return this;
_writer.ThrowIfDisposed();
@@ -115,7 +188,7 @@ internal sealed class ExcelRange : IRange
return MoveTo(rowIndex, CellAddressHelper.ColumnLetterToIndex(colIndex));
}
private enum CopyOrder
enum CopyOrder
{
Any,
LeftToRight,
@@ -125,7 +198,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Копирует ячейки из исходного диапазона в целевой, поддерживая различные порядки обхода.</summary>
private void CopyCells(uint srcRowStart, uint srcColStart, uint srcRowEnd, uint srcColEnd,
void CopyCells(uint srcRowStart, uint srcColStart, uint srcRowEnd, uint srcColEnd,
uint dstRowStart, uint dstColStart, CopyOrder order)
{
// Определяем все строки, которые могут понадобиться (исходные и целевые)
@@ -172,7 +245,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Строит словарь строк для указанных индексов строк.</summary>
private Dictionary<uint, Row> GetRowDictionary(HashSet<uint> rowIndices)
Dictionary<uint, Row> GetRowDictionary(HashSet<uint> rowIndices)
{
var dict = new Dictionary<uint, Row>();
var sheetData = _sheet.GetSheetData();
@@ -185,7 +258,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Быстрый поиск ячейки в строке (линейный, подходит для типичного количества ячеек в строке).</summary>
private Cell? FindCellInRowFast(Row row, uint colIndex)
Cell? FindCellInRowFast(Row row, uint colIndex)
{
foreach (var cell in row.Elements<Cell>())
{
@@ -196,14 +269,14 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Проверяет, пересекаются ли два прямоугольных диапазона.</summary>
private static bool RangesOverlap(uint r1s, uint c1s, uint r1e, uint c1e,
static bool RangesOverlap(uint r1s, uint c1s, uint r1e, uint c1e,
uint r2s, uint c2s, uint r2e, uint c2e)
{
return !(r1e < r2s || r2e < r1s || c1e < c2s || c2e < c1s);
}
/// <summary>Вставляет ячейку в указанную позицию, предварительно удаляя существующую.</summary>
private void InsertCellAt(uint row, uint col, Cell cell)
void InsertCellAt(uint row, uint col, Cell cell)
{
DeleteCellAt(row, col);
var sheetData = _sheet.GetSheetData();
@@ -212,7 +285,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Удаляет ячейку, если она существует.</summary>
private void DeleteCellAt(uint row, uint col)
void DeleteCellAt(uint row, uint col)
{
var sheetData = _sheet.GetSheetData();
var rowElement = FindRowElement(sheetData, row);
@@ -222,7 +295,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Получает или создаёт строку с указанным индексом.</summary>
private Row GetOrCreateRowElement(SheetData sheetData, uint rowIndex)
Row GetOrCreateRowElement(SheetData sheetData, uint rowIndex)
{
var existing = FindRowElement(sheetData, rowIndex);
if (existing != null) return existing;
@@ -232,7 +305,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Вставляет ячейку в строку с сохранением порядка столбцов.</summary>
private void InsertCellInRow(Row row, Cell cell, uint colIndex)
void InsertCellInRow(Row row, Cell cell, uint colIndex)
{
string newRef = $"{CellAddressHelper.ColumnIndexToLetter(colIndex)}{row.RowIndex?.Value ?? 1}";
cell.CellReference = newRef;
@@ -251,7 +324,7 @@ internal sealed class ExcelRange : IRange
}
/// <summary>Очищает данные (содержимое) в указанном диапазоне.</summary>
private void ClearRangeData(uint rowStart, uint colStart, uint rowEnd, uint colEnd)
void ClearRangeData(uint rowStart, uint colStart, uint rowEnd, uint colEnd)
{
for (uint r = rowStart; r <= rowEnd; r++)
{
@@ -334,103 +407,12 @@ internal sealed class ExcelRange : IRange
return merged.Equals(range);
}
public IRange SetNumberFormat(NumberFormatPattern format)
{
if (format == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int fmtId = GetOrCreateNumberFormatId(format);
for (uint r = _rowStart; r <= _rowEnd; r++)
{
for (uint c = _colStart; c <= _colEnd; c++)
{
var cell = GetCellInternal(r, c);
cell?.StyleIndex = (uint)fmtId;
}
}
}
return this;
}
/// <inheritdoc />
public IRange SetCellAlign(CellAlign align)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int styleIndex = _writer.GetOrCreateCellFormatId(
numberFormat: null,
font: null,
fill: null,
border: null,
align: align
);
ApplyStyleToRange((uint)styleIndex);
return this;
}
}
/// <inheritdoc />
public IRange SetCellBorder(CellBorder border)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int styleIndex = _writer.GetOrCreateCellFormatId(
numberFormat: null,
font: null,
fill: null,
border: border,
align: null
);
ApplyStyleToRange((uint)styleIndex);
return this;
}
}
/// <inheritdoc />
public IRange SetCellFill(CellFill fill)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int styleIndex = _writer.GetOrCreateCellFormatId(
numberFormat: null,
font: null,
fill: fill,
border: null,
align: null
);
ApplyStyleToRange((uint)styleIndex);
return this;
}
}
/// <inheritdoc />
public IRange SetCellFont(CellFont font)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
int styleIndex = _writer.GetOrCreateCellFormatId(
numberFormat: null,
font: font,
fill: null,
border: null,
align: null
);
ApplyStyleToRange((uint)styleIndex);
return this;
}
}
/// <summary>
/// Применяет стиль ко всем ячейкам диапазона.
/// </summary>
/// <param name="styleIndex">Индекс стиля для применения.</param>
/// <param name="createIfMissing">Если true, создаёт недостающие ячейки (по умолчанию true).</param>
private void ApplyStyleToRange(uint styleIndex, bool createIfMissing = true)
void ApplyStyleToRange(uint styleIndex, bool createIfMissing = true)
{
for (uint row = _rowStart; row <= _rowEnd; row++)
{
@@ -517,7 +499,7 @@ internal sealed class ExcelRange : IRange
{
if (GetSubCell(row, col, out var cell))
{
cell.Set(formula, format);
cell.SetFormula(formula, format);
return true;
}
return false;
@@ -619,7 +601,7 @@ internal sealed class ExcelRange : IRange
{
if (GetSubCell(row, col, out var cell))
{
cell.Set(formula, format);
cell.SetFormula(formula, format);
return true;
}
return false;
@@ -727,7 +709,7 @@ internal sealed class ExcelRange : IRange
// Вспомогательные методы
private Cell? GetCellInternal(uint row, uint col)
Cell? GetCellInternal(uint row, uint col)
{
var sheetData = _sheet.GetSheetData();
var rowElement = FindRowElement(sheetData, row);
@@ -735,7 +717,7 @@ internal sealed class ExcelRange : IRange
return FindCellInRow(rowElement, col);
}
private void CopyData(uint srcRowStart, uint srcColStart, uint srcRowEnd, uint srcColEnd, uint dstRowStart, uint dstColStart)
void CopyData(uint srcRowStart, uint srcColStart, uint srcRowEnd, uint srcColEnd, uint dstRowStart, uint dstColStart)
{
// Сохраняем все значения и форматы из исходного диапазона
var cellsData = new List<(uint row, uint col, Cell cell)>();
@@ -784,7 +766,7 @@ internal sealed class ExcelRange : IRange
}
}
private Row GetOrCreateRowElement(uint rowIndex)
Row GetOrCreateRowElement(uint rowIndex)
{
var existing = FindRowElement(_sheet.GetSheetData(), rowIndex);
if (existing != null) return existing;
@@ -793,14 +775,14 @@ internal sealed class ExcelRange : IRange
return newRow;
}
private static Row? FindRowElement(SheetData sheetData, uint rowIndex)
static Row? FindRowElement(SheetData sheetData, uint rowIndex)
{
foreach (var row in sheetData.Elements<Row>())
if (row.RowIndex?.Value == rowIndex) return row;
return null;
}
private static void InsertRowElement(SheetData sheetData, Row row, uint rowIndex)
static void InsertRowElement(SheetData sheetData, Row row, uint rowIndex)
{
bool inserted = false;
foreach (var existing in sheetData.Elements<Row>().ToList())
@@ -815,7 +797,7 @@ internal sealed class ExcelRange : IRange
if (!inserted) sheetData.Append(row);
}
private Cell? FindCellInRow(Row row, uint colIndex)
Cell? FindCellInRow(Row row, uint colIndex)
{
foreach (var cell in row.Elements<Cell>())
{
@@ -825,7 +807,7 @@ internal sealed class ExcelRange : IRange
return null;
}
private Column? GetColumnElementForIndex(uint col)
Column? GetColumnElementForIndex(uint col)
{
var worksheet = _sheet.Worksheet;
var cols = worksheet.GetFirstChild<Columns>();
@@ -839,7 +821,7 @@ internal sealed class ExcelRange : IRange
return null;
}
private Column GetOrCreateColumnElementForIndex(uint col)
Column GetOrCreateColumnElementForIndex(uint col)
{
var existing = GetColumnElementForIndex(col);
if (existing != null) return existing;
@@ -861,10 +843,10 @@ internal sealed class ExcelRange : IRange
return newCol;
}
private MergeCells? GetMergeCells() =>
MergeCells? GetMergeCells() =>
_sheet.Worksheet.GetFirstChild<MergeCells>();
private bool TryParseRangeReference(string reference, out ExcelRange range)
bool TryParseRangeReference(string reference, out ExcelRange range)
{
range = null!;
if (string.IsNullOrEmpty(reference)) return false;
@@ -877,10 +859,4 @@ internal sealed class ExcelRange : IRange
range = new ExcelRange(_writer, _sheet, rowStart, colStart, rowEnd, colEnd);
return true;
}
// создаёт стиль только с числовым форматом
private int GetOrCreateNumberFormatId(NumberFormatPattern format)
{
return _writer.GetOrCreateCellFormatId(numberFormat: format);
}
}

View File

@@ -5,9 +5,9 @@
/// </summary>
internal sealed class ExcelRow : IRow
{
private readonly ExcelWriter _writer;
private readonly ExcelSheet _sheet;
private uint _rowIndex;
readonly ExcelWriter _writer;
readonly ExcelSheet _sheet;
uint _rowIndex;
internal ExcelRow(ExcelWriter writer, ExcelSheet sheet, uint rowIndex)
{
@@ -47,7 +47,7 @@ internal sealed class ExcelRow : IRow
}
// Вспомогательные методы
private static Row? GetRowElementInternal(ExcelSheet sheet, uint rowIndex)
static Row? GetRowElementInternal(ExcelSheet sheet, uint rowIndex)
{
var sheetData = sheet.GetSheetData();
foreach (var row in sheetData.Elements<Row>())
@@ -55,7 +55,7 @@ internal sealed class ExcelRow : IRow
return null;
}
private static Row GetOrCreateRowElementInternal(ExcelSheet sheet, uint rowIndex)
static Row GetOrCreateRowElementInternal(ExcelSheet sheet, uint rowIndex)
{
var existing = GetRowElementInternal(sheet, rowIndex);
if (existing != null) return existing;
@@ -66,7 +66,7 @@ internal sealed class ExcelRow : IRow
return newRow;
}
private static void InsertRowElementInternal(SheetData sheetData, Row row, uint rowIndex)
static void InsertRowElementInternal(SheetData sheetData, Row row, uint rowIndex)
{
bool inserted = false;
foreach (var existing in sheetData.Elements<Row>().ToList())
@@ -136,25 +136,157 @@ internal sealed class ExcelRow : IRow
}
}
}
private void ApplyStyleToRowCells(CellStyle style)
{
var sheetData = _sheet.GetSheetData();
var rowElement = FindRowElement(sheetData, _rowIndex);
if (rowElement == null) return;
// Применяем стиль ко всем существующим ячейкам
foreach (var cellElement in rowElement.Elements<Cell>())
{
var colIndex = GetColumnIndex(cellElement.CellReference?.Value);
if (colIndex > 0)
{
var cell = new ExcelCell(_writer, _sheet, _rowIndex, colIndex);
// Применяем стиль через Set, чтобы объединить с существующим
cell.Set(style);
}
}
}
static uint GetColumnIndex(string? cellReference)
{
if (string.IsNullOrEmpty(cellReference)) return 0;
int i = 0;
while (i < cellReference!.Length && char.IsLetter(cellReference[i])) i++;
if (i == 0) return 0;
string colPart = cellReference.Substring(0, i);
return CellAddressHelper.ColumnLetterToIndex(colPart);
}
CellStyle? GetRowStyle(Row rowElement)
{
if (rowElement.StyleIndex?.Value is not uint styleIndex)
return null;
return _writer.GetCellStyle(styleIndex);
}
/// <inheritdoc />
public IRow SetNumberFormat(NumberFormatPattern format)
public IRow Set(NumberFormatPattern format)
{
if (format == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
// Находим все ячейки в этой строке и устанавливаем формат
var sheetData = _sheet.GetSheetData();
var rowElement = FindRowElement(sheetData, _rowIndex);
if (rowElement == null) return this;
int formatIndex = GetOrCreateNumberFormatId(format);
foreach (var cell in rowElement.Elements<Cell>())
cell.StyleIndex = (uint)formatIndex;
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(format, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
return this;
}
/// <inheritdoc />
public IRow Set(CellAlign align)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(align, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IRow Set(CellBorder border)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(border, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IRow Set(CellFill fill)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(fill, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IRow Set(CellFont font)
{
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(font, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
}
/// <inheritdoc />
public IRow Set(CellStyle style)
{
if (style == null) return this;
_writer.ThrowIfDisposed();
lock (_writer._syncLock)
{
var rowElement = GetOrCreateRowElement();
var currentStyle = GetRowStyle(rowElement) ?? new CellStyle();
if (currentStyle.TryMerge(style, out var newStyle))
{
ApplyStyleToRow(newStyle);
ApplyStyleToRowCells(newStyle);
}
return this;
}
}
void ApplyStyleToRow(CellStyle style)
{
var rowElement = GetOrCreateRowElement();
int styleIndex = _writer.GetOrCreateStyleId(style);
rowElement.StyleIndex = (uint)styleIndex;
rowElement.CustomFormat = true;
}
/// <inheritdoc />
public ICell Cell(uint col) => new ExcelCell(_writer, _sheet, _rowIndex, col);
@@ -176,7 +308,7 @@ internal sealed class ExcelRow : IRow
public IRow Cell(uint col, string value, NumberFormatPattern? format = null)
{
Cell(col).Set(value, format); return this;
Cell(col).SetFormula(value, format); return this;
}
public IRow Cell(uint col, DateTime value, NumberFormatPattern? format = null)
@@ -226,7 +358,7 @@ internal sealed class ExcelRow : IRow
public IRow Cell(string col, string value, NumberFormatPattern? format = null)
{
Cell(col).Set(value, format); return this;
Cell(col).SetFormula(value, format); return this;
}
public IRow Cell(string col, DateTime value, NumberFormatPattern? format = null)
@@ -347,7 +479,7 @@ internal sealed class ExcelRow : IRow
// Вспомогательные методы
private Row GetOrCreateRowElement()
Row GetOrCreateRowElement()
{
var sheetData = _sheet.GetSheetData();
var existing = FindRowElement(sheetData, _rowIndex);
@@ -357,7 +489,7 @@ internal sealed class ExcelRow : IRow
return newRow;
}
private static Row? FindRowElement(SheetData sheetData, uint rowIndex)
static Row? FindRowElement(SheetData sheetData, uint rowIndex)
{
// Поиск по атрибуту RowIndex. В Open XML строки могут идти не по порядку.
foreach (var row in sheetData.Elements<Row>())
@@ -368,7 +500,7 @@ internal sealed class ExcelRow : IRow
return null;
}
private static void InsertRowElement(SheetData sheetData, Row row, uint rowIndex)
static void InsertRowElement(SheetData sheetData, Row row, uint rowIndex)
{
// Вставка с сохранением сортировки по RowIndex
bool inserted = false;
@@ -385,7 +517,7 @@ internal sealed class ExcelRow : IRow
sheetData.Append(row);
}
private int GetOrCreateNumberFormatId(NumberFormatPattern format)
int GetOrCreateNumberFormatId(NumberFormatPattern format)
{
// Создаём стиль, содержащий только числовой формат, и возвращаем его индекс
return _writer.GetOrCreateCellFormatId(numberFormat: format);

View File

@@ -6,8 +6,8 @@
/// </summary>
internal sealed class ExcelRun : IRun
{
private string _text = string.Empty;
private RunFormat? _format;
string _text = string.Empty;
RunFormat? _format;
/// <inheritdoc />
public string Text

View File

@@ -77,7 +77,7 @@ internal sealed class ExcelSheet : ISheet
public ISheet Cell(uint row, uint col, string formula, NumberFormatPattern? format = null)
{
Cell(row, col).Set(formula, format); return this;
Cell(row, col).SetFormula(formula, format); return this;
}
public ISheet Cell(uint row, uint col, DateTime value, NumberFormatPattern? format = null)
@@ -129,7 +129,7 @@ internal sealed class ExcelSheet : ISheet
public ISheet Cell(uint row, string col, string formula, NumberFormatPattern? format = null)
{
Cell(row, col).Set(formula, format); return this;
Cell(row, col).SetFormula(formula, format); return this;
}
public ISheet Cell(uint row, string col, DateTime value, NumberFormatPattern? format = null)
@@ -212,20 +212,63 @@ internal sealed class ExcelSheet : ISheet
/// <inheritdoc />
public IRange RangeByIndexes(uint startRow, uint startCol, uint endRow, uint endCol)
{
return new ExcelRange(Book, this, startRow, startCol, endRow, endCol);
if (startRow == 0) throw new ArgumentException("startRow must be >= 1", nameof(startRow));
if (startCol == 0) throw new ArgumentException("startCol must be >= 1", nameof(startCol));
if (endRow == 0) throw new ArgumentException("endRow must be >= 1", nameof(endRow));
if (endCol == 0) throw new ArgumentException("endCol must be >= 1", nameof(endCol));
// Приводим к корректному порядку (пользователь мог передать start > end)
uint rowStart = Math.Min(startRow, endRow);
uint rowEnd = Math.Max(startRow, endRow);
uint colStart = Math.Min(startCol, endCol);
uint colEnd = Math.Max(startCol, endCol);
return new ExcelRange(Book, this, rowStart, colStart, rowEnd, colEnd);
}
/// <inheritdoc />
public IRange RangeByIndexes(uint startRow, string startCol, uint endRow, string endCol)
{
if (string.IsNullOrEmpty(startCol)) throw new ArgumentException("startCol cannot be null or empty", nameof(startCol));
if (string.IsNullOrEmpty(endCol)) throw new ArgumentException("endCol cannot be null or empty", nameof(endCol));
uint startColIdx = CellAddressHelper.ColumnLetterToIndex(startCol);
uint endColIdx = CellAddressHelper.ColumnLetterToIndex(endCol);
return new ExcelRange(Book, this, startRow, startColIdx, endRow, endColIdx);
if (startColIdx == 0) throw new ArgumentException($"Invalid column letter: '{startCol}'", nameof(startCol));
if (endColIdx == 0) throw new ArgumentException($"Invalid column letter: '{endCol}'", nameof(endCol));
return RangeByIndexes(startRow, startColIdx, endRow, endColIdx);
}
/// <inheritdoc />
public IRange RangeByLength(uint startRow, uint startCol, uint rows, uint cols)
{
if (startRow == 0) throw new ArgumentException("startRow must be >= 1", nameof(startRow));
if (startCol == 0) throw new ArgumentException("startCol must be >= 1", nameof(startCol));
if (rows == 0) throw new ArgumentException("rows must be > 0", nameof(rows));
if (cols == 0) throw new ArgumentException("cols must be > 0", nameof(cols));
checked
{
uint endRow = startRow + rows - 1;
uint endCol = startCol + cols - 1;
return new ExcelRange(Book, this, startRow, startCol, endRow, endCol);
}
}
/// <inheritdoc />
public IRange RangeByLength(uint startRow, string startCol, uint rows, uint cols)
{
if (string.IsNullOrEmpty(startCol)) throw new ArgumentException("startCol cannot be null or empty", nameof(startCol));
uint startColIdx = CellAddressHelper.ColumnLetterToIndex(startCol);
if (startColIdx == 0) throw new ArgumentException($"Invalid column letter: '{startCol}'", nameof(startCol));
return RangeByLength(startRow, startColIdx, rows, cols);
}
/// <inheritdoc />
public ISheet RangeByIndexes(uint startRow, uint startCol, uint endRow, uint endCol, Action<IRange> edit)
{
if (edit is null) throw new ArgumentNullException(nameof(edit));
var range = RangeByIndexes(startRow, startCol, endRow, endCol);
edit(range);
return this;
@@ -234,31 +277,16 @@ internal sealed class ExcelSheet : ISheet
/// <inheritdoc />
public ISheet RangeByIndexes(uint startRow, string startCol, uint endRow, string endCol, Action<IRange> edit)
{
if (edit is null) throw new ArgumentNullException(nameof(edit));
var range = RangeByIndexes(startRow, startCol, endRow, endCol);
edit(range);
return this;
}
/// <inheritdoc />
public IRange RangeByLength(uint startRow, uint startCol, uint rows, uint cols)
{
if (rows == 0 || cols == 0)
throw new ArgumentException("Rows and columns must be greater than 0");
uint endRow = startRow + rows - 1;
uint endCol = startCol + cols - 1;
return new ExcelRange(Book, this, startRow, startCol, endRow, endCol);
}
/// <inheritdoc />
public IRange RangeByLength(uint startRow, string startCol, uint rows, uint cols)
{
uint startColIdx = CellAddressHelper.ColumnLetterToIndex(startCol);
return RangeByLength(startRow, startColIdx, rows, cols);
}
/// <inheritdoc />
public ISheet RangeByLength(uint startRow, uint startCol, uint rows, uint cols, Action<IRange> edit)
{
if (edit is null) throw new ArgumentNullException(nameof(edit));
var range = RangeByLength(startRow, startCol, rows, cols);
edit(range);
return this;
@@ -267,6 +295,7 @@ internal sealed class ExcelSheet : ISheet
/// <inheritdoc />
public ISheet RangeByLength(uint startRow, string startCol, uint rows, uint cols, Action<IRange> edit)
{
if (edit is null) throw new ArgumentNullException(nameof(edit));
var range = RangeByLength(startRow, startCol, rows, cols);
edit(range);
return this;
@@ -274,4 +303,27 @@ internal sealed class ExcelSheet : ISheet
#endregion
#region Merge Operations
/// <inheritdoc />
public bool TryMergeByIndexes(uint startRow, uint startCol, uint endRow, uint endCol) =>
RangeByIndexes(startRow, startCol, endRow, endCol).TryMerge();
/// <inheritdoc />
public bool TryMergeByIndexes(uint startRow, string startCol, uint endRow, string endCol) =>
RangeByIndexes(startRow, startCol, endRow, endCol).TryMerge();
/// <inheritdoc />
public bool TryMergeByLength(uint startRow, uint startCol, uint rows, uint cols) =>
RangeByLength(startRow, startCol, rows, cols).TryMerge();
/// <inheritdoc />
public bool TryMergeByLength(uint startRow, string startCol, uint rows, uint cols) =>
RangeByLength(startRow, startCol, rows, cols).TryMerge();
#endregion
}

View File

@@ -99,6 +99,18 @@ public interface ISheet
/// <summary>Редактирует диапазон, заданный начальной ячейкой и размером (буква столбца).</summary>
ISheet RangeByLength(uint startRow, string startCol, uint rows, uint cols, Action<IRange> edit);
/// <summary>Возвращает диапазон ячеек по начальным и конечным индексам строк и столбцов.</summary>
bool TryMergeByIndexes(uint startRow, uint startCol, uint endRow, uint endCol);
/// <summary>Возвращает диапазон ячеек по начальным и конечным координатам с буквенным обозначением столбцов.</summary>
bool TryMergeByIndexes(uint startRow, string startCol, uint endRow, string endCol);
/// <summary>Возвращает диапазон, начиная с указанной ячейки, заданной размером (строки x столбцы).</summary>
bool TryMergeByLength(uint startRow, uint startCol, uint rows, uint cols);
/// <summary>Возвращает диапазон по начальной ячейке и размеру с буквенным обозначением столбца.</summary>
bool TryMergeByLength(uint startRow, string startCol, uint rows, uint cols);
/// <summary>Возвращает ячейку по номеру строки и столбца (оба начиная с 1).</summary>
ICell Cell(uint row, uint col);
@@ -207,7 +219,22 @@ public interface IRow
RowHeight Height { get; set; }
/// <summary>Устанавливает числовой формат для всех ячеек строки.</summary>
IRow SetNumberFormat(NumberFormatPattern format);
IRow Set(NumberFormatPattern format);
/// <summary>Устанавливает выравнивание для всех ячеек диапазона.</summary>
IRow Set(CellAlign format);
/// <summary>Устанавливает границы для всех ячеек диапазона.</summary>
IRow Set(CellBorder format);
/// <summary>Устанавливает заливку для всех ячеек диапазона.</summary>
IRow Set(CellFill format);
/// <summary>Устанавливает шрифт для всех ячеек диапазона.</summary>
IRow Set(CellFont format);
/// <summary>Устанавливает шрифт ячейки.</summary>
IRow Set(CellStyle font);
/// <summary>Возвращает ячейку в заданном столбце (индекс с 1).</summary>
ICell Cell(uint col);
@@ -334,7 +361,22 @@ public interface IColumn
ColumnWidth Width { get; set; }
/// <summary>Устанавливает числовой формат для всех ячеек столбца.</summary>
IColumn SetNumberFormat(NumberFormatPattern format);
IColumn Set(NumberFormatPattern format);
/// <summary>Устанавливает выравнивание для всех ячеек диапазона.</summary>
IColumn Set(CellAlign format);
/// <summary>Устанавливает границы для всех ячеек диапазона.</summary>
IColumn Set(CellBorder format);
/// <summary>Устанавливает заливку для всех ячеек диапазона.</summary>
IColumn Set(CellFill format);
/// <summary>Устанавливает шрифт для всех ячеек диапазона.</summary>
IColumn Set(CellFont format);
/// <summary>Устанавливает шрифт ячейки.</summary>
IColumn Set(CellStyle font);
/// <summary>Возвращает ячейку в заданной строке (индекс с 1).</summary>
ICell Cell(uint row);
@@ -464,19 +506,22 @@ public interface IRange
IRange MoveTo(uint rowIndex, string colIndex);
/// <summary>Устанавливает числовой формат для всех ячеек диапазона.</summary>
IRange SetNumberFormat(NumberFormatPattern format);
IRange Set(NumberFormatPattern format);
/// <summary>Устанавливает выравнивание для всех ячеек диапазона.</summary>
IRange SetCellAlign(CellAlign format);
IRange Set(CellAlign format);
/// <summary>Устанавливает границы для всех ячеек диапазона.</summary>
IRange SetCellBorder(CellBorder format);
IRange Set(CellBorder format);
/// <summary>Устанавливает заливку для всех ячеек диапазона.</summary>
IRange SetCellFill(CellFill format);
IRange Set(CellFill format);
/// <summary>Устанавливает шрифт для всех ячеек диапазона.</summary>
IRange SetCellFont(CellFont format);
IRange Set(CellFont format);
/// <summary>Устанавливает шрифт ячейки.</summary>
IRange Set(CellStyle font);
/// <summary>Перечисляет все ячейки диапазона (по строкам).</summary>
IEnumerable<ICell> Cells { get; }
@@ -560,6 +605,69 @@ public interface IRange
/// <summary>Представляет одну ячейку на листе.</summary>
public interface ICell
{
/// <summary>Количество фрагментов (Run) в тексте.</summary>
int RunsCount { get; }
/// <summary>Возвращает все фрагменты.</summary>
IEnumerable<IRun> GetRuns();
/// <summary>Возвращает фрагмент по индексу или null.</summary>
IRun? GetRunAt(int index);
/// <summary>Пытается получить фрагмент по индексу.</summary>
bool TryGetRunAt(int index, out IRun run);
/// <summary>Первый фрагмент или null.</summary>
IRun? First();
/// <summary>Пытается получить первый фрагмент.</summary>
bool TryGetFirst(out IRun run);
/// <summary>Последний фрагмент или null.</summary>
IRun? Last();
/// <summary>Пытается получить последний фрагмент.</summary>
bool TryGetLast(out IRun run);
/// <summary>Пытается удалить фрагмент.</summary>
bool TryRemoveRun(IRun run);
/// <summary>Удаляет фрагмент по индексу.</summary>
bool TryRemoveRun(int index);
/// <summary>Удаляет фрагмент по индексу и возвращает удалённый.</summary>
bool TryRemoveRun(int index, out IRun? removed);
/// <summary>Добавляет разрыв строки (перенос внутри ячейки).</summary>
ICell Break();
/// <summary>Добавляет обычный текстовый фрагмент.</summary>
ICell Run(string text, RunFormat? format = null);
/// <summary>Добавляет фрагмент с последующим разрывом строки.</summary>
ICell RunBreak(string text, RunFormat? format = null);
/// <summary>Добавляет подстрочный фрагмент (эквивалентно AddRun с Vertical = Subscript).</summary>
ICell Sub(string text, RunFormat? format = null);
/// <summary>Добавляет надстрочный фрагмент.</summary>
ICell Sup(string text, RunFormat? format = null);
/// <summary>Вставляет фрагмент по индексу.</summary>
bool TryInsertRun(int index, string text, RunFormat? format = null);
/// <summary>Вставляет фрагмент с последующим разрывом строки по индексу.</summary>
bool TryInsertRunBreak(int index, string text, RunFormat? format = null);
/// <summary>Вставляет подстрочный фрагмент по индексу.</summary>
bool TryInsertSub(int index, string text, RunFormat? format = null);
/// <summary>Вставляет надстрочный фрагмент по индексу.</summary>
bool TryInsertSup(int index, string text, RunFormat? format = null);
/// <summary>Применяет заданный формат ко всем существующим фрагментам (поверх их текущего форматирования, заменяя неуказанные свойства).</summary>
void ApplyFormatToAllRuns(RunFormat format);
/// <summary>Проверяет, входит ли ячейка в объединённый диапазон.</summary>
bool IsMerged { get; }
@@ -658,6 +766,9 @@ public interface ICell
/// <summary>Возвращает шрифт ячейки.</summary>
CellFont GetCellFont();
/// <summary>Возвращает шрифт ячейки.</summary>
CellStyle? GetCellStyle();
/// <summary>Пытается извлечь логическое значение.</summary>
bool TryGetBoolean(out bool value);
@@ -679,34 +790,28 @@ public interface ICell
/// <summary>Пытается установить формулу (без вычисленного значения).</summary>
/// <param name="formula">Текст формулы (например, "SUM(A1:A5)").</param>
/// <param name="format">Необязательный числовой формат для результата.</param>
bool TrySet(string formula, NumberFormatPattern? format = null);
bool TrySetFormula(string formula, NumberFormatPattern? format = null);
/// <summary>Устанавливает формулу (выбрасывает исключение при ошибке).</summary>
ICell Set(string formula, NumberFormatPattern? format = null);
ICell SetFormula(string formula, NumberFormatPattern? format = null);
/// <summary>Устанавливает числовой формат ячейки (не меняя значение).</summary>
ICell Set(NumberFormatPattern format);
/// <summary>Устанавливает выравнивание текста ячейки.</summary>
ICell Set(CellAlign format);
ICell Set(CellAlign align);
/// <summary>Устанавливает границы ячейки.</summary>
ICell Set(CellBorder format);
ICell Set(CellBorder border);
/// <summary>Устанавливает заливку ячейки.</summary>
ICell Set(CellFill format);
ICell Set(CellFill fill);
/// <summary>Устанавливает шрифт ячейки.</summary>
ICell Set(CellFont format);
ICell Set(CellFont font);
/// <summary>Устанавливает богатый текст (форматированный) с помощью делегата.</summary>
ICell Text(Action<ICellText> value);
/// <summary>Возвращает объект для редактирования богатого текста ячейки (если ячейка содержит InlineString).</summary>
bool TryText(out ICellText cellText);
/// <summary>Возвращает объект для редактирования богатого текста ячейки (если ячейка содержит InlineString).</summary>
ICellText? Text();
/// <summary>Устанавливает стиль ячейки.</summary>
ICell Set(CellStyle style);
/// <summary>Устанавливает простое текстовое значение (без форматирования).</summary>
ICell Set(string value);
@@ -742,76 +847,6 @@ public interface ICell
void Clear();
}
/// <summary>Представляет богатый текст внутри ячейки (несколько форматированных фрагментов).</summary>
public interface ICellText
{
/// <summary>Количество фрагментов (Run) в тексте.</summary>
int Count { get; }
/// <summary>Возвращает все фрагменты.</summary>
IEnumerable<IRun> GetRuns();
/// <summary>Возвращает фрагмент по индексу или null.</summary>
IRun? GetRunAt(int index);
/// <summary>Пытается получить фрагмент по индексу.</summary>
bool TryGetRunAt(int index, out IRun run);
/// <summary>Первый фрагмент или null.</summary>
IRun? First();
/// <summary>Пытается получить первый фрагмент.</summary>
bool TryGetFirst(out IRun run);
/// <summary>Последний фрагмент или null.</summary>
IRun? Last();
/// <summary>Пытается получить последний фрагмент.</summary>
bool TryGetLast(out IRun run);
/// <summary>Пытается удалить фрагмент.</summary>
bool TryRemoveRun(IRun run);
/// <summary>Удаляет фрагмент по индексу.</summary>
bool TryRemoveRun(int index);
/// <summary>Удаляет фрагмент по индексу и возвращает удалённый.</summary>
bool TryRemoveRun(int index, out IRun? removed);
/// <summary>Добавляет разрыв строки (перенос внутри ячейки).</summary>
ICellText AddBreak();
/// <summary>Добавляет обычный текстовый фрагмент.</summary>
ICellText AddRun(string text, RunFormat? format = null);
/// <summary>Добавляет фрагмент с последующим разрывом строки.</summary>
ICellText AddRunBreak(string text, RunFormat? format = null);
/// <summary>Добавляет подстрочный фрагмент (эквивалентно AddRun с Vertical = Subscript).</summary>
ICellText AddSubRun(string text, RunFormat? format = null);
/// <summary>Добавляет надстрочный фрагмент.</summary>
ICellText AddSupRun(string text, RunFormat? format = null);
/// <summary>Вставляет фрагмент по индексу.</summary>
bool TryInsertRun(int index, string text, RunFormat? format = null);
/// <summary>Вставляет фрагмент с последующим разрывом строки по индексу.</summary>
bool TryInsertRunBreak(int index, string text, RunFormat? format = null);
/// <summary>Вставляет подстрочный фрагмент по индексу.</summary>
bool TryInsertSubRun(int index, string text, RunFormat? format = null);
/// <summary>Вставляет надстрочный фрагмент по индексу.</summary>
bool TryInsertSupRun(int index, string text, RunFormat? format = null);
/// <summary>Применяет заданный формат ко всем существующим фрагментам (поверх их текущего форматирования, заменяя неуказанные свойства).</summary>
void ApplyFormatToAllRuns(RunFormat format);
/// <summary>Удаляет все фрагменты, очищая текст ячейки.</summary>
void Clear();
}
/// <summary>Представляет один форматированный фрагмент текста внутри ячейки.</summary>
public interface IRun
{

View File

@@ -3,7 +3,7 @@
public class NumberFormatPattern
{
public string Format { get; }
internal int? Id { get; private set; }
internal int? Id { get; set; }
public NumberFormatPattern(string format, ushort id = 0)
{

View File

@@ -0,0 +1,149 @@
namespace QWERTYkez.ExcelProcessor;
/// <summary>
/// Методы расширения для установки границ диапазона.
/// </summary>
public static class RangeBorderExtensions
{
/// <summary>
/// Устанавливает границы для диапазона с указанным стилем и цветом.
/// </summary>
/// <param name="range">Диапазон ячеек.</param>
/// <param name="style">Стиль линии границы.</param>
/// <param name="color">Цвет границы (необязательно).</param>
/// <param name="target">Какие границы применять (по умолчанию Outside).</param>
/// <returns>Тот же диапазон для цепочки вызовов.</returns>
public static IRange Set(this IRange range, BorderStyle style, BorderTarget target = BorderTarget.Outside,
System.Drawing.Color? color = null) => range.Set(new() { Style = style, Color = color ?? System.Drawing.Color.Black }, target);
/// <summary>
/// Устанавливает границы для диапазона с указанными параметрами.
/// </summary>
/// <param name="range">Диапазон ячеек.</param>
/// <param name="borderSide">Стиль и цвет границы.</param>
/// <param name="target">Какие границы применять (по умолчанию Outside).</param>
/// <returns>Тот же диапазон для цепочки вызовов.</returns>
public static IRange Set(this IRange range, BorderSide borderSide, BorderTarget target = BorderTarget.Outside)
{
if (range is null) throw new ArgumentNullException(nameof(range));
if (range is not ExcelRange excelRange)
throw new ArgumentException("Range must be of type ExcelRange", nameof(range));
var writer = excelRange._writer;
var sheet = excelRange._sheet;
uint rowStart = range.RowStart;
uint rowEnd = range.RowEnd;
uint colStart = range.ColStart;
uint colEnd = range.ColEnd;
writer.ThrowIfDisposed();
lock (writer._syncLock)
{
// Определяем, какие стороны нужны для каждой ячейки
for (uint r = rowStart; r <= rowEnd; r++)
{
for (uint c = colStart; c <= colEnd; c++)
{
var cell = new ExcelCell(writer, sheet, r, c);
var newBorder = new CellBorder();
bool isTopRow = (r == rowStart);
bool isBottomRow = (r == rowEnd);
bool isLeftCol = (c == colStart);
bool isRightCol = (c == colEnd);
// В зависимости от target, определяем, какие стороны устанавливать
if (target == BorderTarget.All || target == BorderTarget.Outside)
{
if (isTopRow) newBorder = newBorder with { TopBorder = borderSide };
if (isBottomRow) newBorder = newBorder with { BottomBorder = borderSide };
if (isLeftCol) newBorder = newBorder with { LeftBorder = borderSide };
if (isRightCol) newBorder = newBorder with { RightBorder = borderSide };
}
if (target == BorderTarget.All || target == BorderTarget.Inside)
{
// Внутренние границы: для каждой ячейки устанавливаем правую и нижнюю,
// если есть сосед справа/снизу внутри диапазона
if (c < colEnd) newBorder = newBorder with { RightBorder = borderSide };
if (r < rowEnd) newBorder = newBorder with { BottomBorder = borderSide };
}
// Отдельные стороны
if (target == BorderTarget.Top && isTopRow)
newBorder = newBorder with { TopBorder = borderSide };
if (target == BorderTarget.Bottom && isBottomRow)
newBorder = newBorder with { BottomBorder = borderSide };
if (target == BorderTarget.Left && isLeftCol)
newBorder = newBorder with { LeftBorder = borderSide };
if (target == BorderTarget.Right && isRightCol)
newBorder = newBorder with { RightBorder = borderSide };
// Применяем границу к ячейке (изолированно, без каскада)
cell.SetBorderIsolate(newBorder);
// Если ячейка находится на внешней границе диапазона, очищаем соответствующую сторону у соседа вне диапазона
if (isTopRow && (target == BorderTarget.Outside || target == BorderTarget.All || target == BorderTarget.Top))
ClearNeighborBorder(cell, -1, 0, b => b with { BottomBorder = null });
if (isBottomRow && (target == BorderTarget.Outside || target == BorderTarget.All || target == BorderTarget.Bottom))
ClearNeighborBorder(cell, 1, 0, b => b with { TopBorder = null });
if (isLeftCol && (target == BorderTarget.Outside || target == BorderTarget.All || target == BorderTarget.Left))
ClearNeighborBorder(cell, 0, -1, b => b with { RightBorder = null });
if (isRightCol && (target == BorderTarget.Outside || target == BorderTarget.All || target == BorderTarget.Right))
ClearNeighborBorder(cell, 0, 1, b => b with { LeftBorder = null });
}
}
}
return range;
}
/// <summary>
/// Очищает все границы в диапазоне (сбрасывает до None).
/// </summary>
public static IRange ClearBorders(this IRange range)
{
if (range is null) throw new ArgumentNullException(nameof(range));
if (range is not ExcelRange excelRange)
throw new ArgumentException("Range must be of type ExcelRange", nameof(range));
var writer = excelRange._writer;
var sheet = excelRange._sheet;
uint rowStart = range.RowStart;
uint rowEnd = range.RowEnd;
uint colStart = range.ColStart;
uint colEnd = range.ColEnd;
writer.ThrowIfDisposed();
lock (writer._syncLock)
{
for (uint r = rowStart; r <= rowEnd; r++)
{
for (uint c = colStart; c <= colEnd; c++)
{
var cell = new ExcelCell(writer, sheet, r, c);
// Удаляем все границы
cell.SetBorderIsolate(new CellBorder()); // пустая граница
}
}
}
return range;
}
// Вспомогательный метод для очистки конкретной стороны у соседа
static void ClearNeighborBorder(ExcelCell cell, int rowOffset, int colOffset, Func<CellBorder, CellBorder> clearFunc)
{
var neighbor = cell.GetNeighbor(rowOffset, colOffset);
if (neighbor is null)
return;
var neighborBorder = neighbor.GetCellBorder();
var newBorder = clearFunc(neighborBorder);
if (!neighborBorder.Equals(newBorder))
neighbor.SetBorderIsolate(newBorder);
}
}

View File

@@ -5,13 +5,13 @@
/// </summary>
public readonly struct RowHeight(double points)
{
private const double POINTS_PER_INCH = 72.0;
private const double INCH_PER_CM = 1.0 / 2.54;
private const double POINTS_PER_CM = POINTS_PER_INCH * INCH_PER_CM; // ≈ 28.3464566929
private const double POINTS_PER_MM = POINTS_PER_CM / 10.0; // ≈ 2.83464566929
const double POINTS_PER_INCH = 72.0;
const double INCH_PER_CM = 1.0 / 2.54;
const double POINTS_PER_CM = POINTS_PER_INCH * INCH_PER_CM; // ≈ 28.3464566929
const double POINTS_PER_MM = POINTS_PER_CM / 10.0; // ≈ 2.83464566929
private const double DXA_PER_POINT = 20.0; // 1 point = 20 dxa
private const double POINTS_PER_DXA = 1.0 / DXA_PER_POINT;
const double DXA_PER_POINT = 20.0; // 1 point = 20 dxa
const double POINTS_PER_DXA = 1.0 / DXA_PER_POINT;
/// <summary>Высота в пунктах (points).</summary>
public double Points { get; } = points;

View File

@@ -19,7 +19,7 @@ public readonly struct RunFormat
public bool? IsStrike { get; init; }
/// <summary>Цвет текста фрагмента.</summary>
public ExColor? Color { get; init; }
public System.Drawing.Color? Color { get; init; }
/// <summary>Размер шрифта фрагмента в пунктах.</summary>
public double? FontSize { get; init; }
@@ -80,9 +80,7 @@ public readonly struct RunFormat
// DoubleStrike в Excel не поддерживается, опускаем
Underline = underline,
Color = rPr.GetFirstChild<Color>()?.Rgb is not null
? ExColor.FromRgb(rPr.GetFirstChild<Color>()!.Rgb!.Value!)
: null!,
Color = FromExcelColor(rPr.GetFirstChild<Color>()?.Rgb),
FontSize = rPr.GetFirstChild<FontSize>()?.Val?.Value,
FontFamily = rPr.GetFirstChild<RunFont>()?.Val,
Vertical = vertical
@@ -90,6 +88,45 @@ public readonly struct RunFormat
return fmt;
}
static System.Drawing.Color? FromExcelColor(string? rgb)
{
if (string.IsNullOrWhiteSpace(rgb)) return null;
if (rgb!.Length == 6)
{
byte r = Convert.ToByte(rgb.Substring(0, 2), 16);
byte g = Convert.ToByte(rgb.Substring(2, 2), 16);
byte b = Convert.ToByte(rgb.Substring(4, 2), 16);
return System.Drawing.Color.FromArgb(r, g, b);
}
else if (rgb.Length == 8)
{
byte a = Convert.ToByte(rgb.Substring(0, 2), 16);
byte r = Convert.ToByte(rgb.Substring(2, 2), 16);
byte g = Convert.ToByte(rgb.Substring(4, 2), 16);
byte b = Convert.ToByte(rgb.Substring(6, 2), 16);
return System.Drawing.Color.FromArgb(a, r, g, b);
}
else return null;
}
public bool TryGetExcelColor(out Color excelColor)
{
if (Color is { } c)
{
excelColor = new Color() { Rgb = $"{c.R:X2}{c.G:X2}{c.B:X2}" };
return true;
}
else
{
excelColor = null!;
return false;
}
}
/*
методы для извлечения OpenXmlElement или других более удобных типов

View File

@@ -8,9 +8,56 @@ namespace QWERTYkez.ExcelProcessor;
/// </summary>
internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
{
readonly Dictionary<CellStyle, int> _styleCache = [];
internal int GetOrCreateStyleId(CellStyle style)
{
lock (_syncLock)
{
if (_styleCache.TryGetValue(style, out int id))
return id;
// Создаём CellFormat через существующий метод GetOrCreateCellFormatId
int newId = GetOrCreateCellFormatId(
style.NumberFormat,
style.Font,
style.Fill,
style.Border,
style.Align
);
_styleCache[style] = newId;
return newId;
}
}
internal CellStyle? GetCellStyle(uint styleIndex)
{
if (styleIndex == 0) return null;
var align = GetCellAlign(styleIndex);
var font = GetCellFont(styleIndex);
var fill = GetCellFill(styleIndex);
var border = GetCellBorder(styleIndex);
var numberFormat = GetNumberFormat(styleIndex);
bool hasAny = !align.Equals(default) || !font.Equals(default) || !fill.Equals(default) ||
!border.Equals(default) || numberFormat != null;
if (!hasAny) return null;
return new CellStyle
{
Align = align.Equals(default) ? null : align,
Font = font.Equals(default) ? null : font,
Fill = fill.Equals(default) ? null : fill,
Border = border.Equals(default) ? null : border,
NumberFormat = numberFormat
};
}
// Работа с общей таблицей строк
private SharedStringTablePart? _sharedStringPart;
private SharedStringTable? _sharedStringTable;
SharedStringTablePart? _sharedStringPart;
SharedStringTable? _sharedStringTable;
internal static Dictionary<int, double>? _calibrationTable; // cw -> width_pts
@@ -62,7 +109,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return true;
}
private Dictionary<int, double> CalibrateWidthCoeffUsingInterop()
Dictionary<int, double> CalibrateWidthCoeffUsingInterop()
{
object? excelApp = null;
object? workbooks = null;
@@ -133,7 +180,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
}
private void EnsureSharedStringTable()
void EnsureSharedStringTable()
{
if (_sharedStringPart != null) return;
_sharedStringPart = _doc.WorkbookPart?.GetPartsOfType<SharedStringTablePart>().FirstOrDefault();
@@ -193,17 +240,17 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
}
// Кэши числовых форматов
private readonly Dictionary<string, NumberFormatPattern> _numberFormatCache = [];
private readonly Dictionary<uint, NumberFormatPattern> _numberFormatIdToPattern = [];
readonly Dictionary<string, NumberFormatPattern> _numberFormatCache = [];
readonly Dictionary<uint, NumberFormatPattern> _numberFormatIdToPattern = [];
// Кэши для компонентов стилей (чтобы не создавать дубликаты)
private readonly Dictionary<CellFont, int> _fontCache = [];
private readonly Dictionary<CellFill, int> _fillCache = [];
private readonly Dictionary<CellBorder, int> _borderCache = [];
private readonly Dictionary<CellAlign, int> _alignmentCache = [];
readonly Dictionary<CellFont, int> _fontCache = [];
readonly Dictionary<CellFill, int> _fillCache = [];
readonly Dictionary<CellBorder, int> _borderCache = [];
readonly Dictionary<CellAlign, int> _alignmentCache = [];
// Кэш составных стилей (CellFormat)
private readonly Dictionary<(int fontId, int fillId, int borderId, int alignId, int numFmtId), int> _cellFormatCache = [];
readonly Dictionary<(int fontId, int fillId, int borderId, int alignId, int numFmtId), int> _cellFormatCache = [];
// Конструктор, фабричные методы без изменений (опущены)
@@ -294,10 +341,8 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
int numFmtId = -1;
if (numberFormat != null)
{
if (numberFormat.Id.HasValue && numberFormat.Id.Value < 164)
numFmtId = numberFormat.Id.Value;
else
numFmtId = (int)GetOrCreateNumberFormatId(numberFormat);
numFmtId = numberFormat.Id.HasValue && numberFormat.Id.Value < 164
? numberFormat.Id.Value : (int)GetOrCreateNumberFormatId(numberFormat);
}
// Получаем или создаём Font, Fill, Border, Alignment
@@ -352,7 +397,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
#region Вспомогательные методы для работы со стилями
private Stylesheet EnsureStylesheet()
Stylesheet EnsureStylesheet()
{
var workbookPart = _doc.WorkbookPart ?? throw new InvalidOperationException("No WorkbookPart");
var stylesPart = workbookPart.WorkbookStylesPart;
@@ -390,7 +435,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return stylesheet;
}
private uint GetOrCreateNumberFormatId(NumberFormatPattern pattern)
uint GetOrCreateNumberFormatId(NumberFormatPattern pattern)
{
if (pattern.Id.HasValue)
return (uint)pattern.Id.Value;
@@ -400,7 +445,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return (uint)created.Id!.Value;
}
private int GetOrCreateFontId(CellFont font)
int GetOrCreateFontId(CellFont font)
{
if (_fontCache.TryGetValue(font, out int id))
return id;
@@ -414,7 +459,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return _fontCache[font] = (int)newId;
}
private int GetOrCreateFillId(CellFill fill)
int GetOrCreateFillId(CellFill fill)
{
if (_fillCache.TryGetValue(fill, out int id))
return id;
@@ -428,7 +473,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return _fillCache[fill] = (int)newId;
}
private int GetOrCreateBorderId(CellBorder border)
int GetOrCreateBorderId(CellBorder border)
{
if (_borderCache.TryGetValue(border, out int id))
return id;
@@ -443,7 +488,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return newId;
}
private int GetOrCreateAlignmentId(CellAlign align)
int GetOrCreateAlignmentId(CellAlign align)
{
if (_alignmentCache.TryGetValue(align, out int id))
return id;
@@ -456,7 +501,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return id;
}
private Alignment GetAlignmentFromCache(int alignId)
Alignment GetAlignmentFromCache(int alignId)
{
foreach (var pair in _alignmentCache)
if (pair.Value == alignId)
@@ -464,7 +509,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
return new Alignment();
}
private CellFormat? GetCellFormatAt(uint index)
CellFormat? GetCellFormatAt(uint index)
{
var stylesheet = EnsureStylesheet();
if (stylesheet.CellFormats == null || index >= stylesheet.CellFormats.Count!.Value)
@@ -473,14 +518,14 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
}
// Вспомогательные создания коллекций
private static NumberingFormats CreateNumberingFormats(Stylesheet stylesheet)
static NumberingFormats CreateNumberingFormats(Stylesheet stylesheet)
{
var nfs = new NumberingFormats();
stylesheet.NumberingFormats = nfs;
return nfs;
}
private static CellFormats CreateCellFormats(Stylesheet stylesheet)
static CellFormats CreateCellFormats(Stylesheet stylesheet)
{
var cfs = new CellFormats();
stylesheet.CellFormats = cfs;
@@ -491,7 +536,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
private bool _isModified = false;
bool _isModified = false;
internal ExcelWriter() { }
@@ -1006,7 +1051,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
#endregion
private void EnsureFullCalculationOnLoad()
void EnsureFullCalculationOnLoad()
{
if (_doc?.WorkbookPart?.Workbook == null) return;
var workbook = _doc.WorkbookPart.Workbook;
@@ -1217,7 +1262,7 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
// Внутренний метод, предполагает, что _syncLock уже захвачен вызывающим
private NumberFormatPattern CreateNumberFormatInternal(string formatCode)
NumberFormatPattern CreateNumberFormatInternal(string formatCode)
{
// Проверяем кэш по коду
if (_numberFormatCache.TryGetValue(formatCode, out var existing))
@@ -1257,36 +1302,42 @@ internal sealed class ExcelWriter : ExcelReader, IExcelReader, IExcelWriter
// Вспомогательные методы для извлечения элементов из стилей (нужно закешировать или обращаться напрямую)
private Border? GetBorderById(uint borderId)
Border? GetBorderById(uint borderId)
{
if (_cachedBorders == null)
var borders = EnsureStylesheet().Borders;
if (borders == null) return null;
int index = 0;
foreach (var border in borders.Elements<Border>())
{
var borders = EnsureStylesheet().Borders;
_cachedBorders = borders?.Elements<Border>().ToList() ?? [];
if (index == borderId) return border;
index++;
}
return borderId < _cachedBorders.Count ? _cachedBorders[(int)borderId] : null;
return null;
}
private List<Border>? _cachedBorders;
private Fill? GetFillById(uint borderId)
Fill? GetFillById(uint fillId)
{
if (_cachedFills == null)
var fills = EnsureStylesheet().Fills;
if (fills == null) return null;
int index = 0;
foreach (var fill in fills.Elements<Fill>())
{
var fills = EnsureStylesheet().Fills;
_cachedFills = fills?.Elements<Fill>().ToList() ?? [];
if (index == fillId) return fill;
index++;
}
return borderId < _cachedFills.Count ? _cachedFills[(int)borderId] : null;
return null;
}
private List<Fill>? _cachedFills;
private Font? GetFontById(uint borderId)
Font? GetFontById(uint fontId)
{
if (_cachedFonts == null)
var fonts = EnsureStylesheet().Fonts;
if (fonts == null) return null;
int index = 0;
foreach (var font in fonts.Elements<Font>())
{
var borders = EnsureStylesheet().Borders;
_cachedFonts = borders?.Elements<Font>().ToList() ?? [];
if (index == fontId) return font;
index++;
}
return borderId < _cachedFonts.Count ? _cachedFonts[(int)borderId] : null;
return null;
}
private List<Font>? _cachedFonts;
}

View File

@@ -10,7 +10,7 @@ namespace QWERTYkez.ExcelProcessor;
/// </summary>
internal class NormalizedSet : ISet<string>
{
private readonly HashSet<string> _inner;
readonly HashSet<string> _inner;
/// <summary>
/// Создаёт пустое нормализованное множество.
@@ -32,7 +32,7 @@ internal class NormalizedSet : ISet<string>
/// <summary>
/// Нормализует строку: верхний регистр и удаление диакритики.
/// </summary>
private static string Normalize(string s)
static string Normalize(string s)
{
if (string.IsNullOrEmpty(s))
return s;

View File

@@ -22,7 +22,7 @@ internal static class PlaceholderFinder
return FindInDocument(doc, [.. doc.WorkbookPart.WorksheetParts]);
}
private static void ProcessWorksheet(WorksheetPart wsPart, SharedStringTable? sharedStrings, ISet<string> result)
static void ProcessWorksheet(WorksheetPart wsPart, SharedStringTable? sharedStrings, ISet<string> result)
{
var worksheet = wsPart.Worksheet;
if (worksheet is null) return;
@@ -31,7 +31,7 @@ internal static class PlaceholderFinder
ProcessHeaderFooter(worksheet, result);
}
private static void ProcessCells(Worksheet worksheet, SharedStringTable? sharedStrings, ISet<string> result)
static void ProcessCells(Worksheet worksheet, SharedStringTable? sharedStrings, ISet<string> result)
{
var sheetData = worksheet.GetFirstChild<SheetData>();
if (sheetData is null) return;
@@ -59,7 +59,7 @@ internal static class PlaceholderFinder
}
}
private static void ProcessHeaderFooter(Worksheet worksheet, ISet<string> result)
static void ProcessHeaderFooter(Worksheet worksheet, ISet<string> result)
{
var hf = worksheet.Descendants<HeaderFooter>().FirstOrDefault();
if (hf is null) return;
@@ -84,7 +84,7 @@ internal static class PlaceholderFinder
}
}
private static string GetCellValue(Cell cell, SharedStringTable? sharedStrings)
static string GetCellValue(Cell cell, SharedStringTable? sharedStrings)
{
if (cell?.CellValue is null && cell?.InlineString is null)
return string.Empty;
@@ -107,7 +107,7 @@ internal static class PlaceholderFinder
return raw;
}
private static unsafe bool FindPlaceholdersInText(string text, ref List<string>? output)
static unsafe bool FindPlaceholdersInText(string text, ref List<string>? output)
{
fixed (char* pText = text)
{

View File

@@ -158,7 +158,7 @@ internal static class ReplaceNumericExtensions
// =========================== ОБЩАЯ ЛОГИКА ===========================
private static void ReplaceNumericCore<T>(
static void ReplaceNumericCore<T>(
WorkbookPart workbookPart,
WorksheetPart[] worksheets,
IEnumerable<KeyValuePair<string, T>> numericReplacements,
@@ -255,7 +255,7 @@ internal static class ReplaceNumericExtensions
UpdateSharedStringTable(workbookPart, allSharedStrings);
}
private static void ReplaceSingleCore<T>(
static void ReplaceSingleCore<T>(
WorkbookPart workbookPart,
WorksheetPart[] worksheets,
string oldValue,
@@ -343,7 +343,7 @@ internal static class ReplaceNumericExtensions
// =========================== ВСПОМОГАТЕЛЬНЫЕ МЕТОДЫ ===========================
private static string GetCellTextForNumeric(Cell cell, List<string> allSharedStrings)
static string GetCellTextForNumeric(Cell cell, List<string> allSharedStrings)
{
if (cell?.CellValue == null) return string.Empty;
@@ -366,7 +366,7 @@ internal static class ReplaceNumericExtensions
return val;
}
private static void SetCellText(Cell cell, string newText,
static void SetCellText(Cell cell, string newText,
List<string> allSharedStrings, Dictionary<string, int> sharedStringIndexMap)
{
if (cell.InlineString != null)
@@ -388,7 +388,7 @@ internal static class ReplaceNumericExtensions
cell.CellValue = new CellValue(index.ToString());
}
private static void UpdateSharedStringTable(WorkbookPart workbookPart, List<string> allSharedStrings)
static void UpdateSharedStringTable(WorkbookPart workbookPart, List<string> allSharedStrings)
{
var ssPart = workbookPart.SharedStringTablePart;
ssPart ??= workbookPart.AddNewPart<SharedStringTablePart>();
@@ -399,7 +399,7 @@ internal static class ReplaceNumericExtensions
sharedStringTable.Save();
}
private static string ConcatTexts(IEnumerable<Text> texts)
static string ConcatTexts(IEnumerable<Text> texts)
{
var sb = new StringBuilder();
foreach (var t in texts)
@@ -408,7 +408,7 @@ internal static class ReplaceNumericExtensions
}
// Оптимизированная замена подстроки через string.Create (без unsafe)
private static unsafe string ReplaceSubstring(string original, int start, int length, string replacement)
static unsafe string ReplaceSubstring(string original, int start, int length, string replacement)
{
if (length == 0) return original;
int newLen = original.Length - length + replacement.Length;
@@ -428,7 +428,7 @@ internal static class ReplaceNumericExtensions
}
}
private static uint CreateNumberFormat(WorkbookPart workbookPart, string format)
static uint CreateNumberFormat(WorkbookPart workbookPart, string format)
{
var stylesPart = workbookPart.WorkbookStylesPart;
if (stylesPart == null)
@@ -457,7 +457,7 @@ internal static class ReplaceNumericExtensions
// =========================== КОЛОНТИТУЛЫ И КОММЕНТАРИИ ===========================
private static void ReplaceInHeadersFooters(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceInHeadersFooters(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
var worksheet = worksheetPart.Worksheet;
if (worksheet is null) return;
@@ -468,7 +468,7 @@ internal static class ReplaceNumericExtensions
ReplaceHeaderFooter(elem, replacementDict, comparisonType);
}
private static void ReplaceHeaderFooter(OpenXmlLeafTextElement? element, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceHeaderFooter(OpenXmlLeafTextElement? element, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
if (element?.Text is null) return;
string original = element.Text;
@@ -477,7 +477,7 @@ internal static class ReplaceNumericExtensions
element.Text = processed;
}
private static void ReplaceInComments(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceInComments(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
var commentsPart = worksheetPart.WorksheetCommentsPart;
if (commentsPart?.Comments is null) return;
@@ -493,7 +493,7 @@ internal static class ReplaceNumericExtensions
}
}
private static string ProcessReplacements(string input, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static string ProcessReplacements(string input, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
if (string.IsNullOrEmpty(input) || replacementDict.Count == 0) return input;
string result = input;
@@ -505,7 +505,7 @@ internal static class ReplaceNumericExtensions
return result;
}
private static string ReplaceInString(string original, string oldValue, string newValue, StringComparison comparisonType)
static string ReplaceInString(string original, string oldValue, string newValue, StringComparison comparisonType)
{
int idx = original.IndexOf(oldValue, comparisonType);
if (idx < 0) return original;

View File

@@ -30,7 +30,7 @@ internal static class ReplaceStringExtensions
}
// --- Общий приватный метод, содержащий всю логику замены ---
private static void ReplaceCore(SpreadsheetDocument doc, WorksheetPart[] worksheets, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceCore(SpreadsheetDocument doc, WorksheetPart[] worksheets, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
var workbookPart = doc.WorkbookPart!;
@@ -63,7 +63,7 @@ internal static class ReplaceStringExtensions
}
// --- Остальные вспомогательные методы (без изменений) ---
private static IEqualityComparer<string> GetComparerForStringComparison(StringComparison comparisonType) =>
static IEqualityComparer<string> GetComparerForStringComparison(StringComparison comparisonType) =>
comparisonType switch
{
StringComparison.Ordinal => StringComparer.Ordinal,
@@ -75,7 +75,7 @@ internal static class ReplaceStringExtensions
_ => StringComparer.OrdinalIgnoreCase,
};
private static void CollectCellChanges(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType, Dictionary<Cell, string> cellChanges, List<string> allSharedStrings)
static void CollectCellChanges(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType, Dictionary<Cell, string> cellChanges, List<string> allSharedStrings)
{
var worksheet = worksheetPart.Worksheet;
if (worksheet is null) return;
@@ -93,7 +93,7 @@ internal static class ReplaceStringExtensions
}
}
private static void ApplyCellChanges(Dictionary<Cell, string> cellChanges, List<string> allSharedStrings)
static void ApplyCellChanges(Dictionary<Cell, string> cellChanges, List<string> allSharedStrings)
{
foreach (var kvp in cellChanges)
{
@@ -114,7 +114,7 @@ internal static class ReplaceStringExtensions
}
}
private static void UpdateSharedStringTable(WorkbookPart workbookPart, List<string> allSharedStrings)
static void UpdateSharedStringTable(WorkbookPart workbookPart, List<string> allSharedStrings)
{
var ssPart = workbookPart.SharedStringTablePart;
ssPart ??= workbookPart.AddNewPart<SharedStringTablePart>();
@@ -125,7 +125,7 @@ internal static class ReplaceStringExtensions
sharedStringTable.Save();
}
private static string ProcessReplacements(string input, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static string ProcessReplacements(string input, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
if (string.IsNullOrEmpty(input) || replacementDict.Count == 0) return input;
string result = input;
@@ -137,7 +137,7 @@ internal static class ReplaceStringExtensions
return result;
}
private static string ReplaceInString(string original, string oldValue, string newValue, StringComparison comparisonType)
static string ReplaceInString(string original, string oldValue, string newValue, StringComparison comparisonType)
{
int idx = original.IndexOf(oldValue, comparisonType);
if (idx < 0) return original;
@@ -154,7 +154,7 @@ internal static class ReplaceStringExtensions
return sb.ToString();
}
private static string GetCellText(Cell cell, List<string> allSharedStrings)
static string GetCellText(Cell cell, List<string> allSharedStrings)
{
if (cell?.CellValue is null) return string.Empty;
if (cell.InlineString is not null)
@@ -169,7 +169,7 @@ internal static class ReplaceStringExtensions
return value;
}
private static int AddOrFindStringIndex(List<string> allSharedStrings, string text)
static int AddOrFindStringIndex(List<string> allSharedStrings, string text)
{
int idx = allSharedStrings.IndexOf(text);
if (idx >= 0) return idx;
@@ -178,7 +178,7 @@ internal static class ReplaceStringExtensions
}
// --- Колонтитулы (обобщённый метод) ---
private static void ReplaceInHeadersFooters(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceInHeadersFooters(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
var worksheet = worksheetPart.Worksheet;
if (worksheet is null) return;
@@ -189,7 +189,7 @@ internal static class ReplaceStringExtensions
ReplaceHeaderFooter(elem, replacementDict, comparisonType);
}
private static void ReplaceHeaderFooter(OpenXmlLeafTextElement? element, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceHeaderFooter(OpenXmlLeafTextElement? element, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
if (element?.Text is null) return;
string original = element.Text;
@@ -199,7 +199,7 @@ internal static class ReplaceStringExtensions
}
// --- Комментарии ---
private static void ReplaceInComments(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
static void ReplaceInComments(WorksheetPart worksheetPart, Dictionary<string, string> replacementDict, StringComparison comparisonType)
{
var commentsPart = worksheetPart.WorksheetCommentsPart;
if (commentsPart?.Comments is null) return;