Browse our Products

Aspose.Words for C++ 21.6 Release Notes

Major Features

We have added the following features from Aspose.Words for .NET on this regular monthly release:

  • Implemented an ability to set fill and stroke formatting for chart series, data points and markers.
  • Introduced a new API for working with textures.
  • Implemented rendering of OOXML Ink (InkML subset).
  • TxtSaveOptions.MaxCharactersPerLine property was introduced.
  • A new Document.LayoutOption is added to control page numbering in a continuous section.

Limitations and API Differences

Aspose.Words for C++ has some differences as compared to its equivalent .NET version of the API. This section contains information about all such functionality that is not available in the current release. The missing features will be added in future releases.

  • The current release does not support Metered license.
  • The current release does not support LINQ and Reporting features.
  • The current release does not support OpenGL 3D Shapes rendering.
  • The current release does not support loading PDF documents.
  • The current release has limited support for database features - C++ doesn’t have common API for DB like .NET System.Data.
  • The current release supports Microsoft Visual C++ version 2017 or higher.
  • The current release supports GCC 6.3 or higher and Clang 3.9.1 or higher on Linux and only for the x86_x64 platform.

Full List of Issues Covering all Changes in this Release

KeySummaryCategory
WORDSNET-21647DOCX to PDF: Content displacement occursNew Feature
WORDSNET-12748Provide API to change Color of Category Series of ChartNew Feature
WORDSNET-13907Support rendering DrawingML InkML ContentPartNew Feature
WORDSNET-12275Add feature to change the color of ChartSeriesNew Feature
WORDSNET-21847Advanced OpenType font features not appliedNew Feature
WORDSNET-21871Add feature to create Scatter chart with linesNew Feature
WORDSNET-12529Add feature to get/set Asian Character Spacing propertiesNew Feature
WORDSNET-22082storeItemChecksum get/set methodNew Feature
WORDSNET-22002Implement Fill.PresetTextured() methodNew Feature
WORDSNET-20023Allow to change Pie chart colorsNew Feature
WORDSNET-15201Implement MS Word 2013 behavior for continuous sections with footnotesNew Feature
WORDSNET-17510Aspose.Words does not mimic MS Word for document structure tagsEnhancement
WORDSNET-18186Ink annotations lost during conversion from Word to PDFEnhancement
WORDSNET-20020Transparency in converted Word Document is not shown in PDFEnhancement
WORDSNET-12640Add feature to set/get Chart series colorsEnhancement
WORDSNET-20462Handwritten contents are lost after DOCX to PDFEnhancement
WORDSNET-21156Incorrect value returned by LayoutCollectorEnhancement
WORDSNET-19199DOCX to PDF with free hand drawing imagesEnhancement
WORDSNET-22090Each Line in TXT File should have a Fix Length of Characters | Maximum Allowed Characters in LineEnhancement
WORDSNET-21795DOCX to PDF/A conversion and validation fails: Problem with word/spaces separation inside headerEnhancement
WORDSNET-15771Table overflows to next page while converting to PDFBug
WORDSNET-22155DOCX to PDF/A conversion: significant tagging overheadBug
WORDSNET-18780Table is rendered on the next page in output PDFBug
WORDSNET-22158MailMerge CleanupOptions does not remove the empty table rowsBug
WORDSNET-14523After conversion to PDF, content is missingBug
WORDSNET-22171Missing SVG Shape in PDFBug
WORDSNET-16700PrinterMetrics does not work on Xamarin/NetStandardBug
WORDSNET-18091UpdateFields method throws System.InvalidOperationExceptionBug
WORDSNET-22186Text of Frame is overlapped after DOCX to PDF conversionBug
WORDSNET-21845DOCX to PDF > Incorrect rendering of footnotesBug
WORDSNET-21848Text in images gets cut off after DOCX to PDF converisonBug
WORDSNET-22189Incorrect list numbering when converting from DOCX to HTMLBug
WORDSNET-18945System.InvalidOperationException is thrown while saving DOCX to PDFBug
WORDSNET-18275Document.UpdatePageLayout throws System.InvalidCastExceptionBug
WORDSNET-22193Hidden text is rendered after DOCX to PDF conversionBug
WORDSNET-18618Document.UpdatePageLayout throws System.InvalidOperationExceptionBug
WORDSNET-20733Incorrect rendering of Footnotes ContentBug
WORDSNET-17613Content pushed down to subsequent pages in PDF causing overlapping in FootnotesBug
WORDSNET-22201System.InvalidOperationException is thrown after removing stylesBug
WORDSNET-22166Footnotes break continuous section break rendering (rendered as page break)Bug
WORDSNET-21332Overlapping - Embedded Table contents from 2nd & 3rd pages move to 1st page in TIFFBug
WORDSNET-21557Text is overlapped after DOCX to PDF conversionBug
WORDSNET-22207Bullet points order issue during Word to PDF conversionBug
WORDSNET-22209NullReferenceException occurs upon DOC to PDF conversionBug
WORDSNET-22208NullReferenceException occurs upon DOCX to PDF conversionBug
WORDSNET-22213Document.PageCount throws System.IndexOutOfRangeExceptionBug
WORDSNET-22219Incorrect Image resolution and dimension units while importing MarkdownBug
WORDSNET-20315A Legend Entry in Chart is missing when Converting Chart to JPEGBug
WORDSNET-19458DOCX to PDF conversion issue with document structureBug
WORDSNET-22224Hidden footnotes throws asserts during reflowBug
WORDSNET-22185Mail Merge field is lost after re-saving DOCXBug
WORDSNET-18562DOCX to PDF conversion issue with Chart’s Data LabelBug
WORDSNET-21707Font name of Hebrew text is changed to ‘Times New Roman’ after re-saving DOCXBug
WORDSNET-20920StructuredDocumentTag content is changed on page extraction.Bug
WORDSNET-15553Aspose.Words generated PDFA size issueBug
WORDSNET-20943Incorrect table positionBug
WORDSNET-21730DOCX to PDF conversion issue with ExportDocumentStructure (incorrect tags)Bug
WORDSNET-22232Aspose.Words does not mimic MS Word for document structure tagsBug
WORDSNET-22177Wrong Page Number when word converted to pdfBug
WORDSNET-22235Image is rendered as cross after inserting HTML into document at LinuxBug
WORDSNET-21993DOCX to PDF conversion issue - PDF is not editableBug
WORDSNET-21609DOCX->HTML->DOCX conversion: table size changes in the outputBug
WORDSNET-22241Checked symbol of content control is changed after first clickBug
WORDSNET-22244Process hangs when calling UpdateFields methodBug
WORDSNET-21701NullReferenceException is thrown when call UpdatePageLayoutBug
WORDSNET-16663Problem with saving DOC with footnote to PDFBug
WORDSNET-21896FileCorruptedException occurs when loading a MHTML documentBug
WORDSNET-22195Aspose.Words UpdateLayout method throws exception “Infinite loop”Bug
WORDSNET-22253Hyperlink is lost after re-saving DOCXBug
WORDSNET-21352ExportDocumentStructure creates incorrect tags in output PDFBug
WORDSNET-21937System.NullReferenceException is thrown when DOCX is saved to PDFBug
WORDSNET-22257AppendContent throws exceptionBug
WORDSNET-20631ExportDocumentStructure and NVDA applicationBug
WORDSNET-14495Alignment of OfficeMath is incorrect when it is inserted by InsertHtmlBug
WORDSNET-22103Merging two documents and list numbering doesn’t continueBug
WORDSNET-22276Incorrect rendering “Rectangle:Rounded Corners” shapeBug
WORDSNET-22174DOCX to ODT conversion throws System.InvalidOperationExceptionBug
WORDSNET-22259“Value was either too large or too small for an Int32.” exception when open DOCX fileBug
WORDSNET-21968System.IndexOutOfRangeException occurs upon Word XML Document to PDF conversionBug
WORDSNET-21969DOCX to PDF conversion | Even numbers are skipped from x-axis of chartBug
WORDSNET-21975Heading is not properly rendered when exported to PDFBug
WORDSNET-22109System.NullReferenceException occurs upon DOCX to PDF conversionBug
WORDSNET-22113Text font is changed after DOCX to PDF conversionBug
WORDSNET-22129DocumentBuilder.InsertFootnote causes System.IndexOutOfRangeException when saving to PDFBug
WORDSNET-21731DOCX to PDF conversion issue with ExportDocumentStructure (incorrect list)Bug
WORDSNET-22094Document.ExtractPages method removes BookmarksBug
WORDSNET-22149Paragraph become list item after inserting one document into anotherBug
WORDSNET-22286InvalidOperationException: Unexpected ST_TrueFalse value occurs upon loading a DOCXBug
WORDSNET-21622System.InvalidOperationException occurs upon DOCX to HTML conversionBug
WORDSNET-22292Update storeItemChecksum after XmlMapping updateBug
WORDSNET-5293REF field should show hyperlink tooltipBug
WORDSNET-16691Printer metrics is rendered improperlyBug
WORDSNET-17724DOCX to PDF conversion issue with table renderingBug
WORDSNET-17769Converting DOCX to PDF overlaps the tablesBug

Public API and Backward Incompatible Changes

This section lists public API changes that were introduced in Aspose.Words 21.6. It includes not only new and obsoleted public methods, but also a description of any changes in the behavior behind the scenes in Aspose.Words which may affect existing code. Any behavior introduced that could be seen as regression and modifies the existing behavior is especially important and is documented here.

Added a new LayoutEntityType and new values for the Kind property within the LayoutEnumerator class

A new value Note has been added to the public enum LayoutEntityType:

public enum LayoutEntityType
{
...
  /// <summary>
  /// Represents placeholder for note content.
  ///
  /// Note may have <see cref="Line"/> and <see cref="Row"/> child entities.
  /// </summary>
  Note
...
}

This will help users iterate over notes inside footnotes/endnotes containers.

Use Case:

auto doc = System::MakeObject<Document>(u"SomeDocument.docx")
auto en = System::MakeObject<LayoutEnumerator>(doc);

// We start from the first page.
assert(en->get_Type() == LayoutEntityType::Page);

// Move to the first column on the page.
en->MoveFirstChild();
assert(en->get_Type() == LayoutEntityType::Column);

// Move to the first child in the column.
en->MoveFirstChild();

do
{
  // Iterate to a footnote container.
  if (en->get_Type() == LayoutEntityType::Footnote)
  {
    break;
  }
}
while(en->MoveNext());

// If the footnote container exists in the column, we will process notes.
if (en->get_Type() == LayoutEntityType::Footnote)
{
  // Move to the first note in the footnote container.
  if (en->MoveFirstChild())
  {
    do
    {
     // Move over notes inside the footnote container.
     assert(en->get_Type() == LayoutEntityType::Note);
     // Do something.
    }
    while (en->MoveNext());
   }
 }
}

New Kind enum values for the LayoutEnumerator class

8 new Kind enum values have been added for the LayoutEnumerator class:

"FOOTNOTES"
"ENDNOTES"
"FOOTNOTESEPARATOR"
"ENDNOTESEPARATOR"
"FOOTNOTECONTINUATIONSEPARATOR"
"ENDNOTECONTINUATIONSEPARATOR"
"FOOTNOTECONTINUATIONNOTICE"
"ENDNOTECONTINUATIONNOTICE"

The most useful values are the kinds that allow to unambiguously determine what type of separator you are working with. This is true because for all kinds of separators the LayoutEntityType is LayoutEntityType.NoteSeparator.

Use Case:

auto doc = System::MakeObject<Document>(u"SomeDocument.docx")
auto en = System::MakeObject<LayoutEnumerator>(doc);

// We start from the first page.
assert(en->get_Type() == LayoutEntityType::Page);

// Move to the first column on the page.
en->MoveFirstChild();
assert(en->get_Type() == LayoutEntityType::Column);

// Move to the first child in the column.
en.MoveFirstChild();

do
{
   if (en->get_Type() == LayoutEntityType::NoteSeparator && en->get_Kind() == u"FOOTNOTESEPARATOR")
   {
      // Do something.
   }

   if (en->get_Type() == LayoutEntityType::NoteSeparator && en->get_Kind() == u"FOOTNOTECONTINUATIONSEPARATOR")
   {
      // Do something.
   }

   if (en->get_Type() == LayoutEntityType::NoteSeparator && en->get_Kind() == u"FOOTNOTECONTINUATIONNOTICE")
   {
      // Do something.
   }
}
while(en->MoveNext());
}

Added a new public property CustomXmlPart.DataChecksum

Related issue: WORDSNET-22082

Added a new public property CustomXmlPart.DataChecksum:

/// <summary>
/// Specifies a cyclic redundancy check (CRC) checksum of the <see cref="Data"/> content.
/// </summary>
public long DataChecksum

It’s readonly. It is numeric representation of the storeItemChecksum element. The checksum is computed using the data of the corresponding custom XML data part.

Use Case:

auto doc = System::MakeObject<Document>(u"Document.docx");
auto sdt = System::StaticCast<StructuredDocumentTag>(doc->get_GetChild(NodeType::StructuredDocumentTag, 0, true));
std::cout << sdt->get_XmlMapping()->get_CustomXmlPart()->get_DataChecksum() << '\n';
auto doc = System::MakeObject<Document>();

auto sdt = System::MakeObject<StructuredDocumentTag>(doc, SdtType::RichText, MarkupLevel::Block);
doc->get_FirstSection()->get_Body()->Insert(sdt, nullptr, false);
             
sdt->get_XmlMapping()->SetMapping(doc->get_CustomXmlParts()->Add(u"C8FFD3B6-7EF8-4963-895B-3565F68A03E7",
    u"<root><text>test</text></root>"), u"/root/text", u"");
std::cout << sdt->get_XmlMapping()->get_CustomXmlPart()->get_DataChecksum() << '\n';

A new Document.LayoutOption is added to control page numbering in a continuous section

Related issues: WORDSNET-22177, WORDSNET-17760.

A new layout option is added to control Aspose.Words behavior when computing page numbers in a continuous section that restarts page numbering:

namespace Aspose.Words.Layout
{
...
    public class LayoutOptions
    {
...
        /// <summary>
        /// Gets or sets the mode of behavior for computing page numbers when a continuous section
        /// restarts the page numbering.
        /// </summary>
        /// <remarks>
        /// The default behavior matches the latest MS Word version.
        /// Older page numbering logic demonstrated by MS Word 2016 is available via this option.
        /// </remarks>
        public ContinuosSectionRestart ContinuousSectionPageNumberingRestart
}
...
    /// <summary>
    /// Represents different behaviors when computing page numbers in a continuous section that restarts page numbering.
    /// </summary>
    public enum ContinuosSectionRestart
    {
        /// <summary>
        /// Page numbering always restarts regardless of content flow.
        /// </summary>
        /// <remarks>
        /// This behavior is demonstrated by all MS Word versions, except Word 2016.
        /// </remarks>
        Always = 0,
        /// <summary>
        /// Page numbering restarts only if there is no other content before the section on the page where the section starts.
        /// </summary>
        /// <remarks>
        /// The behavior is demonstrated by MS Word 2016.
        /// </remarks>
        FromNewPageOnly,
    }
}

After the changes, the default Aspose.Words behavior matches the current MS Word version (2019). The older behavior implemented per WORDSNET-17760 is still avaialable via the introduced option:

auto doc = System::MakeObject<Document>(u"input.docx");
doc->get_LayoutOptions()->set_ContinuousSectionPageNumberingRestart(ContinuosSectionRestart::FromNewPageOnly);
doc->Save(u"out.pdf");

Note: It is required to rebuild the document page layout (via Document.UpdatePageLayout() method) after changing the Document.LayoutOptions values.

Implemented ability to set fill and stroke formatting for chart series, data points and markers

Related issue: WORDSNET-12275

Properties of the new ChartFormat type have been added to the ChartSeries, ChartDataPoint and ChartMarker classes.

public class ChartSeries
{
    /// <summary>
    /// Provides access to fill and line formatting of the series.
    /// </summary>
    public ChartFormat Format { get; }
}

public class ChartDataPoint
{
    /// <summary>
    /// Provides access to fill and line formatting of this data point.
    /// </summary>
    public ChartFormat Format { get; }
}

public class ChartMarker
{
    /// <summary>
    /// Provides access to fill and line formatting of this marker.
    /// </summary>
    public ChartFormat Format { get; }
}

/// <summary>
/// Represents the formatting of a chart element.
/// </summary>
public class ChartFormat
{
    /// <summary>
    /// Gets fill formatting for the parent chart element.
    /// </summary>
    public Fill Fill { get; }

    /// <summary>
    /// Gets line formatting for the parent chart element.
    /// </summary>
    public Stroke Stroke { get; }
}

Also aliases to some existing properties have been added to the Stroke class: ForeColor, BackColor, Visible and Transparency. The original Color, Color2, On and Opacity properties will be made obsolete in a future version of Aspose.Words.

public class Stroke
{
    /// <summary>
    /// Gets or sets the foreground color of the stroke.
    /// </summary>
    public Color ForeColor { get; set; }

    /// <summary>
    /// Gets or sets the background color of the stroke.
    /// </summary>
    public Color BackColor { get; set; }
     
    /// <summary>
    /// Gets or sets a flag indicating whether the stroke is visible.
    /// </summary>
    public bool Visible { get; set; }
     
    /// <summary>
    /// Gets or sets a value between 0.0 (opaque) and 1.0 (clear) representing the degree of transparency of the stroke.
    /// </summary>
    /// <remarks>
    /// The default value is 0.
    /// </remarks>
    public double Transparency { get; set; }
}

Use Case: Explains how to set series color

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();
auto seriesColl = chart->get_Series();

// Delete default generated series.
seriesColl->Clear();

// Create category names array.
auto categories = System::MakeArray<System::String>({ u"AW Category 1", u"AW Category 2" });

// Adding new series. Value and category arrays must be the same size.
auto series1 = seriesColl->Add(u"AW Series 1", categories, System::MakeArray<double>({ 1, 2 }));
auto series2 = seriesColl->Add(u"AW Series 2", categories, System::MakeArray<double>({ 3, 4 }));
auto series3 = seriesColl->Add(u"AW Series 3", categories, System::MakeArray<double>({ 5, 6 }));

// Set series color.
series1->get_Format()->get_Fill()->set_ForeColor(Color::get_Red());
series2->get_Format()->get_Fill()->set_ForeColor(Color::get_Yellow());
series3->get_Format()->get_Fill()->set_ForeColor(Color::get_Blue());

doc->Save(u"ColumnColor.docx");

Use Case: Explains how to set individual formatting for categories of a column chart

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();

// Delete default generated series.
chart->get_Series()->Clear();

// Adding new series.
auto series = chart->get_Series->Add(u"AW Series 1", 
    System::MakeArray<System::String>({ u"AW Category 1", u"AW Category 2", u"AW Category 3", u"AW Category 4" }), 
    System::MakeArray<double>({ 1, 2, 3, 4 }));

// Set column formatting.
auto dataPoints = series->get_DataPoints();
dataPoints->idx_get(0)->get_Format()->get_Fill()->PresetTextured(PresetTexture::Denim);
dataPoints->idx_get(1)->get_Format()->get_Fill()->set_ForeColor(Color::get_Red());
dataPoints->idx_get(2)->get_Format()->get_Fill()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(3)->get_Format()->get_Fill()->set_ForeColor(Color::get_Blue());

doc->Save("IndividualColumnFormatting.docx");

Use Case: Explains how to set line color and weight

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();
auto seriesColl = chart->get_Series();

// Delete default generated series.
seriesColl->Clear();

// Adding new series.
auto series1 = seriesColl->Add(u"AW Series 1", System::MakeArray<double>({ 0.7, 1.8, 2.6 }), System::MakeArray<double>({ 2.7, 3.2, 0.8 }));
auto series2 = seriesColl->Add(u"AW Series 2", System::MakeArray<double>({ 0.5, 1.5, 2.5 }), System::MakeArray<double>({ 3, 1, 2 }));

// Set series color.
series1->get_Format()->get_Stroke()->set_ForeColor(Color::get_Red());
series1->get_Format()->get_Stroke()->set_Weight(5);
series2->get_Format()->get_Stroke()->set_ForeColor(Color::get_LightGreen());
series2->get_Format()->get_Stroke()->set_Weight(5);

doc->Save("LineColorAndWeight.docx");

Use Case: Explains how to set line segment colors

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();

// Delete default generated series.
chart->get_Series()->Clear();

// Adding new series.
auto series = chart->get_Series()->Add(u"AW Series 1", System::MakeArray<double>({ 0.7, 1.8, 2.6, 3.5 }), System::MakeArray<double>({ 2.7, 3.2, 0.8, 0.1 }));
series->get_Format()->get_Stroke()->set_Weight(4);

// Set line colors.
auto dataPoints = series->get_DataPoints();
dataPoints->idx_get(1)->get_Format()->get_Stroke()->set_ForeColor(Color::get_Red());
dataPoints->idx_get(2)->get_Format()->get_Stroke()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(3)->get_Format()->get_Stroke()->set_ForeColor(Color::get_Blue());

doc->get_Save(u"LineSegmentColor.docx");

Use Case: Explains how to set Pie chart colors

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();

// Delete default generated series.
chart->get_Series()->Clear();

// Adding new series.
auto series = chart->get_Series()->Add(u"Series 1", System::MakeArray<System::String>({ "Category1", "Category2", "Category3" }), System::MakeArray<double>({ 2.7, 3.2, 0.8 }));

// Set data point color.
auto dataPoints = series->get_DataPoints();
dataPoints->idx_get(0)->get_Format()->get_Fill()->set_ForeColor(Color::get_Red());
dataPoints->idx_get(1)->get_Format()->get_Fill()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(2)->get_Format()->get_Fill()->set_ForeColor(Color::get_Green());

doc->Save(u"PieColor.docx");

Use Case: Explains how to set marker formatting

auto doc = System::MakeObject<Document>();
auto builder = System::MakeObject<DocumentBuilder>(doc);

auto shape = builder->InsertChart(ChartType::Column, 432, 252);

auto chart = shape->get_Chart();

// Delete default generated series.
chart->get_Series()->Clear();

// Adding new series.
auto series = chart->get_Series()->Add(u"AW Series 1", System::MakeArray<double>({ 0.7, 1.8, 2.6, 3.9 }), System::MakeArray<double>({ 2.7, 3.2, 0.8, 1.7 }));

// Set marker formatting.
series->get_Marker()->set_Size(40);
series->get_Marker()->set_Symbol(MarkerSymbol::Square);
auto dataPoints = series->DataPoints();

dataPoints->idx_get(0)->get_Marker()->get_Format()->get_Fill()->PresetTextured(PresetTexture::Denim);
dataPoints->idx_get(0)->get_Marker()->get_Format()->get_Stroke()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(1)->get_Marker()->get_Format()->get_Fill()->PresetTextured(PresetTexture::WaterDroplets);
dataPoints->idx_get(1)->get_Marker()->get_Format()->get_Stroke()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(2)->get_Marker()->get_Format()->get_Fill()->PresetTextured(PresetTexture::GreenMarble);
dataPoints->idx_get(2)->get_Marker()->get_Format()->get_Stroke()->set_ForeColor(Color::get_Yellow());
dataPoints->idx_get(3)->get_Marker()->get_Format()->get_Fill()->PresetTextured(PresetTexture::Oak);
dataPoints->idx_get(3)->get_Marker()->get_Format()->get_Stroke()->set_ForeColor(Color::get_Yellow());

doc->Save(u"MarkerFormatting.docx");

Implemented Rendering of OOXML Ink (InkML subset)

OOXML Ink content is specified by a subset of syntax and semantics for the Ink Markup Language (see “InkML specification”). The details of the subset are described in “2.1.4 Ink Content Part” of OOXML specification.

Prior to this version Aspose.Words could render only fallback shapes for OOXML Ink objects, i.e. InkML was not actually processed and a simple prerendered image was used instead. And now OOXML ink content part can be rendered directly (“cold” rendering).

To control the mode of Ink rendering a new public property SaveOptions.ImlRenderingMode was introduced and the corresponding enum was added:

/// <summary>
/// Gets or sets a value determining how ink (InkML) objects are rendered.
/// </summary>
/// <remarks>
/// The default value is <see cref="Aspose.Words.Saving.ImlRenderingMode.InkML"/>.
/// <para>This property is used when the document is exported to fixed page formats.</para>
/// </remarks>
public ImlRenderingMode ImlRenderingMode
/// <summary>
/// Specifies how ink (InkML) objects are rendered to fixed page formats.
/// </summary>
public enum ImlRenderingMode
{
    /// <summary>
    /// If fall-back shape is available for ink (InkML) object, Aspose.Words renders fall-back shape instead of the InkML.
    /// </summary>
    /// <remarks>
    /// Please note that after saving a document to a fixed page format with fall-back rendering mode,
    /// InkML objects in the Aspose.Words document model are permanently replaced with their fall-back counterparts.
    /// As a result, saving the same document again will always use fall-back shapes, even if ImlRenderingMode is set to InkML.
    /// </remarks>
    Fallback,
    /// <summary>
    /// Aspose.Words ignores fall-back shape of ink (InkML) object and renders InkML itself.
    /// This is the default mode.
    /// </summary>
    InkML
}

In addition to limitations described in “2.1.4 Ink Content Part” of OOXML specification, the first version of InkML cold rendering has the following limitations:

InkML Structural Element:

  • channel
    • The name attribute can be any string, but strings other than the following strings are ignored: X, Y, F.
    • The units attribute can be any string, but any value other than the following value is ignored: dev, in, cm
  • traceGroup
    • Any content other than trace and traceGroup is ignored and not persisted.
  • brushProperty
    • rasterOp - A value that defines how the colors of the pen and background interact. Is ignored.
    • antiAliased - A value that indicates if the drawn ink is anti-aliased. Is ignored.
    • fitToCurve - A value that indicates if the ink is rendered as a series of curves versus as lines between pen sample points. Is ignored.
  • annotationXML
    • Is ignored and not persisted.

Due to the limitations described above the Aspose.Words output may not exactly match the MS Word output in some cases. If the rendering result is unsatisfactory, please use the fallback mode to get a result similar to previous versions of Aspose.Words.

ImportFormatOptions.MergePastedLists property was introduced

Related issue: WORDSNET-22103

The following new public property was added into the ImportFormatOptions class:

/// <summary>
/// Gets or sets a boolean value that specifies whether pasted lists will be merged with surrounding lists.
/// The default value is <c>false</c>.
/// </summary>
public bool MergePastedLists

Use Case: Explains how to merge pasted lists with surrounding lists when adding one document to another

auto srcDoc = System::MakeObject<Document>(u"src.docx");
auto dstDoc = System::MakeObject<Document>(u"dst.docx");

auto importFormatOptions = System::MakeObject<ImportFormatOptions>();
importFormatOptions->set_MergePastedLists(true);
dstDoc->AppendDocument(srcDoc, ImportFormatMode::UseDestinationStyles, importFormatOptions);

dstDoc->Save(u"out.docx");

Public API for working with textures has been introduced

Related issue: WORDSNET-22002

A new Fill.PresetTextured() public method has been added:

/// <summary>
/// Sets the fill to a preset texture.
/// <param name="presetTexture"><see cref="Drawing.PresetTexture"/></param>
/// </summary>
public void PresetTextured(PresetTexture presetTexture)

A new Fill.PresetTexture public property has been added:

/// <summary>
/// Gets a <see cref="Drawing.PresetTexture"/> for the fill.
/// </summary>
public PresetTexture PresetTexture{ get; }

A new public enum has been introduced:

/// <summary>
/// Specifies texture to be used to fill a shape.
/// </summary>
public enum PresetTexture

Use Case: Explains how to get and apply a texture to a fill

// Open some document with a shape.
auto doc = System::MakeObject<Document>(u"DocWithShape.docx");

// Get Fill object for the first shape.
auto fill = doc->get_FirstSection()->get_Body()->get_Shapes()->idx_get(0)->get_Fill();

// Check Fill PresetTexture value.
std::cout << "PresetTexture value is: " << System::ObjectExt::ToString(fill->get_PresetTexture()).ToUtf8String() << '\n';

// Apply BrownMarble texture to the shape fill.
fill->PresetTextured(PresetTexture::BrownMarble);

doc->Save(u"BrownMarble.docx");

TxtSaveOptions.MaxCharactersPerLine property was introduced

Related issue: WORDSNET-22090

The following new public property was added into the TxtSaveOptions class:

/// <summary>
/// Gets or sets an integer value that specifies the maximum number of characters per one line.
/// The default value is 0, that means no limit.
/// </summary>
public int MaxCharactersPerLine { get; set; }

Use Case: Explains how to limit maximum characters per line in output document of TXT format:

auto doc = System::MakeObject<Document>(u"input.docx");

auto saveOptions = System::MakeObject<TxtSaveOptions>();
// Set 60 characters as maximum allowed per one line.
saveOptions->set_MaxCharactersPerLine(60);

doc->Save("out.txt", saveOptions);