Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Workspaces are the place where users in the same organization collaborate and share maps. A user may form part of several workspaces but, at the very least, always forms part of one.
API tokens are created per-workspace. If you wish to interact with several workspaces via the Felt API, you must create a different API token for each one.
Creating a new map is as simple as making a POST
request to the maps endpoint.
Notice in the response the "id"
property. Every map has a unique ID, which is also a part of the map's URL. Let's take note of it for future API calls.
Also part of the response is a "url"
property, which is the URL to your newly-created map.
Performing a GET
request to a map URL will give you useful information about that map, including title, URL, layers, thumbnail URL, creation and visited timestamps.
To remove a map from your workspace, simply perform a DELETE
request to the map's URL:
To move a map to a different folder or project, send a POST request to the map's move URL:
There are a variety of ways to interact with Felt’s modern GIS platform outside of the user interface. They can be grouped into two buckets: tools for programmatically creating and modifying maps, and tools for building custom experiences for map viewers. These tools can be used to solve distinct challenges and also be used in tandem with one another.
Felt’s user interface allows a large amount of customization, offering the ability to generate complex cartographic designs, adding components to create a dashboard, and much more.
All Felt API endpoints are hosted at the following base URL:
All calls to the Felt API must be authenticated. The easiest way to authenticate your API calls is by creating a API token and providing it as a Bearer
token in the request header.
Learn more about API tokens here:
The easiest way to interact with the Felt API is by using our felt-python
SDK. You can install it with the following command:
Creating a new map is as simple as making a POST request to the maps endpoint.
Notice in the response the "id"
property. Every map has a unique ID, which is also a part of the map's URL. Let's take note of it for future API calls.
Also part of the response is a "url"
property, which is the URL to your newly-created map. Feel free to open it! For now, it should just show a blank map.
Now that we've created a new map, let's add some data to it. We'll need the map_id
included in the previous call's response.
Like maps, layers also have unique identifiers. Let's take note of this one (also called "id"
in the response) so we can style it in the next call.
You can see the uploaded result in your map:
Layers can be styled at upload time or afterwards. Let's change the style of our newly-created earthquakes layer so that points are bigger and in green color:
Go to your map to see how your new layer looks:
A layer must have finished uploading successfully before it can be refreshed
Now go to your map and see if any new earthquakes have occured!
Like maps, layers also have unique identifiers. Make sure to take note of them for subsequent calls, like styling a layer or removing it.
Files aren't uploaded to the Felt app — instead, they're uploaded directly to Amazon S3. Therefore, creating a layer from a file on your computer is a two-step process:
Perform a POST
request to receive an S3 presigned URL which you can later upload your files to:
You can check the upload status of a layer by querying it:
Elements live at the top layer of a map, and are created directly inside the Felt app.
Elements live at the top layer of a map, and are created directly inside the Felt app.
Returns a list of GeoJSON Feature Collections, one for each element group.
Each element is represented by a feature in the POST
ed GeoJSON Feature Collection.
For each feature, including an existing element ID (felt:id
) will result in the element being updated on the map. If no element ID is provided (or a non-existent one) , a new element will be created.
Elements can be deleted by referencing them by ID
A great way of building data-driven apps using Felt is by triggering a workflow whenever something changes on a map, like someone drawing a polygon around an area of interest or updating the details on a pin.
Instead of polling by listing elements, comments or data layers on a fixed interval, a better alternative is to set up a webhook where Felt will send a notification any time a map is updated. This allows you to build integrations on top, such as sending a Slack message or performing calculations for the newly-drawn area.
Two things are needed in order to make use of webhooks:
A Felt map which will serve as the basis for the webhook. Updates will be sent whenever something on this map changes.
A webhook URL where the updates will be sent in the form of POST
requests.
Simply click on Create a new webhook
, select a map to listen to changes and paste in a webhook URL where the updates will be sent to.
In order to use webhooks effectively, a receiving layer must be set up to trigger actions based on the updates sent by the Felt API. Here are some examples of how to set up a webhook using Felt and an external service.
Felt’s allows editors to interact with the Felt platform via code, performing actions such as creating new maps, adding data to maps, styling layers, and more. The REST API can be leveraged from any environment that is capable of sending GET and POST requests.
For Python users, interactions with the REST API are simplified through the module, which can be installed with pip
and used to call the REST API endpoints directly from Python functions.
However, sometimes application developers need further control over the experience of viewing and/or interacting with a map. For example, they may want to run custom logic after a user clicks on a feature in a layer, or animate data on the map based on other types of user input elsewhere on the webpage. For these situations and many more, Felt’s allows developers to load a Felt map into their application while providing opportunities to hook into the interaction loop and more broadly customize the experience of the map viewer.
You can create an API token in the :
Felt supports . In this case, we'll import all the recent earthquakes from :
Layer styles are defined in the , a JSON-based specification that allows customizing a layer's style, legend, label and popups.
Similar to , refreshing an existing URL layer is just a matter of making a single POST
request:
Felt supports a myriad of formats, both as files and hosted URLs, up to a limit of 5GB. Check out the full list .
The easiest way of uploading data into a Felt map via the API is to import from a URL. Here's an example importing all the recent earthquakes from :
Combining elements with is a great way to create interactive data apps in Felt.
Elements are returned as a .
Workspace admins can set up webhooks in the .
is an easy way to collect webhook requests and even run custom code as a result.
In Felt, navigate to the and click on Create new webhook
Navigate to the and click on Create new webhook
Returns a GeoJSON FeatureCollection containing all the elements in a map that are not in an element group
Returns a GeoJSON FeatureCollection containing all the elements in a single group
Returns a list of GeoJSON Feature Collections, one for each element group
Each element is represented by a feature in the POST
ed GeoJSON Feature Collection.
For each feature, including an existing element ID will result in the element being updated on the map. If no element ID is provided (or a non-existent one) , a new element will be created.
For each Element Group, including an existing Element Group id
will result in the Element Group being updated. If no id
is provided, a new Element Group will be created.
You may assign Elements to an Element Group by setting the felt:parentId
property of an Element to the ID of an Element Group.
Deletes the element with the ID specified in the path.
Several aspects can be customized when creating a new map, including:
Title
Initial location (latitude, longitude and zoom level)
Sharing permissions (defaults to viewing and commenting for users with the map URL)
An array of URLs to import on map creation
Returns details including title, URL, thumbnail URL, creation and visited timestamps.
Update the title, description and access permissions of a map.
Move a map to a different project or folder. Project IDs and Folder IDs can be found inside map settings.
Returns an export of the map comments.
The export can be generated in JSON or CSV format, you can specify the format with the format
query parameter, which can be either json
or csv
. It defaults to json
.
Resolve an individual comment based on it's ID.
The ID of the map to list elements from.
The ID of the map.
The ID of the element group.
The ID of the map to list groups from.
The ID of the map to delete the element from.
The ID of the element to delete.
No Content
The ID of the map to delete
The ID of the map to export comments from.
The format to export the comments in, either 'csv' or 'json' (default)
The ID of the map to create the element in
The ID of the map to create the group in
#C93535
not_provided
Example: OI22G7wJRzOo2p1RQeHIPB
My Element Group
dot
The basemap to use for the new map. Defaults to "default". Valid values are "default", "light", "dark", "satellite", a valid raster tile URL with {x}, {y}, and {z} parameters, or a hex color string like #ff0000.
A description to display in the map legend
If no data has been uploaded to the map, the initial latitude to center the map display on.
An array of urls to use to create layers in the map. Only tile URLs for raster layers are supported at the moment.
If no data has been uploaded to the map, the initial longitude to center the map display on.
The level of access to grant to the map. Defaults to "view_only".
The title to be used for the map. Defaults to "Untitled Map"
The workspace to create the map in. Defaults to the latest used workspace
If no data has been uploaded to the map, the initial zoom level for the map to display.
The ID of the map to update
A description to display in the map legend
The level of access to grant to the map. Defaults to "view_only".
The new title for the map
It's common to have data update on a regular basis, such as every week or every month. Instead of having to re-upload and style the new data, it can be very convenient to simply refresh a layer using a new data source.
A layer must have finished uploading successfully before it can be refreshed
Perform a POST
request to receive an S3 presigned URL which you can later upload your files to:
Here is an example of a simple visualization, expressed in FSL:
A layer's FSL can be retrieved by performing a simple GET
request to a layer's endpoint:
To update a layer's style, we can send a POST
request with the new FSL to the same layer's /update_style
endpoint.
List projects that you have access to.
Returns details of a project including name, visibility, id and a list of references to the maps in the project.
Create a new project with the provided name and visibility.
Update a project's name or visibility.
Delete a project and all of it's contents.
Caution: Deleting a project deletes all of the folders and maps inside!
The /upload
endpoint can be used for both URL and file uploads:
For URL uploads, simply making a single POST
request to the upload endpoint is enough
For file uploads, the response of the initial POST
request will include a target URL and some pre-signed attributes, which will be used to upload the new file to Amazon S3.
This endpoint is used to create a layer, and obtain a pre-signed url to upload the layer files to S3.
Layer files aren’t uploaded directly to the Felt API. Instead, they are uploaded by your client directly to an S3 bucket.
You will receive a single set of URL and pre-signed params to upload the file. Only a single file may be uploaded — if you wish to upload several files at once, consider wrapping them in a zip file.
To upload the file using the pre-signed params, you must perform a multipart upload, and include the file contents in the file
field:
After uploading a file or URL, you may want to update the resulting layer with some new data. The process is quite similar to the above:
For URL uploads, simply making a single POST
request to the refresh endpoint is enough
For file uploads, the response of the initial POST
request will include a URL and some presigned attributes, which will be used to upload the new file to Amazon S3. See Uploading the file to Amazon S3 for more details.
A layer's style may be updated by providing a new Felt Style Language object. Learn more in the guide:
Get the details of all the layer on the map that are not within a layer group.
Get the details of a single layer on the map. These details include:
Name of the layer
Upload status and progress
Other metadata, such as the geometry type, visibility state in the legend, etc
Update a layer's name or move it into or out of a layer group.
To move a layer into a group, set the layer_group_id
to the id
of the layer group you want to move the layer into. To move a layer out of a group, set the layer_group_id
to null
.
Provide an array of layer group objects to create new groups or update existing ones.
For each layer group object, including an existing ID will result in the group's details (name, subtitle and legend order) being updated. If no layer group ID is provided (or a non-existent one is provided), a new layer group will be created.
List all layers in your workspace's Library
Publish a layer to your workspace's library
Publish a layer group to your workspace's library
Get a link to export a layer as a GeoPackage (vector layers) or GeoTIFF (raster layers)
Create an export request of a layer as a GeoPackage, GeoJSON, or CSV. Optionally include filters with the layer. Export requests are asynchronous. A successful response will return a poll_endpoint to check the status of the export.
Check the status of an Custom Export. If successful, the response will include a download_url
Duplicate an array of layers or layer groups to a map
The Felt SDK allows you to embed, connect to and ultimately control Felt map embeds, allowing you to build powerful, interactive custom applications around Felt.
You are able to control many aspects of the Felt UI and map contents, as well as being notified of events happening in the map such as clicks, selections, etc.
Assuming you are using a modern JavaScript environment, install the SDK using your preferred package manager:
Create an HTML page with a container element:
Embed a Felt map in your container element and use the SDK to control it:
The Felt SDK provides several methods for reading data from your map's entities (layers, elements, etc.) and staying in sync with their changes. This guide will show you how to read entity data and react to changes.
The Felt SDK provides both singular and plural versions of getter methods for various entity types. This allows you to retrieve either a single entity or multiple entities of the same type.
For example, for layers:
All of the methods that return multiple entities accept constraint parameters to filter the results:
To stay in sync with entities, use the appropriate on[EntityType]Change
method. For example, to monitor layer changes:
Cleanup: Always store and call the unsubscribe functions when you're done listening for changes so that the listener doesn't continue to run in the background.
Error handling: When getting entities, remember that the methods may return null
if the entity doesn't exist:
Batch operations: When you need multiple entities, use the bulk methods (getLayers
, getElements
, etc.) with constraints rather than making multiple individual calls:
This approach to reading and monitoring entities gives you full control over your map's data and allows you to build interactive applications that stay in sync with the map's state.
All calls to the Felt API require authorization using a Bearer
token in the request header:
These are tokens associated to your account only, and that you have to manually provide to the application you want to use.
Since these tokens grant access to your account, you must store them securely and treat them as a password to your account.
Be sure to take note of the token before closing the dialog; you won’t have a second chance to view it.
Once you have your API token, you can authenticate your requests to the Felt API by using it as a bearer token in your Authorization
header:
Here's an example showing how to create a new Felt map:
The Felt SDK has a number of methods for interacting with maps, depending on how you set up your HTML.
All Felt maps are embedded in iframes, and the SDK can do this for you or can connect to an existing Felt iframe.
Felt map IDs are unique identifiers for Felt maps. They are used to embed maps in iframes, and to connect to existing iframes.
To get the ID of a Felt map, click the Map settings button in the main toolbar, and then you can see the Map ID in the Developers section.
Alternatively, you can look at the URL of the map. For example, the map at https://felt.com/map/Map-title-xPV9BqMuYQxmUraVWy9C89BNA
has the ID xPV9BqMuYQxmUraVWy9C89BNA
.
Throughout the documentation, we'll use the placeholder FELT_MAP_ID
to refer to a Felt map ID.
Felt.embed
to create an iframeCreate an HTML page with a container element:
Embed a Felt map in your container element and use the SDK to control it by calling Felt.embed
, passing the container element as the first argument:
Felt.embed
to mount into an existing iframeIn some cases, you may want to add a "template" iframe to your page. This can be useful if you want to style your iframe in a specific way, or if you already have one map embedded and want to mount and control a different map.
In this case, you can call Felt.embed
with the iframe element as the first argument:
Felt.connect
to connect to an existing embedded Felt mapThere may be cases where you already have a Felt map embedded in an iframe, and you want to control it using the SDK. This can be useful if your HTML is server-rendered with the Felt map already embedded.
In this case, you can call Felt.connect
with the iframe's window as the first argument:
Note that in this case, you don't need to pass the Felt map ID to Felt.connect
, because we are connecting to a map that has already been embedded.
Creates a short-lived (15 minutes) token for authenticating a visitor to view a private embedded map view without being logged into Felt.
Generate a token by making a call to this API from your server
Securely pass the token to your frontend client
Include the token as a query parameter on the Embed URL in an iframe
<iframe src="https://felt.com/embed/map/{mapId}?token={token}"></iframe>
You can allow EmbedToken based page views to export layer data.
Include a user_email
query parameter when creating the EmbedToken
Turn on "Viewer permissions: Export data" in Map settings
Just like , refreshing a layer with a new file is a two-step process:
Similar to , refreshing an existing URL layer is just a matter of making a single POST
request:
A layer's style is defined in a JSON-based called , or FSL for short. Editors can view the current style of a layer inside a Felt map by clicking on Actions > Edit styles
in a layer's overflow menu (three dots).
You can find examples of FSL for different visualization types in of these docs:
: same color and size for all features (vector) or pixels (raster).
: different color per feature or pixel, based on a categorical attribute
: different color or size per feature or pixel, based on a numeric attribute.
: a density-based visualization style, for vector point layers.
: a special kind of visualization for raster elevation layers.
Check our docs to see what URLs are supported.
Style, expressed in the
This feature is available to customers on the . Reach out to our team to .
For more information on how to control a map, see .
If you are building a React application, you can use the to get started quickly.
You can also read our guide on to learn more about how to use the Felt SDK with React.
You can create an API token in the :
The Felt SDK allows you to filter which features are visible in a layer using expressions that evaluate against feature properties. Filters can come from different sources and are combined to determine what's visible.
By understanding how filters work and combine, you can create dynamic views of your data that respond to user interactions and application state.
Layer filters can come from multiple sources, which are combined to create the final filter:
Style filters: Set by the map creator in the Felt UI
Component filters: Set through interactive legend components
Ephemeral filters: Set temporarily through the SDK
You can inspect these different filter sources using getLayerFilters
:
Use setLayerFilters
to apply ephemeral filters to a layer:
The following operators are available:
Comparison: lt
(less than), gt
(greater than), le
(less than or equal), ge
(greater than or equal), eq
(equal), ne
(not equal)
Text: cn
(contains), nc
(does not contain)
Boolean: and
, or
Lookup: in
(contained in list), ni
(not contained in list)
You can combine multiple conditions using boolean operators:
Here's a common use case where we filter a layer to show only features that match a property of a selected feature:
Clear filters: Set filters to null
to remove them entirely:
Check existing filters: Remember that your ephemeral filters combine with existing style and component filters:
Type safety: Use TypeScript to ensure your filter expressions are valid:
The ID of the map hosting the layer to refresh
The ID of the layer to refresh
The ID of the map to delete the layer from
The ID of the layer to delete
No Content
The ID of the map to delete the layer group from
The ID of the layer group to delete
No Content
Defaults to listing library layers for your "workspace". Use "felt" to list layers from the Felt data library. Use "all" to list layers from both sources.
workspace
Available options: The ID of the map where the layer is located
The ID of the layer to export
The ID of the map where the layer is located
The ID of the layer to export
The ID of the export
See for more details on filter operators.
The ID of the map to upload the layer to.
A list of hints for interpreting the data in the upload.
A public URL containing geodata to import, in place of uploading a file.
(Image uploads only) The latitude of the image center.
(Image uploads only) The longitude of the image center.
The display name for the new layer.
(Image uploads only) The zoom level of the image.
The ID of the map where the layer is located
The ID of the layer to update the style of
The new layer style, specified in Felt Style Language format
not_provided
Example: A very interesting dataset
nOFB8jOFSne1DuDr2uFn4A
not_provided
Example: KFFhKAbvS4anD3wxtwNEpD
not_provided
Example: My Layer
Deprecated: use caption
instead.
not_provided
Example: A very interesting group
not_provided
Example: OI22G7wJRzOo2p1RQeHIPB
My Layer Group
Deprecated: use caption
instead.
The ID of the map where the layer is located
The ID of the layer to publish
The name to publish the layer under
not_provided
Example: My Layer
The ID of the map where the layer group is located
The ID of the layer group to publish
The name to publish the layer group under
not_provided
Example: My Layer Group
The ID of the map where the layer is located
The ID of the layer to export
Send an email to the requesting user when the export completes. Defaults to true
Filters for the layer in specified in Felt Style Language filter format
csv
Available options: The Felt SDK provides methods to control the visibility of various entities like layers, layer groups, element groups, and legend items. These methods are designed to efficiently handle bulk operations.
All visibility methods use a consistent structure that allows both showing and hiding entities in a single call:
Control visibility of layer groups using setLayerVisibility
:
Control visibility of layer groups using setLayerGroupVisibility
:
Similarly, control element group visibility with setElementGroupVisibility
:
Legend items require both a layer ID and an item ID to identify them. Use setLegendItemVisibility
:
To focus on a single layer by hiding all others, first get all layers and then use their IDs:
When implementing a toggle, you can use empty arrays for the operation you don't need:
Batch operations: Use a single call with multiple IDs rather than making multiple calls:
Omit unused properties: When you only need to show or hide, omit the unused property rather than including it with an empty array:
This guide covers common patterns and concepts used throughout the Felt SDK.
By following these patterns consistently throughout the SDK, we aim to make the API predictable and easy to use while maintaining flexibility for future enhancements.
All methods in the Felt SDK are asynchronous and return Promises. This means you'll need to use await
or .then()
when calling them:
The SDK follows a consistent pattern for getting entities. For each entity type, there are usually two methods:
A singular getter for retrieving one entity by ID:
A plural getter that accepts constraints for retrieving multiple entities:
The plural getters allow you to pass no constraints, in which case they'll return all entities of that type:
Each entity type has a corresponding change listener method following the pattern on{EntityType}Change
:
This is true of selection, which acts as a kind-of entity in its own right:
All change listeners return an unsubscribe function that should be called when you no longer need the listener:
This is particularly important in frameworks like React where you should clean up listeners when components unmount:
Change listeners always take a single object parameter containing both options
and handler
. This structure makes it easier to add new options in the future without breaking existing code:
When dealing with mixed collections of entities (like in selection events), each entity is wrapped in an EntityNode
object that includes type information:
The Felt SDK provides methods to read and react to selection changes in your map. Selection refers to the currently highlighted entities (elements, features, etc.) that the user has clicked on or selected through other means.
You can get the current selection state using getSelection()
:
This returns an array of EntityNode
objects, each representing a selected entity. The selection can include various types of entities at the same time, such as elements and features.
To stay in sync with selection changes, use the onSelectionChange
method:
Clean up listeners: Always store and call the unsubscribe function when you no longer need to listen for selection changes:
Handle empty selection: Remember that the selection array might be empty if nothing is selected:
By following these patterns, you can build robust interactions based on what users select in your Felt map.
The popup block contains information on how the popup is displayed and which attributes to show.
These are the fields that each popup block can contain:
titleAttribute
Optional. The attribute that will be used to title the popup if available
imageAttribute
Optional. The attribute that will be used to populate the popup image if available
popupLayout
Optional. One of either “table” or “list”. The way the popup will show its contents. Defaults to "table"
keyAttributes
Optional. A list of attributes to show in the popup following the order defined here. If it’s not defined, only attributes with a value will show in the popup. If it’s defined, all attributes here will show even if the selected feature doesn’t include them.
The Felt Style Language is a way for advanced users to style data quickly, through simple JSON code. This document is a guide to use as you test data files in Felt. For an exhaustive guide to all supported properties see the Reference Documentation.
Learn how to define and configure the code blocks that compose the Felt Style Language
Learn about visualization types, including simple, categorical, numeric (color by & size by), heatmaps and hillshade.
Details on how to customize legends on a per-layer basis.
Definitions for errors raised when validating the Felt Style Language.
To work with Felt embeds in React, we have a starter template that you can use as a starting point.
In that repo, you will find a feltUtils.ts
file that demonstrates some ways to make using the Felt SDK in React easier.
useFeltEmbed
A common use case for building apps on Felt is to be notified when entities are updated. The main example of this is when you want to change the visibility of say a layer, and have your own UI reflect that change.
Rather than keeping track of the visibility of entities yourself, you can use the Felt SDK to listen for changes to the visibility of entities.
Here is an example of how you might do this for layers assuming you already have a reference to a Layer
object, e.g. from calling map.getLayers()
:
The legend block contains information on how the legend will be displayed for this visualization.
These are the fields that each legend block can contain:
A style in its most basic form contains a version definition but it can be extended to define how we want geometry and labels to show on the map, how the legend should look like, what information is shown in popups and the formatting used when displaying feature properties.
The table below describes which properties can be used in the style definition.
Lists Sources that you have access to. To see the datasets in an individual Source, fetch it from the Show Source endpoint.
Fetches a single Source and all the datasets it contains.
Triggers the creation of a new Source. Each connection type has it's own set of required parameters, depending on what kind of Source it is. These are provided in the connection
field.
Connecting the Source and inspecting it's datasets will happen asynchronously after the API response is returned. To determine when the inspection process has completed, poll the Show Source endpoint and check for sync_status: completed
.
Triggers Felt to reconnect to your Source and re-inspect all the datasets it contains. Syncing will happen asynchronously after the API response is returned. To determine when the inspection process has completed, poll the Show Source endpoint and check for sync_status: completed
.
Updates details of a Source. Connecting the Source and inspecting it's datasets will happen asynchronously after the API response is returned. To determine when the inspection process has completed, poll the Show Source endpoint and check for sync_status: completed
.
Deletes a Source.
Any layers created from the Source will remain after it is deleted, but they will no longer be refreshed.
Categorical visualizations use a categorical attribute and the categories within it to apply styling to discrete categories of the attribute. On raster datasets, it uses a band instead of an attribute.
Categorical visualizations are defined using "type": "categorical"
and, for every supported style and label property used, either a single value that will apply to all categories or an array of different values for each category.
The Global Power Plants layer in Felt is an example of a categorical layer on a vector dataset
and is defined by the following style
Notice that we are saying that the primary_fuel
data attribute will be used to categorize elements and that the possible values of that attribute that we are interested in are "Nuclear"
, "Oil"
, "Coal"
, "Gas"
, "Wind"
, "Hydro"
and "Solar"
. Also notice that we are defining either a single value that will apply to all categories (i.e. size
) or a value for each category (i.e. color
)
The Cropscape CDL layer in Felt is an example of a categorical layer on a raster dataset
and is defined by the following style:
Properties common to all visualization types.
The following properties are available for the simple
type of visualization
These are the properties available to define label rendering. Anchors can be lines or points, polygons are not supported.
The Felt SDK provides methods to control the map's viewport (the visible area of the map) and handle user interactions like clicking and hovering on the viewport.
You can get the current viewport state using getViewport()
:
There are two main ways to set the viewport: moving to a specific point, or fitting to bounds.
Use setViewport()
to move the map to a specific location:
Use fitViewportToBounds()
to adjust the viewport to show a specific rectangular area:
To stay in sync with viewport changes, use the onViewportMove
method:
Listen for click events on the map using onPointerClick
:
Track mouse movement over the map using onPointerMove
:
Cleanup: Always store and call unsubscribe functions when you're done listening for events:
Throttling: For pointer move events, consider throttling your handler if you're doing expensive operations:
By using these viewport controls and interaction handlers, you can create rich, interactive experiences with your Felt map.
This is a sample application showing how to use the Felt SDK to build an app with the following features:
listing the map's layers
toggling layer visibility
moving the viewport to center it on predefined city locations
The commented code in its entirety is shown below.
The attributes block contains information on how attributes will be shown both on the popup and the table. Each attribute definition can contain the following properties:
The filters block contains information on how the layer is being filtered before displaying. In order for a feature to be shown on the map it must evaluate the filter expression to true
.
Filters are written using a JSON infix notation that looks like one of [identifier, operator, operand]
, true
or false
.
Valid identifiers are either a feature property or a nested expression.
Valid operators are:
"lt"
– Less than
"gt"
– Greater than
"le"
– Less than or equal to
"ge"
– Greater than or equal to
"eq"
– Equal to
"ne"
– Not equal to
"and"
– And, cast to boolean
"or"
– Or, cast to boolean
"cn"
– Contains the operand, cast to string
"nc"
– Does not contain the operand, cast to string
"in"
– Contained in the operand list
"ni"
– Not contained in the operand list
"is"
– Used to match against null values
"isnt"
– Used to match against null values
Operands are:
A numerical value, a string value, a boolean value
An array of numerical, string, or boolean values, a shorthand expanded to these patterns:
Input 1: [id, "in", [element1, …, elementN]
Expansion 1: id
is equal (”eq”
) to one of more of the elements
Input 2: [id, "ni", [element1, …, elementN]
Expansion 2: id
is not equal (”ne”
) to any of the elements
Not defined for operators other than "in"
and "ni"
A nested expression
In cases of type mismatch cast the identifier value to the operand’s type
Type casting applies element-wise to lists with "in"
and "ni"
operators
Simple visualizations are those that show each feature in a vector dataset using the same style or the image as it is in raster ones.
Simple visualizations must define "type": "simple"
and a single value for each supported style and label properties.
The Airports layer in Felt is an example of a simple visualization using a vector dataset
and is defined by the following style:
This is an example of a simple visualization using a raster dataset
and is defined by the following style:
Hillshades are used to visualize the valleys and ridges encoded in elevation raster data.
Hillshade visualizations are defined using “type”: “hillshade”
and, support the following properties:
The following map is an example of a raster layer using a hillshade visualization
with the following style
Felt also supports adding color to hillshades by defining a color
property in the style
which is defined by the following style
This is available on GitHub in the repository.
Take a look at to see how it works.
displayName
Optional. In categorical visualizations, a dictionary that maps from each category to what will be displayed on the legend. In or visualizations, a dictionary that maps from the index of each class to what will be displayed on the legend.
version
Mandatory. Defines which version this style adheres to
type
Optional. One of , , or . Defaults to simple.
config
Optional. A block that contains some configuration options to be used across the style. .
style
Optional. An object that defines how the data will be drawn.
label
Optional. An object that defines how the labels will be drawn.
legend
Optional. Defines how this layer will be shown on the layer panel.
popup
Optional. Defines how the popup is shown and what’s included.
attributes
Optional. Defines how attributes are shown both in the popup and the table.
filters
Optional. A data filter definition that defines which data will be rendered.
Manage your with Felt REST API.
See for the default values of these attributes on each geometry type.
The following properties are available for the categorical
and numerical
type visualizations. (You can see an example of a categorical
visualization )
See of these attributes on each label type.
If using a categorical
or numeric
visualization, all the previous properties must be arrays. If there's a single value in the array, that value is used in all categories. If there are as many values as categories
, the corresponding value will be used for each category. You can see an example of a categorical
viz .
isClickable
boolean
true
Optional. A flag to tell if features should be clickable
isHoverable
boolean
false
Optional. A flag to tell if features should be hoverable
isSandwiched
boolean
true
Optional. A flag to tell if features affected by this visualization need to be rendered below the basemap road and water layers. Only applies to polygon features, point and line features are already rendered on top of the basemap
maxZoom
number
22
Optional. The maximum zoom level at which the visualization will be shown
minZoom
number
0
Optional. The minimum zoom level at which the visualization will be shown
renderAsLines
boolean
false
Optional. Decides if a polygon dataset should be render as lines thus making them render above the basemap. Note that using this requires that the style uses line properties instead of polygon ones.
color
"#EE4D5A"
"#826DBA"
"#4CC8A3"
highlightColor
"#EA3891"
"#EA3891"
"#EA3891"
highlightStrokeColor
"#EA3891"
"#EA3891"
"#EA3891"
dashArray
-
-
lineCap
-
-
"round"
lineJoin
-
-
"round"
opacity
0.9
0.8
1
isSandwiched
-
false
-
size
4
-
2
strokeColor
"#F9F8Fb"
"#777777"
-
strokeWidth
1
1
-
color
An array of colors that will be used in the hillshade. From less elevation to more.
source
The light angle. 0 is North, 90 East, …
intensity
Controls the intensity of a hillshade.
The config block contains configuration options for a given visualization.
These are the fields that each config block can contain:
band
Optional. Used in raster numeric visualizations. The raster band that we’ll use to get the data from.
categoricalAttribute
Mandatory for categorical visualizations. The attribute that contains the categorical attributes that will be used.
categories
Mandatory for a categorical visualization. An array of the values that will be used as categories. Categories will be rendered from top to bottom following the definition order.
labelAttribute
Optional. Defines which dataset attribute or attributes to use for labeling. If multiple values are provided, the first available one will be used.
method
Optional. Used in raster algebra numeric visualizations. The type of operation and the bands used for it.
noData
Optional. Used in raster visualizations. Defines values that won’t be shown
numericAttribute
Mandatory for a numeric visualization. The attribute that contains the numeric values used.
otherOrder
Optional. Used in categorical visualizations. It can be set to either “below” or “above” to make features that do not match any of the defined categories render below or above the other ones. The default position is “below”.
rasterResampling
Optional. Used on raster data visualizations. It can be set to either “nearest” or “linear”. Defaults to “nearest”
showOther
Optional. Used in categorical visualizations. If this field is set to true it will show all features that do not match any of the defined categories and add an extra entry as the last item in the legend.
steps
Mandatory for a numeric visualization. An array of values that are either step based depending on the classification method and number of steps chosen or, for a continuous visualization, the min and max values from the numericAttribute.
rasterResampling
-
-
-
“nearest”
highlightColor
"#EA3891"
"#EA3891"
"#EA3891"
-
highlightStrokeColor
"#EA3891"
"#EA3891"
"#EA3891"
-
dashArray
-
-
-
lineCap
-
-
"round"
-
lineJoin
-
-
"round"
-
opacity
0.9
0.8
1
-
isSandwiched
-
false
-
-
size
4
-
2
-
strokeColor
"#F9F8Fb"
"#777777"
-
-
strokeWidth
1
1
-
-
color
Points, lines and polygons
Optional. The color to be used
dashArray
number[]
Lines
Optional. The dash line definition
highlightColor
string
Points, lines and polygons
Optional. The color to be used when a feature is selected
highlightStrokeColor
string
Points, lines and polygons
Optional. The stroke color to be used when a feature is selected
lineCap
Lines
Optional. The shape used to draw the end points of lines
lineJoin
Lines
Optional. The shape used to join two line segments when they meet
opacity
Points, lines and polygons
Optional. The opacity to use from 0 to 1
size
Points and lines
Optional. Point radius or line width in pixels
strokeColor
Points and polygons
Optional. Stroke color
strokeWidth
Points and polygons
Optional. Stroke width in pixels
color
Points, lines and polygons
Optional. The color to be used
dashArray
number[]
Lines
Optional. The dash line definition
lineCap
Lines
Optional. The shape used to draw the end points of lines
lineJoin
Lines
Optional. The shape used to join two line segments when they meet
opacity
Points, lines and polygons
Optional. The opacity to use from 0 to 1
size
Points and lines
Optional. Point radius or line width in pixels
strokeColor
Points and polygons
Optional. Stroke color
strokeWidth
Points and polygons
Optional. Stroke width in pixels
color
Points and lines
Optional. The label color
fontFamily
string
Points and lines
Optional. The font family to use
fontSize
Points and lines
Optional. The font size in pixels
fontStyle
normal | italic | oblique
Points and lines
Optional. The font style
fontWeight
number
Points and lines
Optional. The font weight
haloColor
Points and lines
Optional. The label halo color
haloWidth
Points and lines
Optional. The label halo color width
letterSpacing
number
Points and lines
Optional. Horizontal spacing behaviour between text characters
lineHeight
number
Points and lines
Optional. Sets the height of a line box
maxLineChars
Points and lines
Optional. Defines the max number of characters before a line break
maxZoom
number
Points and lines
Optional. The maximum zoom level at which the label will be shown
minZoom
number
Points and lines
Optional. The minimum zoom level at which the label will be shown
offset
[number, number] | number
Points and lines
Optional. In the case of points, this value must be an array of two numeric offsets that will be applied on the positive X and Y axis defined by the label placement (i.e. an offset of [3,4] with a label placement of NE moves the label 3pixels to the right and 4 pixels above of the anchor point. An offset of [3,4] with a label placement of SW moves the label 3pixels to the left and 4 pixels below of the anchor point). In case of lines, this value is a single number that moves the label following the label position (i.e. an offset of 3 with a label position of Above will move the label 3 pixels above the line following the line normal. An offset of 3 with a label position of Below will mode the label 3 pixels under the line following the line normal)
padding
number
Points and lines
Optional. Adds invisible padding around the label that's used to compute label collisions
placement
TextPlacements[] | auto | LineLabelPlacement
Points and Lines
Optional. If defined on a points dataset, an array of label placements to test when placing the label. If all the placements collide with already existing labels, the label is not shown. If defined on a lines dataset, the label placement on a line
renderAsLines
boolean
Polygons
Optional. Renders labels along lines instead of using the centroids
repeatDistance
number
Lines
Optional. The distance in pixels between label repetitions on a line
textTransform
capitalize | uppercase | lowercase
Points and lines
Optional. Specifies how to capitalize the label
isClickable
boolean
Points, Lines and Polygons
Optional. A flag to tell if labels should be clickable
isHoverable
boolean
Points, Lines and Polygons
Optional. A flag to tell if labels should be hoverable
Numeric visualizations use a numeric attribute to either vary colors or sizes between ranges of values and are defined in FSL with "type: numeric"
.
Jenks Natural Breaks
Finds natural groupings in attribute values to minimize differences within steps and maximize differences between them. A good option for clustered data. This is the default method in Felt.
Equal interval
Divides attribute values into an equal number of steps. Equal interval is a good option when data is equally spread across the range of values.
Quantiles
Places an equal number of values into each step. A good option for evenly distributed data.
Mean Standard Deviation
Steps are calculated based on an attribute values distance from the mean. A good option for data that follows a near-normal distribution.
Manual
Manually define which values to include in each size or color step. A good option when you know your data well.
Continuous
Attribute values are sized or colored based on a continuous scale between the min and max values versus discrete steps. A good option for continuous data.
Color numeric
values in your data using the color
property.
Stepped color
This map shows the percent of renter occupied housing units by US county. Each county is colored according to the step of ranges it falls into using a sequential color palette where light colors are assigned to low values and darker colors for high values.
The style for this map is defined
with the visualization type: numeric
numericAttribute: "Renter occupied (%)"
and the computed steps
using Jenks Natural Breaks over 5 classes
the sequential color
array where each color is applied to a distinct step
range
Continuous Color
The map below shows average accumulated precipitation in the State of California between the years 1900 - 1960 and is colored along a continuous range between the min and max of the precipitation values.
In this case, the numeric style is applied using a continuous method.
In the config
block, the numericAttribute: PRECIP
has steps
that range between the min and max values of [2.5,125]
The color
property has an array of three colors ["#fde89b", "#f37b8a", "#4d53b3"]
that are interpolated to give each precipitation value a unique color
Felt also supports stepped and continuous color numeric visualizations on raster datasets like you can see on the map below
which includes a raster layer with the following style
Notice that in the config
block we are using band
instead of numericAttribute
to define which band will be used for display
Size numeric
values in your point or line data using the size
property.
Stepped size
The map below shows earthquakes over the past year sized with 5 manually defined steps.
In this case, the numeric style is applied using a classed method.
In the config
block, the numericAttribute: mag
has manually-defined steps
for 5 classes
The size
property has an array of five sizes, one for each defined class
Continuous size
To do this the min and max steps
from the numericAttribute: tonnes
are interpolated to be proportionately sized between a min and max point size — size:[5,48]
An example of a layer changing feature colors depending on the zoom level can be found below
On zoom levels lower than 14
, features of this layer will be rendered in red
color. On zoom levels higher than 20
, features of this layer will be rendered in blue
color.
In zooms between 14
and 20
, color will be linearly interpolated between red
and blue.
These are the properties available to define label rendering. Anchors can be lines or points, polygons are not supported.
color
Points and lines
Optional. The label color
fontFamily
string
Points and lines
Optional. The font family to use
fontSize
Points and lines
Optional. The font size in pixels
fontStyle
normal | italic | oblique
Points and lines
Optional. The font style
fontWeight
number
Points and lines
Optional. The font weight
haloColor
Points and lines
Optional. The label halo color
haloWidth
Points and lines
Optional. The label halo color width
letterSpacing
number
Points and lines
Optional. Horizontal spacing behaviour between text characters
lineHeight
number
Points and lines
Optional. Sets the height of a line box
maxLineChars
Points and lines
Optional. Defines the max number of characters before a line break
maxZoom
number
Points and lines
Optional. The maximum zoom level at which the label will be shown
minZoom
number
Points and lines
Optional. The minimum zoom level at which the label will be shown
offset
[number, number] | number
Points and lines
Optional. In the case of points, this value must be an array of two numeric offsets that will be applied on the positive X and Y axis defined by the label placement (i.e. an offset of [3,4] with a label placement of NE moves the label 3pixels to the right and 4 pixels above of the anchor point. An offset of [3,4] with a label placement of SW moves the label 3pixels to the left and 4 pixels below of the anchor point). In case of lines, this value is a single number that moves the label following the label position (i.e. an offset of 3 with a label position of Above will move the label 3 pixels above the line following the line normal. An offset of 3 with a label position of Below will mode the label 3 pixels under the line following the line normal)
padding
number
Points and lines
Optional. Adds invisible padding around the label that's used to compute label collisions
placement
TextPlacements[] | auto | LineLabelPlacement
Points and Lines
Optional. If defined on a points dataset, an array of label placements to test when placing the label. If all the placements collide with already existing labels, the label is not shown. If defined on a lines dataset, the label placement on a line
renderAsLines
boolean
Polygons
Optional. Renders labels along lines instead of using the centroids
repeatDistance
number
Lines
Optional. The distance in pixels between label repetitions on a line
textTransform
capitalize | uppercase | lowercase
Points and lines
Optional. Specifies how to capitalize the label
isClickable
boolean
Points, Lines and Polygons
Optional. A flag to tell if labels should be clickable
isHoverable
boolean
Points, Lines and Polygons
Optional. A flag to tell if labels should be hoverable
color
"#333333"
"#333333"
"#333333"
fontFamily
"Atlas Grotesk LC"
"Atlas Grotesk LC"
"Atlas Grotesk LC"
fontSize
13
13
13
fontStyle
"Normal"
"Normal"
"Normal"
fontWeight
500
400
500
haloColor
"#fbfcfb"
"#fbfcfb"
"#fbfcfb"
haloWidth
1
1
1
justify
“auto”
“auto”
“auto”
letterSpacing
0
0
0
lineHeight
1.2
1.2
1.2
maxLineChars
10
-
10
maxAngle
-
30
-
maxZoom
23
23
23
minZoom
23
23
23
offset
[8, 8]
0
-
padding
2
1
0
placements
“auto”
“Above”
“Center”
repeatDistance
-
250
-
textTransform
"none"
"none"
"none"
Adding a legend block to a visualization makes a legend entry appear for this visualization.
Each legend entry is shown with the geometry type and color defined by the dataset and the visualization block.
While simple visualizations will generate a single legend entry, categorical visualizations will generate a legend entry per category.
The Biodiversity Hotspots layer in Felt has a simple visualization with a legend defined as follows:
The Plant Hardiness Zones layer in Felt has a categorical visualization with a legend defined as follows:
The visual display of numeric legends varies based on the style method (stepped or continuous) and the geometry type (point, line, polygon).
The displayName
can be modified in the legend
block similar to simple and categorical style types.
Heatmap legends are defined as follows:
Notice that the displayName
mapping goes from 0
(left value) to 1
(right value)
Problem: One of the values set in the style has an unsupported value or an invalid type.
Solution: Change the value to be valid.
Error messages:
Attribute 'displayName'
on a legend item of type simple must be a string.
Attribute attribute_name
is not a number.
Attribute attribute_name
is not a string.
Attribute 'lineCap
’ is not a supported value. Supported values are butt, round, square.
Attribute 'lineJoin’
is not a supported value. Supported values are bevel, round, miter.
Attribute ‘dashArray’
has to be a two-numbers array.
Attribute 'offset'
must be either an array of numbers or a number.
Attribute 'placement'
contains a not supported value. Supported values are N
, NE
, E
, SE
, S
, SW
, W
, NW
, Center
.
Attribute 'placement'
contains a not supported value. Supported values are Above
, Center
, Below
.
All values in 'labelAttribute'
must be a string.
Visualization 'type'
definition must be one of simple
, categorical
.
Attribute 'showOther'
must be one of above
, below
.
Problem: The style defines a categorical visualization, but the maps are not showing the layer
Error messages:
Heatmaps are used to visualize the density of points on a map.
Heatmaps visualizations are defined using “type”: “heatmap”
and, allow the following properties to be set:
This is an example of a heatmap visualization
defined with the following visualization
Heatmaps do not support labels.
Interpolators are functions that use the current zoom level to get you a value. The following interpolators are currently supported:
{ "step": [output0, Stops[]] }
: Computes discrete results by evaluating a piecewise-constant function defined by stops on a given input. Returns the output value of the stop with a stop input value just less than the input one. If the input value is less than the input of the first stop, output0
is returned.
Stops are defined as pairs of [zoom, value]
where zoom
is the minimum zoom level where value
is returned and value
can be number | string | boolean
. Note that stops need to be defined by increasing zoom level.
The following image shows the behavior of this definition:
{ "linear": Stops[] }
: Linearly interpolates between stop values less than or equal and greater than the input value
The following image shows the behaviour of this definitions
{ "linear": [number, number] }
: Expands to { "linear": [[minZoom, number], [maxZoom, number] }
Color linear interpolation is done in the HCL color-space
{ "exp": [number, Stops[]] }
: Exponentially interpolates between output stop values less than or equal and greater than the input value. The base parameter controls the rate at which output increases where higher values increase the output value towards the end of the range, lower values increase the output value towards the start of the range, and a base 1 interpolates linearly.
The used value is computed as follows : (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1)
The following images shows the behaviour of this definition
{ "cubicbezier": [number, number, number, number, Stops[]] }
: Interpolates using the bezier curve defined by the curve control points.
The following images shows the behaviour of this definition
displayName
Optional. How this attribute will be shown in different parts of the Felt UI.
format
Optional. A object that encodes how numeric fields should be shown.
string |
"butt"
| "round"
| "square"
|
"bevel"
| "round"
| "miter"
|
number |
number |
string | | auto
number |
string |
"butt"
| "round"
| "square"
|
"bevel"
| "round"
| "miter"
|
number |
number |
string | | auto
number |
string | auto |
number |
string |
string |
number |
Ranges are calculated between 3-10 discrete steps using a or on a continuous scale between an attribute’s min and max values. Felt offers the following methods to symbolize numeric data:
Any time there are fewer colors than values in a numeric style, they are interpolated in the ().
Map link:
The map below shows tonnes of corn that have been exported from Ukraine since August 2022 under the . The symbol size for each country is proportionate to its value in the data.
Most of the properties used on the style
and label
blocks can be defined using interpolators to enable zoom-based styling. Take a look at the full list of properties that can be interpolated in .
We support multiple types of interpolators: Step functions, linear, exponential and cubic bezier to enable your map looking like you want at each zoom level. See the page.
string | auto |
number |
string |
string |
number |
See of these attributes on each label type.
If using a categorical
or numeric
visualization, all the previous properties must be arrays. If there's a single value in the array, that value is used in all categories. If there are as many values as categories
, the corresponding value will be used for each category. You can see an example of a categorical
viz .
To see how a legend is defined, take a look at .
Categories required. A categories
array must be defined in the config block when defining a categorical visualization. Read more about categorical visualizations .
Not enough or too many attribute_name
values. When defining a categorical visualization, all style and label properties must be an array with either a single value that will apply to all categories or an array with as many values as categories defined in the config block. Read more about categorical visualizations .
color
An array of colors that will be used in the heatmap. From less density to more.
size
Controls the size of each point.
intensity
Controls the intensity of a heightmap.
The ID of the map that contains the comment.
The ID of the comment to delete.
No Content
The ID of the map that contains the comment.
The ID of the comment to resolve.
Only needed when using the API as part of a plugin
The ID of the Project to delete. Note: This will delete all Folders and Maps inside the project!
No Content
Only needed when using the API as part of a plugin
Optionally assign the token to a user email address. Providing an email will enable the viewer to export data if the Map allows it.
The name to be used for the Project
Either viewable by all members of the workspace, or private to users who are invited.
The ID of the project to update
The name to be used for the Project
Either viewable by all members of the workspace, or private to users who are invited.
The ID of the source to show
The ID of the source to sync
The ID of the source to delete
No Content
The ID of the source to update