Edit features with the Editor component/widget, configure forms, manage subtypes, handle attachments, and work with branch versioning.
67
60%
Does it follow best practices?
Impact
Pending
No eval scenarios have been run
Advisory
Suggest reviewing before use
Optimize this skill with Tessl
npx tessl skill review --optimize ./contexts/5.0/skills/arcgis-editing/SKILL.mdUse this skill for editing features including Editor configuration, form templates, subtypes, attachments, versioning, and programmatic edits.
import Editor from "@arcgis/core/widgets/Editor.js";
import FeatureForm from "@arcgis/core/widgets/FeatureForm.js";
import FormTemplate from "@arcgis/core/form/FormTemplate.js";const Editor = await $arcgis.import("@arcgis/core/widgets/Editor.js");
const FeatureForm = await $arcgis.import("@arcgis/core/widgets/FeatureForm.js");
const FormTemplate = await $arcgis.import("@arcgis/core/form/FormTemplate.js");Note: The examples in this skill use Direct ESM imports. For CDN usage, replace
import X from "path"withconst X = await $arcgis.import("path").
The simplest way to add editing is the <arcgis-editor> component.
<arcgis-map item-id="YOUR_EDITABLE_WEBMAP_ID">
<arcgis-editor slot="top-right"></arcgis-editor>
</arcgis-map>| Property | Type | Default | Description |
|---|---|---|---|
heading-level | number | Heading level for accessibility | |
hide-create-features-section | boolean | false | Hide the create features panel |
hide-edit-features-section | boolean | false | Hide the edit features panel |
hide-labels-toggle | boolean | false | Hide the labels toggle |
hide-merge-button | boolean | false | Hide the merge features button |
hide-settings-menu | boolean | false | Hide the settings menu |
hide-sketch | boolean | false | Hide the sketch tools |
hide-split-button | boolean | false | Hide the split feature button |
hide-tooltips-toggle | boolean | false | Hide the tooltips toggle |
hide-undo-redo-buttons | boolean | false | Hide undo/redo buttons |
hide-zoom-to-button | boolean | false | Hide the zoom-to button |
label | string | Component label | |
layer-infos | LayerInfo[] | Layer editing configuration | |
snapping-options | SnappingOptions | Snapping configuration | |
sync-view-selection | boolean | false | Sync selection with view |
| Event | Description |
|---|---|
arcgisSketchCreate | Fires during feature creation sketch |
arcgisSketchUpdate | Fires during feature update sketch |
arcgisPropertyChange | Fires when activeWorkflow or state changes |
arcgisReady | Fires when the component is ready |
| Method | Description |
|---|---|
cancelWorkflow() | Cancel the active editing workflow |
deleteFeatureFromWorkflow() | Delete the feature being edited |
startCreateFeaturesWorkflowAtFeatureTypeSelection() | Start creating features |
startUpdateWorkflowAtFeatureEdit(feature) | Start editing a specific feature |
startUpdateWorkflowAtFeatureSelection() | Start feature selection for update |
<arcgis-map basemap="streets-navigation-vector">
<arcgis-editor slot="top-right"></arcgis-editor>
</arcgis-map>
<script type="module">
const FeatureLayer = await $arcgis.import("@arcgis/core/layers/FeatureLayer.js");
const mapElement = document.querySelector("arcgis-map");
const view = await mapElement.view;
await view.when();
const layer = new FeatureLayer({
url: "https://services.arcgis.com/.../FeatureServer/0"
});
mapElement.map.add(layer);
const editor = document.querySelector("arcgis-editor");
editor.layerInfos = [{
layer: layer,
formTemplate: {
title: "Edit Feature",
elements: [
{ type: "field", fieldName: "name", label: "Name" },
{ type: "field", fieldName: "category", label: "Category" }
]
},
addEnabled: true,
updateEnabled: true,
deleteEnabled: false
}];
</script>import Editor from "@arcgis/core/widgets/Editor.js";
const editor = new Editor({
view: view,
layerInfos: [{
layer: featureLayer,
formTemplate: {
title: "Feature Details",
description: "Enter feature information",
elements: [
{
type: "field",
fieldName: "name",
label: "Name",
description: "Enter the feature name"
},
{
type: "field",
fieldName: "category",
label: "Category"
}
]
},
addEnabled: true,
updateEnabled: true,
deleteEnabled: false
}]
});
view.ui.add(editor, "top-right");| Property | Type | Default | Description |
|---|---|---|---|
layer | FeatureLayer | required | The layer to edit |
formTemplate | FormTemplate | Form configuration | |
enabled | boolean | true | Enable editing on layer |
addEnabled | boolean | true | Allow creating features |
updateEnabled | boolean | true | Allow updating features |
deleteEnabled | boolean | true | Allow deleting features |
geometryUpdatesEnabled | boolean | true | Allow geometry edits |
attributeUpdatesEnabled | boolean | true | Allow attribute edits |
attachmentsOnCreateEnabled | boolean | true | Allow attachments when creating |
attachmentsOnUpdateEnabled | boolean | true | Allow attachments when updating |
import FeatureForm from "@arcgis/core/widgets/FeatureForm.js";
const form = new FeatureForm({
container: "formDiv",
layer: featureLayer,
formTemplate: {
title: "Edit Feature",
elements: [
{
type: "field",
fieldName: "name",
label: "Name"
},
{
type: "group",
label: "Location Details",
elements: [
{ type: "field", fieldName: "address" },
{ type: "field", fieldName: "city" },
{ type: "field", fieldName: "zip" }
]
}
]
}
});
// Set feature to edit
form.feature = graphic;
// Listen for submit
form.on("submit", async () => {
if (form.valid) {
const values = form.getValues();
graphic.attributes = { ...graphic.attributes, ...values };
await featureLayer.applyEdits({
updateFeatures: [graphic]
});
}
});<arcgis-feature-form></arcgis-feature-form>
<script type="module">
const featureForm = document.querySelector("arcgis-feature-form");
// Configure the form
featureForm.formTemplate = {
title: "Edit Feature",
elements: [
{ type: "field", fieldName: "name", label: "Name" },
{ type: "field", fieldName: "status", label: "Status" }
]
};
// Set feature to edit
featureForm.feature = selectedGraphic;
// Listen for value changes
featureForm.addEventListener("arcgisValueChange", (event) => {
console.log("Field changed:", event.detail.fieldName, event.detail.value);
});
// Listen for submit
featureForm.addEventListener("arcgisSubmit", (event) => {
console.log("Form submitted");
});
</script>const formTemplate = {
title: "Asset Details",
elements: [
{
type: "group",
label: "Basic Information",
elements: [
{ type: "field", fieldName: "asset_id" },
{ type: "field", fieldName: "asset_name" },
{ type: "field", fieldName: "asset_type" }
]
},
{
type: "group",
label: "Maintenance",
initialState: "collapsed",
elements: [
{ type: "field", fieldName: "last_inspection" },
{ type: "field", fieldName: "next_inspection" },
{ type: "field", fieldName: "condition" }
]
}
]
};const formTemplate = {
expressionInfos: [
{
name: "type-requires-subtype",
expression: "$feature.type == 'complex'"
}
],
elements: [
{
type: "field",
fieldName: "type"
},
{
type: "field",
fieldName: "subtype",
visibilityExpression: "type-requires-subtype"
}
]
};const form = new FeatureForm({
layer: featureLayer,
formTemplate: {
elements: [{
type: "field",
fieldName: "email",
label: "Email",
requiredExpression: "email-required"
}],
expressionInfos: [{
name: "email-required",
expression: "$feature.contact_method == 'email'"
}]
}
});
// Available expression types on FieldElement:
// - visibilityExpression: controls field visibility
// - editableExpression: controls whether field is editable
// - requiredExpression: controls whether field is required
// - valueExpression: computes a calculated value for the field
if (form.valid) {
// Submit
}// Add features
const addResult = await featureLayer.applyEdits({
addFeatures: [{
attributes: {
name: "New Feature",
category: "Type A"
},
geometry: {
type: "point",
longitude: -118.805,
latitude: 34.027
}
}]
});
// Update features
const updateResult = await featureLayer.applyEdits({
updateFeatures: [{
attributes: {
OBJECTID: 123,
name: "Updated Name"
}
}]
});
// Delete features
const deleteResult = await featureLayer.applyEdits({
deleteFeatures: [{ objectId: 123 }]
});
// Check results
addResult.addFeatureResults.forEach((result) => {
if (result.error) {
console.error("Add failed:", result.error.message);
} else {
console.log("Added feature OID:", result.objectId);
}
});const layer = new FeatureLayer({
url: "https://services.arcgis.com/.../FeatureServer/0"
});
await layer.load();
const subtypeField = layer.subtypeField;
const subtypes = layer.subtypes;
subtypes.forEach(subtype => {
console.log("Code:", subtype.code);
console.log("Name:", subtype.name);
console.log("Default values:", subtype.defaultValues);
});import SubtypeGroupLayer from "@arcgis/core/layers/SubtypeGroupLayer.js";
const subtypeLayer = new SubtypeGroupLayer({
url: "https://services.arcgis.com/.../FeatureServer/0"
});
await subtypeLayer.load();
// Access sublayers by subtype
subtypeLayer.sublayers.forEach(sublayer => {
console.log("Subtype:", sublayer.subtypeCode, sublayer.title);
// Each sublayer can have different renderer/popup
});const editor = new Editor({
view: view,
layerInfos: [{
layer: subtypeGroupLayer,
addEnabled: true,
updateEnabled: true
}]
});Branch versioning allows multiple users to edit the same data simultaneously without conflicts until reconciliation.
import VersionManagementService from "@arcgis/core/versionManagement/VersionManagementService.js";
const vms = new VersionManagementService({
url: "https://services.arcgis.com/.../VersionManagementServer"
});
await vms.load();
console.log("Default version:", vms.defaultVersionIdentifier);const versions = await vms.getVersionInfos();
versions.forEach(v => {
console.log("Version:", v.versionName);
console.log(" Owner:", v.versionOwner);
console.log(" Access:", v.access); // public, protected, private
console.log(" Parent:", v.parentVersionName);
});// Create new version
const newVersion = await vms.createVersion({
versionName: "MyEditVersion",
description: "Version for editing project X",
access: "private",
parentVersionName: "sde.DEFAULT"
});
// Alter version properties
await vms.alterVersion({
versionName: "sde.MyEditVersion",
description: "Updated description",
access: "protected"
});
// Delete a version
await vms.deleteVersion({
versionName: "sde.MyEditVersion"
});// Set version on layer at creation
const featureLayer = new FeatureLayer({
url: "https://services.arcgis.com/.../FeatureServer/0",
gdbVersion: "sde.MyEditVersion"
});
// Or change version dynamically
await featureLayer.load();
featureLayer.gdbVersion = "sde.AnotherVersion";
await featureLayer.refresh();const versionIdentifier = version.versionIdentifier;
// Start editing session
await vms.startEditing({ versionIdentifier });
// Apply edits to feature layer
await featureLayer.applyEdits(edits);
// Reconcile with parent
const reconcileResult = await vms.reconcile({
versionIdentifier: versionIdentifier,
abortIfConflicts: false,
conflictDetection: "by-attribute",
conflictResolution: "favorEditVersion",
withPost: false
});
if (!reconcileResult.hasConflicts) {
await vms.post({ versionIdentifier });
}
// Stop editing
await vms.stopEditing({
versionIdentifier: versionIdentifier,
saveEdits: true
});async function editInVersion(vms, featureLayer, edits) {
const version = await vms.createVersion({
versionName: `Edit_${Date.now()}`,
description: "Temporary edit version",
access: "private"
});
const versionIdentifier = version.versionIdentifier;
try {
featureLayer.gdbVersion = version.versionName;
await featureLayer.refresh();
await vms.startEditing({ versionIdentifier });
await featureLayer.applyEdits(edits);
const result = await vms.reconcile({
versionIdentifier,
abortIfConflicts: false,
conflictResolution: "favorEditVersion",
withPost: false
});
if (result.hasConflicts) {
throw new Error("Conflicts detected during reconcile");
}
await vms.post({ versionIdentifier });
await vms.stopEditing({ versionIdentifier, saveEdits: true });
} finally {
featureLayer.gdbVersion = vms.defaultVersionIdentifier;
await featureLayer.refresh();
await vms.deleteVersion({ versionName: version.versionName });
}
}const relatedRecords = await layer.queryRelatedFeatures({
outFields: ["*"],
relationshipId: 0,
objectIds: [selectedFeature.attributes.OBJECTID]
});
const results = relatedRecords[selectedFeature.attributes.OBJECTID];
results.features.forEach(related => {
console.log("Related:", related.attributes);
});const editor = new Editor({
view: view,
layerInfos: [{
layer: featureLayer,
attachmentsOnCreateEnabled: true,
attachmentsOnUpdateEnabled: true
}]
});// Query attachments
const attachments = await layer.queryAttachments({
objectIds: [featureOID]
});
// Add attachment
const formData = new FormData();
formData.append("attachment", fileBlob, "photo.jpg");
await layer.addAttachment(featureOID, formData);
// Delete attachment
await layer.deleteAttachments(featureOID, [attachmentId]);<!DOCTYPE html>
<html>
<head>
<script src="https://js.arcgis.com/5.0/"></script>
<script type="module" src="https://js.arcgis.com/5.0/map-components/"></script>
<style>
html, body { height: 100%; margin: 0; }
</style>
</head>
<body>
<arcgis-map basemap="streets-navigation-vector" center="-118.805,34.027" zoom="13">
<arcgis-zoom slot="top-left"></arcgis-zoom>
<arcgis-editor slot="top-right"></arcgis-editor>
</arcgis-map>
<script type="module">
const FeatureLayer = await $arcgis.import("@arcgis/core/layers/FeatureLayer.js");
const mapElement = document.querySelector("arcgis-map");
const view = await mapElement.view;
await view.when();
const layer = new FeatureLayer({
url: "https://services.arcgis.com/.../FeatureServer/0"
});
mapElement.map.add(layer);
const editor = document.querySelector("arcgis-editor");
editor.layerInfos = [{
layer: layer,
formTemplate: {
title: "Edit Feature",
elements: [
{
type: "group",
label: "Details",
elements: [
{ type: "field", fieldName: "name", label: "Name" },
{ type: "field", fieldName: "type", label: "Type" },
{ type: "field", fieldName: "status", label: "Status" }
]
}
]
},
attachmentsOnCreateEnabled: true,
attachmentsOnUpdateEnabled: true
}];
</script>
</body>
</html>editor-basic - Basic Editor component usagewidgets-editor-configurable - Configurable Editor widget with layerInfoswidgets-editor-subtypes - Editing with subtype group layerswidgets-editor-form-elements - Form element configuration in Editorediting-applyedits - Programmatic editing with applyEditsediting-groupedfeatureform - Grouped feature form layoutediting-featureform-fieldvisibility - Controlling field visibilitychanging-version - Switching geodatabase versionsEditing permissions: Calling applyEdits on a layer without checking edit capabilities causes silent failures.
// Anti-pattern: calling applyEdits without checking capabilities
const layer = new FeatureLayer({ url: "https://services.arcgis.com/.../FeatureServer/0" });
await layer.applyEdits({ addFeatures: [newFeature] });// Correct: check capabilities before editing
const layer = new FeatureLayer({ url: "https://services.arcgis.com/.../FeatureServer/0" });
await layer.load();
if (layer.editingEnabled && layer.capabilities?.editing?.supportsAddingFeatures) {
await layer.applyEdits({ addFeatures: [newFeature] });
} else {
console.error("Layer does not support adding features");
}Impact: The applyEdits call fails with a cryptic server error or silently returns an error result that is easy to miss.
Version locking: Editing branch-versioned data without proper session management causes version locks.
// Anti-pattern: editing without version management session
layer.gdbVersion = "editor1.design_v1";
await layer.applyEdits({ updateFeatures: [updatedFeature] });
// Version remains locked, blocking other users// Correct: use proper version management session
await vms.startEditing({ versionIdentifier });
layer.gdbVersion = "editor1.design_v1";
await layer.applyEdits({ updateFeatures: [updatedFeature] });
await vms.stopEditing({ versionIdentifier, saveEdits: true });Impact: The version stays locked after editing, preventing other users from accessing or editing it.
applyEdits result checking: Always inspect the result object for individual feature errors.
const result = await layer.applyEdits({ addFeatures: [feature1, feature2] });
result.addFeatureResults.forEach((res) => {
if (res.error) {
console.error("Failed to add feature:", res.error.message);
}
});Form expression names: Must reference expressions by name string, not by expression text.
expressionInfos: [{ name: "my-expr", expression: "$feature.type == 'A'" }],
elements: [{
visibilityExpression: "my-expr" // String reference to name
}]Subtype field: Must match the subtype configuration in the service exactly.
arcgis-tables-forms for FeatureTable and FormTemplate input types.arcgis-interaction for hit testing and sketching.arcgis-layers for FeatureLayer configuration.arcgis-core-maps for autocasting guidance.d84407b
If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.