Advanced Organization Level Configuration
There are a number of advanced options for organization levels. These are shown when you click on the Advanced checkbox on the Organization Levels page.
Enabling Advanced Options for Organization Levels
CommCare offers advanced settings for organization levels to better manage complex projects. These options, available by checking the Advanced checkbox on the Organization Levels page, help customize data access, user roles, and multi-location structures to fit your project’s needs.
Advanced Organization Options
Type Code
A Type Code is a unique identifier assigned to a location type in CommCare. It helps reference specific location types in lookup tables or logic expressions when building forms and reports.
How Does It Work?
Let’s say your project includes different organization levels, like Country, State, and District Warehouses. Each of these levels can have a Type Code to make them easier to reference in formulas.
For example:
If your location type is District Warehouse, the Type Code might be ‘district_warehouse’.
In the example below, the location is a district, so its Type Code is ‘district’.
Using Type Codes helps streamline data organization and ensures that location-based logic works correctly in your CommCare application.
Level to Expand From
By default, CommCare syncs a user’s assigned location, along with its ancestors (higher levels) and descendants (lower levels) in the organization hierarchy.
For example, if the hierarchy is:
District → Clinic → CHW (Community Health Worker)
And a user is assigned to a Clinic, they will receive data for:
Their District (ancestor)
Their Clinic (assigned location)
All CHWs within that Clinic (descendants)
Why Adjust "Level to Expand From"?
Sometimes, a user needs access to locations outside their standard hierarchy.
For instance, if a user is assigned to ‘Clinic A’ but needs access to ‘all Clinics within their District’, you can set the Level to Expand From to ‘District’ instead of Clinic. This would be helpful if you had clinics workers who might register patients in one clinic today, but want the patient’s case to be owned by another clinic long-term (e.g. the clinic they will receive care from, the one closer to their house, etc).
Now that you have adjusted the Level to Expand From to ‘District,' the user will receive:
Their clinic
All Clinics locations in that District (including their own)
This setting allows greater flexibility in data access, ensuring users get the information they need without unnecessary restrictions.
"Level to Expand To" (Performance Optimization)
The Level to Expand To setting helps control which location types are sent to a user. This is mainly used to optimize performance by reducing unnecessary data.
How It Works
By default, if a user is assigned to a District, CommCare sends them data for:
Their District (assigned location)
All Clinics in that District
All CHWs within each Clinic
If there are too many CHWs, it can slow down the app by increasing the amount of data that needs to be processed.
Why Adjust "Level to Expand To"?
If CHW data isn’t needed, you can set Level to Expand To = Clinic. This means the user will now receive:
Their District
All Clinics in that District
🚫 CHWs will NOT be included, reducing data load
This setting improves app performance while ensuring the user still gets the necessary location data.
Include without Expanding/Force Sync
The Force Sync option allows CommCare to send all locations of a specific type to a user without including their lower-level locations (descendants).
How It Works
In a standard hierarchy:
District → Clinic → CHW (Community Health Worker)
If a user is assigned to a Clinic, they normally receive:
Their District
Their Clinic (assigned location)
All CHWs in that Clinic (descendants)
Why Use Force Sync?
Let’s say a user needs to see all Districts but not the Clinics or CHWs within them.
Enabling Force Sync for Districts ensures the user receives:
All Districts
Their assigned Clinic
🚫 No Clinics or CHWs from other Districts
This setting is useful when users need high-level location access without being overwhelmed by unnecessary data.
Include Only
The Include Only setting lets you control which location types or levels a user can see in CommCare. This ensures users only access the locations relevant to their work while keeping the interface clean and manageable.
How It Works
Every location type you select must also include its parent levels.
For example, in a District → Clinic → CHW hierarchy:
If Include Only is applied at the Clinic level, the user will see:
Their assigned Clinic
🚫 No Districts (parent level is removed)
🚫 No CHWs (child level is removed)
Why Use "Include Only"?
This setting is useful when:
You want to limit unnecessary information for users.
Users should only focus on their assigned location without seeing higher or lower levels.
It helps simplify workflows by displaying only the data they need.
By keeping the interface focused, ‘Include Only' makes it easier for users to navigate and complete their tasks efficiently.
Translating or Customizing Location Display
Location names in CommCare cannot be translated directly, but there is a way to customize how they appear in your app. This method lets you display locations in different languages and formats as needed.
Understanding the Approach
Normally, multiple-choice or checkbox questions using lookup tables pull location names directly from the system, giving you little control over customization. To work around this, we create a nodeset using a repeat group with a model iteration query. This allows us to define additional properties, like translated names, for use in lookup table questions.
Step 1: Storing Location Translations
To make this work, we first need to store translations for location names. We can do this by adding custom location fields for different languages. Go to ‘Edit Location Fields’ under Organization Structure and Click on ‘Add a Field’. Add new custom fields for each language:
name-en (for English)
name-hin (for Hindi)
name-es (for Spanish)
After adding these fields, we edit each location (e.g., District, Block, CHW) to store the correct names in each language.
Go back to Manage Locations.
Select a location (e.g., a District, Block, or CHW).
Enter the translated names under the new fields (
name-en
,name-hin
,name-es
).Repeat this for all locations.
Save changes.
Step 2: Referencing the Current Language
To use the correct translation, we need to reference the current language selected in the app. This process is described at Lookup Tables | Using Lookup Tables with Multiple LanguagesBy setting up a lang-code question, we can use:
jr:itext('lang-code-label')
to dynamically fetch the right translation.
Step 3: Setting Up a Nodeset for Location Display
Create a Repeat Group:
Under "Advanced," set the Model Iteration ID Query to:
instance('locations')/locations/location/@id
This ensures the repeat group runs for each location in the system (e.g., all Districts, Blocks, and CHWs).
Set Instance ID to
"locations"
and Instance URI to"jr://fixture/locations"
.
Store Location Translations in Hidden Values:
Inside the repeat group, create a hidden value to store the translated name:
cond(jr:itext('lang-code-label') = 'es', instance('locations')/locations/location[@id = ../@id]/location_data/name-es, jr:itext('lang-code-label') = 'en', instance('locations')/locations/location[@id = ../@id]/location_data/name-en, jr:itext('lang-code-label') = 'hin', instance('locations')/locations/location[@id = ../@id]/location_data/name-hin, instance('locations')/locations/location[@id = ../@id]/name)
This ensures the correct language version of the location name is selected dynamically.
Step 4: Setting Up the Lookup Table Question
Create a Lookup Table Question.
Link it to the Nodeset:
Drag the repeat group into the Query Expression field. It should look like:
/data/locations/item
(Ensure it starts with
/data/
, as using#form/
will cause an error.)
Set Display & Value Fields:
Value Field:
@id
Display Text Field:
name
(or any custom field you want to show)
Final Outcome
Now, when users select a location (e.g., District → Block → CHW), they will see the name in their preferred language. This method ensures proper translations while keeping the lookup table flexible and customizable.
Referencing the Location Hierarchy in Applications
What is this feature?
Referencing the Location/Organization Hierarchy means using XPath expressions in CommCare to dynamically pull location-related data into your forms and workflows. This allows apps to automatically retrieve and display information based on a user’s assigned location. This can be incredibly helpful, because it allows the application to find and store information relevant to the case based on the user’s information without burdening the user.
For example, if I have a user who is assigned to District A, and I calculate that in a hidden value, I can then refer to the location hierarchy to get other information about District A, or other locations above or below it. If I wanted to save the name of the state or country that District A is a part of, I could do that easily using calculation conditions in hidden values (without having to make the user manually answer questions).
Why use this feature?
Automate Data Entry – Users don’t have to manually enter their location details; the app can automatically retrieve them.
Control User Access – Helps restrict data visibility based on the user’s assigned location. For example, a CHW can only see data for their assigned village, while a District Manager can access all villages in their district.
Improve Reporting & Analysis – Ensures that reports and dashboards display location-specific data for better decision-making.
Simplify Case Management – Helps link cases to the correct location automatically.
By using this feature, organizations can reduce errors, save time, and ensure accurate data collection in their CommCare applications.
How to use this feature?
Technical Expertise Required - The following documentation requires knowledge of XPath code writing and developers concepts. It provides guidance on how to reference the location hierarchy in various parts of CommCare applications.
Ensuring Access to the Location Hierarchy
To reference the location hierarchy in forms, you need to manually edit the XML source and construct XPath expressions.
Confirming Access to the Location Data Source
Before using location references, ensure the form has access to the Locations data source:
If the form contains at least one select question with locations as choices, the data source will be available automatically.
If not, create a dummy question following these steps:
Add a select question with locations as choices.
Set the Display Condition to
false()
to hide it.
This ensures that location data is accessible throughout the form. If you are already using a Locations-driven select question, the dummy question is not needed.
Enabling Access to Location Hierarchy
All new projects in CommCare automatically include the location hierarchy. However, older projects may be configured to use the Old Hierarchical Fixture. To transition to the new flat fixture format, follow these steps:
Navigate to Project Settings.
Select Pre-Release Features → Location Fixture. (This option will not be visible to new domains, so you can skip this step).
Enable the new flat fixture format.
Once enabled, new app versions will start using the new location format. After all users have transitioned to an updated version of the app, the Old Hierarchical Fixture can be turned off. This process is described in more detail in the documentation.
Retrieving the User’s Location ID
To access the currently logged-in user’s location ID, use the commcare_location_id
session variable. This can be referenced in a form with the following expression used in a hidden value type question -
instance('commcaresession')/session/user/data/commcare_location_id
This ID is linked to the user’s assigned location within the organizational hierarchy and can be used for data filtering, case management, and reporting.
Referencing the User’s Location or Ancestor Location in a Form
You can use the following XPath templates to reference the properties of a user’s location or their ancestor location. These examples assume a State → District → Block → Outlet hierarchy but should be customized based on your specific setup.
Finding the Organization Level Code
Each organization level has a unique code. To find the correct code:
Go to the Organization Level page.
Click on the Advanced checkbox to view the level-specific codes.
Example: Retrieving an Ancestor Location ID
If your hierarchy follows Village → City → County, and you need to retrieve the City ID for a Village-level user, modify the following XPath expression:
General Template:
instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@block_id
Customized for City ID:
instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@city_id
Replace @block_id
with @city_id
or any other relevant location level based on your hierarchy.
By implementing these expressions, you can dynamically pull location-based data for users and customize form logic accordingly.
I am a... | referencing my... | syntax in forms |
outlet | own location's name | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/name |
outlet | own location's type | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@type |
outlet | own location's site code | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/site_code |
outlet | own location's custom data | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/location_data/custom_field_id |
outlet | parent block's id | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@block_id |
outlet | parent block's name | instance('locations')/locations/location[@id = instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@block_id]/name |
outlet | parent block's site code | instance('locations')/locations/location[@id = instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@block_id]/site_code |
outlet | parent district's id | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@district_id |
outlet | parent district's name | instance('locations')/locations/location[@id = instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@district_id]/name |
block | parent district's id | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@district_id |
block | parent district's name | instance('locations')/locations/location[@id = instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@district_id]/name |
district | parent state's id | instance('locations')/locations/location[@id = instance('commcaresession')/session/user/data/commcare_location_id]/@state_id |
Accessing the location additional information (Location Fields)
Once you have the location's ID you can access it to get other information.
These examples assume your ID is stored in a hidden value called location_id
Referencing... | syntax in forms |
---|---|
The gps of the location | instance('locations')/locations/location[@id = /data/location_id]/latitude instance('locations')/locations/location[@id = /data/location_id]/longitude |
Any custom location fields you may have (ex. admin name) | instance('locations')/locations/location[@id = /data/location_id]/location_data/admin_name |
Syncing a Broader Location Hierarchy to Users
By default, CommCare synchronizes only the locations that a user is directly assigned to, along with their ancestors (higher levels) and descendants (lower levels).
For example, if a user is assigned to a ‘block’, their synced location data will include:
Their ‘block’ (assigned location).
The ‘district' and 'state’ (ancestors of the block).
All outlets within the ‘block’ (descendants).
Expanding Location Sync Beyond Default Settings
In some cases, users may need access to locations beyond their direct hierarchy. For instance:
A user assigned to a block might also need to see all districts within the state, even if they are not direct ancestors or descendants.
This can be configured using the "Expand From" settings, which allow additional location levels to be synced based on project needs.
For more details on how to configure this, refer to the "Expand From" section in the documentation.
When to Use Lookup Tables Vs Locations
When working with hierarchical data in CommCare, users often choose between Lookup Tables and Locations to structure their information. Here’s a breakdown of their differences and best use cases.
Lookup Tables
What They Are:
Lookup Tables (or fixture data) store static reference data in a structured format, such as a list of districts, services, or product inventories.
Key Features:
Used for dropdown menus in forms (e.g., selecting a village from a district).
Can be updated via Excel or API.
Data is not linked to user permissions—any user can access it.
Supports parent-child relationships (e.g., Country → State → City).
When to Use:
✅ When the data doesn’t change frequently.
✅ When multiple users need access to the same dataset.
✅ When user access restrictions are not required.
Locations
What They Are:
Locations define a hierarchy of geographic or organizational units within a project (e.g., Country → State → District → Facility). They help manage case ownership and user assignments.
Key Features:
Used to assign users and restrict data access.
Integrated with mobile worker permissions.
Users see only data relevant to their assigned location.
Supports hierarchical relationships similar to lookup tables.
When to Use:
✅ When data needs to be linked to specific users or groups.
✅ When you need to restrict access to location-based data.
✅ When managing case ownership at different geographic or organizational levels.
Key Differences
Feature | Lookup Tables | Locations |
---|
Feature | Lookup Tables | Locations |
---|---|---|
Primary Use | Reference data for dropdowns | Organizational structure & case ownership |
Data Ownership | Shared across users | Assigned to specific users |
Access Control | No restrictions | Restricted based on location |
Hierarchy Support | Yes (parent-child relationships) | Yes (multi-level hierarchy) |
Editable in Mobile | No | No (but affects mobile workers' data access) |
Which One Should You Use?
If you need simple reference data without user-based access, go with Lookup Tables.
If you need to control access to data and assign users based on geography or organization, use Locations.