Hi Laure,
Thank you for additional information. I created a simple code example that should help you to resolve the issue without changing your templates or data source. Please see my comments through the code:
public void Test001()
{
// Supose, you have datasource with column names that contains whitespaces.
// For testing purposes, lets use dummy data.
DataTable data = new DataTable();
data.Columns.Add("NOM PATIENT");
data.Columns.Add("NOM PATIENT MIN");
data.Columns.Add("NOM PATIENT COMPL");
data.Columns.Add("PRENOM PATIENT");
data.Columns.Add("DATE NAISSANCE");
data.Columns.Add("AGE");
data.Columns.Add("SEXE");
data.Columns.Add("ADR_1 PAT");
data.Columns.Add("ADR_2 PAT");
// Add some dummy data.
data.Rows.Add(new object[] { "Andrey", "Noskov", "V", "Mr", "12/12/12", "30", "male", "My address 1", "My Address 2" });
// open source document with palceholders.
Document doc = new Document(@"Test001\input.odt");
// Replace placeholders with mergefields.
// During replacing we will replace whitespaces in fields names with underlines.
doc.Range.Replace(new Regex(@"\~(?.*?)\~"), new ReplaceEvaluatorFindAndInsertMergefield(), false);
// Now we can map names in our datasource (with whitespaces) with names in the template (without whitespaces).
foreach (DataColumn column in data.Columns)
doc.MailMerge.MappedDataFields.Add(column.ColumnName.Replace(" ", "_"), column.ColumnName);
// Now, we can execute mail merge.
doc.MailMerge.Execute(data);
// Save output.
doc.Save(@"Test001\out.odt");
}
private class ReplaceEvaluatorFindAndInsertMergefield : IReplacingCallback
{
///
/// This method is called by the Aspose.Words find and replace engine for each match.
/// This method highlights the match string, even if it spans multiple runs.
///
ReplaceAction IReplacingCallback.Replacing(ReplacingArgs e)
{
// This is a Run node that contains either the beginning or the complete match.
Node currentNode = e.MatchNode;
// The first (and may be the only) run can contain text before the match,
// in this case it is necessary to split the run.
if (e.MatchOffset > 0)
currentNode = SplitRun((Run)currentNode, e.MatchOffset);
// This array is used to store all nodes of the match for further removing.
ArrayList runs = new ArrayList();
// Find all runs that contain parts of the match string.
int remainingLength = e.Match.Value.Length;
while (
(remainingLength > 0) &&
(currentNode != null) &&
(currentNode.GetText().Length <= remainingLength))
{
runs.Add(currentNode);
remainingLength = remainingLength - currentNode.GetText().Length;
// Select the next Run node.
// Have to loop because there could be other nodes such as BookmarkStart etc.
do
{
currentNode = currentNode.NextSibling;
}
while ((currentNode != null) && (currentNode.NodeType != NodeType.Run));
}
// Split the last run that contains the match if there is any text left.
if ((currentNode != null) && (remainingLength > 0))
{
SplitRun((Run)currentNode, remainingLength);
runs.Add(currentNode);
}
// Create Document Buidler aond insert MergeField
DocumentBuilder builder = new DocumentBuilder(e.MatchNode.Document as Document);
builder.MoveTo((Run)runs[runs.Count - 1]);
string fieldName = e.Match.Groups["FieldName"].Value.Replace(" ", "_");
builder.InsertField(string.Format("MERGEFIELD {0}", fieldName), string.Format("«{0}»", fieldName));
// Now remove all runs in the sequence.
foreach (Run run in runs)
run.Remove();
// Signal to the replace engine to do nothing because we have already done all what we wanted.
return ReplaceAction.Skip;
}
///
/// Splits text of the specified run into two runs.
/// Inserts the new run just after the specified run.
///
private static Run SplitRun(Run run, int position)
{
Run afterRun = (Run)run.Clone(true);
afterRun.Text = run.Text.Substring(position);
run.Text = run.Text.Substring(0, position);
run.ParentNode.InsertAfter(afterRun, run);
return afterRun;
}
}
Hope this helps.
Best regards,