- Created by Alecia Elia , last modified by TK Conrad on Jul 08, 2021
You are viewing an old version of this page. View the current version.
Compare with Current View Page History
« Previous Version 16 Next »
Loops can be used to iterate through multiple records and fields to be merged into in your document. Examples of this are displaying a list of contacts, or displaying a list of violations associated with an evaluation. Loops may also be filtered to limit the merge to selected records. An example of this is filtering a contact loop to only merge contacts that have a particular affiliation type/role code.
Loops
A foreach loop will be inserted automatically into the template if a group name (Bold field names) is double-clicked in the field selection list.
Merge fields set up for a loop can be displayed using a foreach statement such as the example.
|
Loop Formatting
The loop above will list all items one after another without any line breaks, to format the foreach loop check the examples below:
Template Code | Output |
---|---|
Prefix<<foreach [permit in permits]>> <<[Item]>><</foreach>>Suffix | Prefix Item1 Item2 Item3 Suffix |
Prefix <<foreach [permit in permits]>><<[Item]>> | Prefix Item1 |
Prefix | Prefix |
1. <<foreach [permit in permits]>><<[Item]>> | 1. Item1 |
<<foreach [item in items]>><<[item.IndexOf() !=0 ? “, “: “”]>><<[item.itemName]>><</foreach>>. | Item1, Item2, Item3. |
The last loop formatting in the examples above usesIndexOf()
and Ternary"
?:"
to decide whether to place a comma "," or not to place anything after the first item, then the "." is placed after closing the foreach loop, which translates into"
is this true ? Yes : No".
The code will result in adding a comma after each item except the last item in a list where a "." will be added.
Extension Methods for loops
Foreach loop can be combined with extension methods to access data within the dataset such as .Where, .Take(), .Orderby()
, etc.. Review the list of methods available.
Filtered Loops
To filter a loop so that it only displays certain data (such as only displaying a contact with a specific affiliation type), add a ".Where" clause to the foreach syntax. For example, the following code only displays those contacts that have an affiliation type of manure hauler ("MNR_HAUL"):
|
To protect against multiple matching contacts being returned (such as 2 or more manure haulers in the example above), add .Take(1) to the markup:
|
To return a different contact role when another role record does not exist, syntax such as the following can be used:
|
In the above example, the foreach loop will filter based on affiliation type "ENGR", Take(1) is used in case of multiple contacts with the same affiliation type, and return Contact Last Name. An if statement is used to check if the count of affiliation type "ENGR" is 0 using .Count() == 0
and return the contact with "OWNR" affiliation type if true.
The x => x.item
is a Lambda expression and any letter can be chosen to replace x
Sorting Loops
To sort loops so that it displays the data in a certain order, add an ".OrderBy" or "OrderByDescending" clause to the foreach syntax.
|
In the above example the code will sort contact based on contact Last Name and display Contact First Name, Contact Last Name in a list.
If trying to sort an ordered list in a tag value (1. 2. 3. etc..) then you'll need to have the .OrderBy only look at those specific values otherwise it will sort incorrectly. One way to do that is to use the .Substring() function, in combination with the .IndexOf() function and the Convert.ToDouble() function, shown below.
|
Explanation of the above string:
- .OrderBy() must be in between the [] of the foreach statement, attached to the last tag name.
- Convert.ToDouble() is used to enclose the specific text you are trying to sort by (i.e. c => c.questions) c.questions here is what you are trying to have OrderBy sort off of. **Please note ToInt() has not worked in this situation, so ToDouble() is best.
- .Substring(start, end) is used to pick out a certain part of the tag value. I want to pick out what ever is before the period since those will be the numbers in my list of questions. So I have the Substring start at 0 (the beginning of the tag value) and then end at the .IndexOf(".") the period.
- .IndexOf() will give us the exact placement of the period in the string so that when it looks at the substring of (0, c.questions.IndexOf(".")) it will only be looking at "1" as the first question. If the numbers were proceeded by a "-" dash, then I would put c.questions.IndexOf("-") instead. You can look for any unique value this way. If it's not a unique value, it will only find the first occurrence of that value.
Breaking Out of a Loop
Sometimes it's useful or necessary to break out of a loop when a particular condition has been met or value has been found. This can be achieved by doing the following:
Create a variable and set it to "true".
At the beginning of the loop, check to see that the variable is equal to "true".
Somewhere in the loop, perhaps when a specific condition has been met, set the value of the variable to "false".
The following code illustrates this. (The code is indented here for legibility but doesn't need to be indented in the document template.)
|
In the above example, the text "W50-51 Failure to meet the applicable requirements..." is displayed if the reference begins with “W50-51”.
Loop in Table
A foreach statement within a table will loop within the table. In the example below, for each feature ID, a new row within the table will be created and the associated fields filled:
Feature ID | Description | Status |
---|---|---|
<<foreach [feature in features.OrderBy(e => e.featureText)]>><<[featureText]>> | <<[featureDescription]>> | <<[status]>> |
Nested Loops
The following example demonstrates a loop for each of a site’s permits and a sub-loop listing each permit’s features in a table:
<<foreach>> syntax example. The foreach tag is used to display the same fields for repeating records. For example, in this figure, the fields between the <<foreach>> and <</foreach>> tags are displayed for all permits associated with this site, regardless of how many permits there are. (Note that the "Site Permits" caption does not repeat for each caption, as it is not included within the <<foreach>> tags. ). Also note the OrderBy syntax. OrderBy should always be used to sort looped data. In this case, the repeated permits are sorted by permit number (permitNum).
<<foreach>> syntax example in a table. In cases in which the foreach tags are used within a table, the template engine is smart enough to know that there should be a row for each looped item.
Date formatting example. This example shows how to format a date in year-month-day order, e.g., 2019.04.25.
Logical if example. This example demonstrates a way of only rendering output for a particular field if data is present in that field. See "Address with check for missing data" below for another example.
Useful methods to use with loops
Method | Description |
---|---|
Where( | Filters a sequence of values based on a predicate. |
Take(int) | Returns a specified number (int) of contiguous elements from the start of a sequence. |
OrderBy() | Sorts the elements of a sequence in ascending order according to a key. |
OrderByDescending() | Sorts the elements of a sequence in descending order according to a key. |
Count() | Returns the number of elements in a sequence. |
Combining methods
The above methods may be combined together.
The following example shows a loop that combines Where( ), Order By( ), and Take(1) methods.
<<foreach [workflowTask in workflowTasks.Where(w => w.TaskStatusDescription == "Unstarted").OrderBy(w => w.taskSequence).Take(1)]>><<[workflowTask.taskName]>><</foreach>>
Additional methods to use with loops
Extension Method | Examples and Notes |
---|---|
| |
| |
| |
| |
| |
| |
| |
| |
| This does not appear to work in our current version of Aspose.Words. See below, "Performing Distinct using foreach and variable". |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
ThenBy(ComparableSelector) |
|
| |
|
Contact Loops - Finding Available Affiliation Types
Navigate to Admin > Lookups > Affiliation Types (Roles), this will display existing affiliation types. The Lookup Code is what should be used when filtering contacts based on Affiliation Types.
Affiliation Types related to one or more functional areas (e.g. Applicant, Permittee) can only be assigned in the corresponding functional area(s). Roles which do not have a functional area specified are available in all functional areas.
Tips and Tricks
A foreach syntax block can be inserted automatically into the template by double clicking a group name in the field selection list.
Performing Distinct using foreach and variable
We had a document template that needed to show the distinct list of receiving waters across multiple permitted features. The distinct is important here because in almost all cases, the receiving water for all features will be the same.
Unfortunately, the Distinct() method does not appear to work in our current version of Aspose.Words. We therefore came up with a workaround that does the following:
- Set a local variable to a default value (e.g., "None").
- Setup a foreach loop to iterate through all of the features.
- Within the loop, compare the receiving water to the local variable. If different:
- Set the local variable = the receiving water;
- Display the receiving water.
- If the receiving water is the same as the local variable, then ignore and move on.
Here is the syntax we used in our document template (I added carriage returns for visibility only, this will cause a lot of unwanted carriage returns as written):
<<var [v_recvWtr = “None”]>> <<foreach [featrItem in coMuni.featr.OrderBy(p => p.recvWtr)]>> <<if[!string.IsNullOrEmpty(featrItem.recvWtr)]>> <<if[featrItem.recvWtr != v_recvWtr]>> <<if[v_recvWtr!=“None”]>>, <</if>> <<var [v_recvWtr = featrItem.recvWtr]>> <<[v_recvWtr]>><</if>><</if>><</foreach>>
On this page
Sub-Topics
Related Content
- No labels