Guides Salesforce Aura component development, maintenance, debugging of bundles/events/server actions/Locker Service, and LWC migration.
How this skill is triggered — by the user, by Claude, or both
Slash command
/salesforce-claude-code:sf-aura-developmentThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Aura is Salesforce's original Lightning component framework (introduced 2014). While LWC is the modern standard, thousands of production orgs still run Aura components. This skill covers authoring, maintaining, and migrating Aura components.
Aura is Salesforce's original Lightning component framework (introduced 2014). While LWC is the modern standard, thousands of production orgs still run Aura components. This skill covers authoring, maintaining, and migrating Aura components.
Note: Aura is in maintenance mode. For new development, use LWC (see sf-lwc-development).
@../_reference/AURA_COMPONENTS.md
An Aura component is a folder (bundle) containing up to eight files. Only the .cmp file is required.
force-app/main/default/aura/AccountManager/
AccountManager.cmp <- Component markup (required)
AccountManagerController.js <- Client-side controller (action handlers)
AccountManagerHelper.js <- Reusable logic (called by controller)
AccountManagerRenderer.js <- Custom rendering overrides (rare)
AccountManager.css <- Component-scoped styles
AccountManager.design <- App Builder property editor config
AccountManager.cmp-meta.xml <- Metadata (apiVersion, description)
<aura:component controller="AccountController"
implements="force:appHostable,flexipage:availableForAllPageTypes"
access="global">
<aura:attribute name="accounts" type="Account[]" default="[]" />
<aura:attribute name="isLoading" type="Boolean" default="true" />
<aura:attribute name="errorMessage" type="String" />
<aura:registerEvent name="accountSelected" type="c:AccountSelectedEvent" />
<aura:handler name="init" value="{!this}" action="{!c.doInit}" />
<lightning:card title="Account Manager" iconName="standard:account">
<aura:if isTrue="{!v.isLoading}">
<lightning:spinner alternativeText="Loading" size="small" />
<aura:set attribute="else">
<aura:iteration items="{!v.accounts}" var="acct">
<lightning:tile label="{!acct.Name}">
<dl class="slds-list_horizontal slds-wrap">
<dt class="slds-item_label">Type:</dt>
<dd class="slds-item_detail">{!acct.Type}</dd>
</dl>
</lightning:tile>
</aura:iteration>
</aura:set>
</aura:if>
</lightning:card>
</aura:component>
<!-- AccountSelectedEvent.evt -->
<aura:event type="COMPONENT" description="Fired when an account is selected">
<aura:attribute name="accountId" type="String" />
</aura:event>
// Child controller — firing
handleAccountClick: function(component, event, helper) {
var compEvent = component.getEvent("accountSelected");
compEvent.setParams({ accountId: event.currentTarget.dataset.accountId });
compEvent.fire();
}
<!-- Parent — handling -->
<c:AccountTile onaccountSelected="{!c.handleAccountSelected}" />
<aura:event type="APPLICATION" description="Broadcast notification">
<aura:attribute name="message" type="String" />
</aura:event>
// Firing
var appEvent = $A.get("e.c:GlobalNotificationEvent");
appEvent.setParams({ message: "Record saved" });
appEvent.fire();
<!-- Any component can handle -->
<aura:handler event="c:GlobalNotificationEvent" action="{!c.handleNotification}" />
Prefer component events over application events. For new cross-component communication, use Lightning Message Service instead.
Keep controllers thin; helpers do the work.
// AccountManagerController.js
({
doInit: function(component, event, helper) {
helper.loadAccounts(component);
},
handleSearch: function(component, event, helper) {
helper.loadAccounts(component);
}
})
// AccountManagerHelper.js
({
loadAccounts: function(component) {
component.set("v.isLoading", true);
var action = component.get("c.getAccounts");
action.setParams({
searchTerm: component.get("v.searchTerm") || ""
});
action.setCallback(this, function(response) {
var state = response.getState();
if (state === "SUCCESS") {
component.set("v.accounts", response.getReturnValue());
} else if (state === "ERROR") {
this.handleErrors(component, response.getError());
} else if (state === "INCOMPLETE") {
component.set("v.errorMessage", "Server unreachable.");
}
component.set("v.isLoading", false);
});
$A.enqueueAction(action);
},
handleErrors: function(component, errors) {
var message = "Unknown error";
if (errors && errors[0] && errors[0].message) {
message = errors[0].message;
}
component.set("v.errorMessage", message);
}
})
All Apex calls in Aura go through the action queue. Handle all three states: SUCCESS, ERROR, INCOMPLETE.
var action = component.get("c.getPicklistValues");
action.setStorable(); // Only for @AuraEnabled(cacheable=true) methods
Callback may fire twice: once from cache, once from server. Do not use for DML operations.
Any code executing outside the Aura lifecycle (setTimeout, Promises, third-party callbacks) must use $A.getCallback():
setTimeout($A.getCallback(function() {
if (component.isValid()) {
component.set("v.status", "Complete");
}
}), 2000);
<!-- AuraWrapper.cmp -->
<aura:component>
<c:lwcRecordDetail
record-id="{!v.selectedRecordId}"
onrecordupdate="{!c.handleRecordUpdate}" />
</aura:component>
Aura to LWC — pass data via attributes mapped to @api properties.
LWC to Aura — dispatch CustomEvent; Aura receives via on{eventname} handler, access detail via event.getParam("detail").
| Aura | LWC |
|---|---|
aura:handler name="init" | connectedCallback() |
aura:handler name="destroy" | disconnectedCallback() |
aura:attribute | @api properties |
aura:if / aura:set | lwc:if / lwc:elseif / lwc:else |
aura:iteration | for:each with key |
$A.enqueueAction() | @wire or imperative await |
component.get("v.attr") | this.propertyName |
component.set("v.attr", val) | this.propertyName = val |
component.find("auraId") | this.template.querySelector() |
| Component events | CustomEvent |
| Application events | Lightning Message Service |
$A.getCallback() | Not needed (LWC handles async natively) |
| Helper.js (separate file) | Class methods (single JS file) |
$A.createComponent() | lwc:component with lwc:is |
npx claudepluginhub jiten-singh-shahi/salesforce-claude-code --plugin salesforce-claude-codeGenerates Lightning Web Components bundles, wire patterns, Apex/GraphQL integration, SLDS 2 styling, accessibility, and Jest tests using PICKLES methodology.
Provides patterns for Salesforce platform development: Lightning Web Components (LWC), Apex triggers/classes, REST/Bulk APIs, Connected Apps, Salesforce DX with scratch orgs and 2GP.
Explains Salesforce LWC Winter '26 features: lightning/graphql module migration from uiGraphQLApi, sf lightning dev component for local development, and platform module access in previews.