- Lightning Framework
- Web Stack Transformation
- Setting Up My Domain and Dev Hub
- Setting Up Project and Scratch Org
- Create First Component in LWC
- Component Deployment
- Local Properties and Data Binding
- @track properties
- Getters
- Conditional Rendering
- Template Looping (for:each and iterator)
- Component Composition
- Accessing Elements in LWC
- Parent to child Communication
- Parent to Child Communication by calling the Child method from the parent component
- Child to Parent Communication
- Setter Method
- Lifecycle Hooks
- Slot
- Create Static Resources
- Internationalization
- Navigation Service
- Navigate to New Case record creation
- Navigate to Case Home Page
- Navigation to record page
- Navigation to custom application
- Navigation to external URL
- Navigation to custom tab
- Navigation to named page
- Navigation to List View
- Navigation to Related List
- Navigation to Files
- Navigation to Tab
- Navigation to Visualforce Page
- Navigation to Custom Aura Component
- Base Lightning Components
- Base Lightning Components
- Lightning Data Service Wire Adapter and Functions
- Apex In LWC
- Flow to LWC data Transfer
- LWC to Flow data Transfer
- Component Communication
The Lightning Component framework is a Ul framework for developing single page applications for mobile and desktop devices.
Lightning Components using two programming models.
- Aura Components Model
- Lightning Web Component Model
https://www.absyz.com/difference-between-aura-and-lwc/
Lightning Web Components is a new programming model for building Lightning components. It uses the core concepts of Web Standards
- Lightweight framework
- Better Performance
- No need to learn different framework to develop application in Salesforce
- Interoperability with lightning Aura components
- Better testing using Jest
- Better Security
Lightning experience is the new user interface (Ul) for salesforce installations. It is a significant upgrade from the Salesforce Classic view, brining modern appearance and functionality to the salesforce Platform.
It is a set of tools that streamlines the entire development life cycle. It improves team development and collaboration, facilitates automated testing and continuous integration, and makes the release cycle more efficient and agile.
- Visual Studio Code
- Salesforce CLI
- Salesforce Extension Pack
Setup my domain in for salesforce org.
Once you'll enable Dev Hub then you can not disable it.
We'll create project with the help of Visual studio and it's extension Salesforce Extension Pack
Following steps we need to follow for setting up project
Step 1: Open Visual Studio and then click on View and then select Command Palette..
Step 2: Type SFDX (Salesforce Developer Experience) in command palette then dropdown will be there then select create project with Manifest.
Step 3: After creating a project, authorize a Dev Hub, that means connect that project with salesforce application.
Step 4: Create Scratch org and connect that scratch org with project. Before create scratch org we need to add hasSampleData key into config json and set that value true.
The folder and its files must follow these naming rules. • Must begin with a lowercase letter • Contain only alphanumeric or underscore characters • Must be unique in the namespace • Can't include whitespace • Can't end with an underscore • Can't contain two consecutive underscores • Can't contain a hyphen (dash)
- Using Terminal
sfdx force:lightning:component:create --type lwc -n helloWorld
- Using Command Palette
VsCode => View => Command Palette => Type Create Lightning Web Component => Hit Enter => Enter desired filename => hit enter => Again hit enter to choose default path
- camelCase: Each word in the middle of the respective phrase begins with a capital letter.
- PascalCase: It is same like Camel Case where first letter always is capitalized.
- kebab-case: Respective phrase will be transferred to all lowercase with hyphen(-) separating words.
Case Name | camelCase | PascalCase | kabab-case |
---|---|---|---|
Example | helloWorld | HelloWorld | hello-world |
Usage | component Name | Component class Name | Component reference and HTML attribute name |
- First of all we need to configure xml file of newly created component, we need to add targets and isExposed should be true.
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>59.0</apiVersion>
<isExposed>true</isExposed>
<targets>
<target>lightning__AppPage</target>
</targets>
</LightningComponentBundle>
- variables which are declared in js file are local properties
- Data binding in the Lightning web component is a synchronization between the controller and the template (HTML)
- In template we can access property value directly if it's primitive or object.
- Dot notation is used to access the property from an object
- LWC doesn't allow computed expressions like Names [2] or {2+2}
- The property in f } must be a valid JavaScript identifier or member expression. Like {name} or {user.name}
- Avoid adding spaces around the property, for example, { data }
Example : See getters section js and HTML file
When a field contains an object or an array, there's a limit to the depth of changes that are tracked. To tell the framework to observe changes to the properties of an object or to the elements of an array, decorate the field with @track.
Normal Property vs @track property Without using @track, the framework observes changes that assign a new value to a field/property. If the new value is not === to the previous value, the component re-renders.
Example : See getters example
File : helloWorld.js
import { LightningElement, track } from "lwc";
export default class HelloWorld extends LightningElement {
fullName = "Rishabh Agrawal";
course = "Aura";
@track address = {
country: "India",
city: "Delhi",
pincode: 112233
};
users = ["Risahbh", "Shikha", "Madhav", "Ram"];
num_1 = 10;
num_2 = 20;
onKeyUpHandler(event) {
this.course = event.target.value;
}
onCityKeyUpHandler(event) {
this.address.city = event.target.value;
}
get getFirstUser() {
return this.users[0];
}
get calculationValue() {
return this.num_1 * this.num_2;
}
}
File : helloWorld.html
<template>
<lightning-card title="Two way data binding">
<div class="slds-m-around_medium">
<lightning-input
type="text"
label="Enter Course Name"
onkeyup={onKeyUpHandler}
></lightning-input>
<div>Hello, {fullName}, learning new course that is : {course}</div>
</div>
</lightning-card>
<lightning-card title="@track Property">
<div class="slds-m-around_medium">
<lightning-input
type="text"
label="Enter City Name"
onkeyup={onCityKeyUpHandler}
></lightning-input>
<div>I am from {address.city} city.</div>
</div>
</lightning-card>
<lightning-card title="Getter function">
<div class="slds-m-around_medium">
<div>{getFirstUser}</div>
<div>The multiplication of {num_1} and {num_2} is {calculationValue}</div>
</div>
</lightning-card>
</template>
File : conditionalComponent.html
<template>
<lightning-card title="Conditional rendring">
<div class="slds-m-around_medium">
<lightning-button variant="brand" label="Show Data" title="Show Data" onclick={handleShowDataButtonClick} class="slds-m-left_x-small"></lightning-button>
<div>
<template lwc:if={isVisible}>
Button Clicked
</template>
<template lwc:else>
Please click button to show data.
</template>
</div>
<div class="slds-m-top_medium">
<lightning-input type="text" label="Enter some text" onkeyup={handleCondtionalInputText}></lightning-input>
<template lwc:if={checkHelloText}>
yooo... you typed hello
</template>
<template lwc:elseif={checkJapanText}>
yooo... you typed Japan
</template>
</div>
</div>
</lightning-card>
</template>
File : conditionalComponent.js
import { LightningElement } from "lwc";
export default class ConditionalComponent extends LightningElement {
isVisible = false;
text;
handleShowDataButtonClick() {
this.isVisible = true;
}
handleCondtionalInputText(event) {
this.text = event.target.value;
}
get checkHelloText() {
return this.text?.toLowerCase() === 'hello'
}
get checkJapanText() {
return this.text?.toLowerCase() === 'japan'
}
}
There are many scenarios in which we have to render the same set of elements with mostly same styling with different data in the HTML. To solve this issue, we have template looping in the LWC Template Looping Types
- for:Each
- iterator
Below is the syntax of the for:each loop
<template for:each={array} for:item="currentItem" for:index="index">
-----Here your repeatable template comes-----
</template>
What is key and it's importance in the loop
- A key is a special string attribute you need to include to the first element inside the template when creating lists of elements.
- Keys help the LWC engine identify which items have changed, are added, or are removed.
- The best way to pick a key is to use a string that uniquely identifies a list item among its siblings.
Note : The key must be a string or a number, it can't be an object. You can't use the index as a value for the key.
Example File Name : templateLoopingForEach.js
import { LightningElement } from "lwc";
export default class TemplateLoopingForEach extends LightningElement {
carList = ["Ford", "Audi", "Maruti", "Hyundai", "Mercedes"];
programmingList = [
{
id: "06868",
language: "HTML"
},
{
id: "19797",
language: "CSS"
},
{
id: "298789",
language: "Javascript"
},
{
id: "398798",
language: "Apex"
},
{
id: "48967",
language: "Aura"
},
{
id: "58798",
language: "Java"
}
];
}
File Name : templateLoopingForEach.html
<template>
<!-- Card for for:each demo with an array -->
<lightning-card title="for:each demo with array" icon-name="custom:custom14">
<ul class="slds-m-around_medium">
<template for:each={carList} for:item="car">
<a href="#" class="list-group-item list-group-item-action" key={car}>{car}</a>
</template>
</ul>
</lightning-card>
<hr />
<!-- Card for for:each demo with an array of objects -->
<lightning-card
title="for:each demo with array of objects"
icon-name="custom:custom14"
>
<ul class="slds-m-around_medium">
<template for:each={programmingList} for:item="program">
<a
href="#"
class="list-group-item list-group-item-action"
key={program.id}
>{program.language}
</a>
</template>
</ul>
</lightning-card>
</template>
To apply a special behavior to the first or last item in a list we prefer iterator over for:each
Below is the syntax of the iterator loop
<template iterator:iteratorName={array}>
-----Here your repeatable template comes-----
</template>
Properties of iterator name Using iterator name you can access the following properties
- value — The value of the item in the list. Use this property to access the properties of the array. For example -iteratorName.value.propertyName
- index — The index of the item in the list. For example -iteratorName.index
- first — A boolean value indicating whether this item is the first item in the list. For example -iteratorName.first
- last — A boolean value indicating whether this item is the last item in the list. For example -iteratorName.last
Note : In iterator loop array item will be store one by one in iteratorName (as per syntax) and those item not directly accessible from iteratorName we will get array item inside value object example -iteratorName.value.propertyName
Example
File Name : templateLoopingIterator.js
import { LightningElement } from "lwc";
export default class TemplateLoopingIterator extends LightningElement {
ceoList = [
{
id: 1,
company: "Google",
name: "Sundar Pichai"
},
{
id: 2,
company: "Apple Inc.",
name: "Tim cook"
},
{
id: 3,
company: "Facebook",
name: "Mark Zuckerberg"
},
{
id: 4,
company: "Amazon.com",
name: "Jeff Bezos"
},
{
id: 5,
company: "Capgemini",
name: "Paul Hermelin"
}
];
}
File Name : templateLoopingIterator.html
<template>
<lightning-card title="Iterator loop demo" icon-name="custom:custom14">
<ul class="slds-m-around_medium">
<div class="list-group-inline">
<template iterator:ceo={ceoList}>
<div key={ceo.value.id}>
<a
href="#"
if:true={ceo.first}
class="list-group-item list-group-item-action header"
>
<strong>List of top companies and there CEO's : </strong>
</a>
<a href="#" class="list-group-item list-group-item-action">
<strong>{ceo.value.company} : </strong>{ceo.value.name}
</a>
<a
href="#"
if:true={ceo.last}
class="list-group-item list-group-item-action footer"
>
<strong>© 2019 Lightning school salesforce </strong>
</a>
</div>
</template>
</div>
</ul>
</lightning-card>
</template>
Composition is Adding Component Within the body of another component • Composition enables you to build complex components from simpler building-block components.
How to refer child components name in parent components
- childComponent <c-child-component></c-child-component>
- childComponentDemo <c-child-component-demo></c-child-component-demo>
- sampleDemoLWC <c-sample-demo-l-w-c></c-sample-demo-l-w-c>
Replace capital letter with small letter and prefixed with hyphen
Try to avoid continuous capital letters in your component name.
To access elements rendered by a component, use the template property.
this.template.querySelector (selector);
this.template.querySelectorAll (selector);
element. template.querySelectorAll (selector);
Note - Don't use ID selector with querySelector
lwc:dom="manual"
Add this directive to a native HTML element to attach an HTML element as a child.
Parent to child communication is always crucial for building a more significant and reusable component in a large application.
Let's understand how parent component pass string data to child component from the below fig
- To make a field/property or method public, decorate it with @api decorator
- When we want to expose the property we decorate the field with @api.
- An owner component that uses the component in its HTML markup can access the component's public properties via HTML attributes.
- Public properties are reactive in nature and if the value of the property changes the component's template re-renders.
Example
Parent component File Name : alertParentComponent.js
import { LightningElement } from 'lwc';
export default class AlertParentComponent extends LightningElement {}
File Name : alertParentComponent.html
<template>
<div class="margin-bottom-2rem">
<lightning-card title="Parent to child data communication using strings" icon-name="custom:custom14">
<div class="slds-m-around_medium">
<div>
<c-alert-child-component message="Indicates a dangerous or potentially negative action"></c-alert-child-component>
<c-alert-child-component class-name="success" message="Success! Indicates a successful or positive action.">
</c-alert-child-component>
<c-alert-child-component class-name="info" message="Info! Indicates a neutral informative change or action.">
</c-alert-child-component>
<c-alert-child-component class-name="warning" message="Warning! Indicates a warning that might need attention.">
</c-alert-child-component>
</div>
</div>
</lightning-card>
</div>
</template>
Note - In Lightning Web Components we should conventionally use camelCase (lower case first letter, upper case subsequent words) to name the component and kebab-case (lower case words preceded with c- and spaced with '-' minus sign) when nesting the components in a composition scenario.
If your child public property is camelCase as in our case className, then we need to use the attribute as class-name in our child component calling
Child component File Name : alertChildComponent.js
import { LightningElement, api } from 'lwc';
export default class AlertChildComponent extends LightningElement {
@api message
@api className
get alertClassName() {
return this.className ? 'alert ' + this.className : 'alert'
}
}
File Name : alertChildComponent.html
<template>
<div class={alertClassName}>
{message}
</div>
</template>
File Name : alertChildComponent.css
.alert {
padding: 20px;
background-color: #f44336;
color: white;
opacity: 1;
transition: opacity 0.6s;
margin-bottom: 15px;
}
.alert.success {background-color: #4CAF50;}
.alert.info {background-color: #2196F3;}
.alert.warning {background-color: #ff9800;}
Example Parent component
File Name : barParentComponent.js
import { LightningElement } from 'lwc';
export default class BarParentComponent extends LightningElement {
changeColor() {
this.template.querySelector('c-bar-child-component').changeBarColor();
}
}
File Name : barParentComponent.html
<template>
<div class="margin-bottom-2rem">
<lightning-card title="Calling child method from parent" icon-name="custom:custom14">
<div class="slds-m-around_medium">
<div class="parent-wrapper">
<button class="btn" onclick={changeColor}>Click me to change bar color</button>
<div class="child-wrapper">
<c-bar-child-component></c-bar-child-component>
</div>
</div>
</div>
</lightning-card>
</div>
</template>
Child component File Name : barChildComponent.js
import { LightningElement, api } from 'lwc';
export default class BarChildComponent extends LightningElement {
className = "greenBar";
@api changeBarColor() {
this.className = "redBar"
}
}
File Name : barChildComponent.html
<template>
<div class={className}>I am child component</div>
</template>
There are Three kind of child to parent communication
- Simple Event
- Event With Data
- Event Bubbling
Example File Name : modalParentComponent.js
import { LightningElement } from 'lwc';
export default class ModalParentComponent extends LightningElement {
showModal = false
showHandler() {
this.showModal = true
}
modalCloseHandler(){
this.showModal = false
}
}
File Name : modalParentComponent.html
<template>
<lightning-card title="Simple Event" icon-name="custom:custom14">
<div class="slds-var-m-around_medium">
<button class="slds-button slds-button_success" onclick={showHandler}>Open Modal</button>
<template if:true={showModal}>
<c-modal-child-component
header-text="Message!!"
body-text="This Modal is a Child Component. Triggered from parent and on click of close button it will dispatch an event to parent handler"
onclose={modalCloseHandler}
></c-modal-child-component>
</template>
</div>
</lightning-card>
</template>
Once close event get's triggered it gonna catch my the method modalCloseHandler which is map to the c-modal-child-component component onclose attribute. we are creating the custom event and passing the data by mapping it to detail property. Once event gets created we dispatch it to parent. Once parent recieve the event it will extract the data using event.detail and show the selectedPlayer on the screen
File Name : modalChildComponent.js
import { LightningElement, api } from 'lwc';
export default class ModalChildComponent extends LightningElement {
@api headerText
@api bodyText
closeHandler(){
const selectEvent = new CustomEvent('mycustomevent', {
detail: 'fooo'
});
this.dispatchEvent(selectEvent);
}
}
File Name : modalChildComponent.html
<template>
<div id="myModal" class="modal">
<div class="modal-content">
<header>
<strong>{headerText}</strong>
</header>
<p>{bodyText}</p>
<footer class="text-right">
<button class="btn danger" onclick={closeHandler}>Close</button>
</footer>
</div>
</div>
</template>
This method is use to modified the data coming from parent component. If Object is passed as data to setter, to mutate the object we have to create a shallow copy. File Name : setterDemoChild.js
import { LightningElement, api } from 'lwc';
export default class SetterDemoChild extends LightningElement {
userDetail
@api
get detail(){
return this.userDetail
}
set detail(data){
let newAge = data.age*2
this.userDetail = {...data, age:newAge, "location":"Melbourne"}
}
}
File Name : setterDemoChild.html
<template>
<p><strong>Name</strong> - {detail.name}</p>
<p><strong>Age</strong> - {detail.age}</p>
<p><strong>location</strong> - {detail.location}</p>
</template>
File Name : setterDemoParent.js
import { LightningElement } from 'lwc';
export default class SetterDemoParent extends LightningElement {
userDetails = {
name:"salesforcetroop",
age:25
}
}
File Name : setterDemoParent.html
<template>
<lightning-card title="Setter Method Demo">
<c-setter-demo-child detail={userDetails}></c-setter-demo-child>
</lightning-card>
</template>
Lifecycle Hooks in Lightning Web Component can be considered as a callback method that is triggered at a particular phase of a component instance’s lifecycle. Lifecycle hooks are special methods or functions that are automatically called at specific stages of a component’s life in a web application.
Here’s a brief overview of some essential lifecycle hooks:
constructor(): This is where the component is initialized. You can set default values and perform one-time setup.
connectedCallback(): After the component is added to the DOM, this hook runs. It’s a great place to perform DOM manipulations and data retrieval.
renderedCallback(): This hook is triggered after rendering occurs. It’s ideal for operations that require knowledge of the rendered DOM.
disconnectedCallback(): When the component is removed from the DOM, this hook is invoked. Use it for cleanup operations and resource releases.
errorCallback(): If an error occurs during rendering, this hook is called. It’s your opportunity to gracefully handle errors.
Flow of Lifecycle hooks
First it called parent constructor then parent connectedCallBack and then if there is any child component then it moves to child constructor → connectedCall Back → renderedCallBack then it moves to parent renderedCallback
- Creation:
constructor()
: This is the first hook to run when an instance of the component is created. It’s where you initialize variables and set default values.
constructor()
- Invoked when the instance of the component is created (similar to init() in aura).
- Fired in the parent component first since it flows from parent to child.
- You have to call super() inside first to call parent class constructor ie. LightningElement.
- Access element in the component template, use this.template.
import { LightningElement } from 'lwc';
export default class LifeCycleHookParent extends LightningElement {
constructor() {
super(); // call LightningElement class constructor console.log('Parent Constructor Called');
let con = this.template //access host element
console.log(con);
}
}
- Initialization:
connectedCallback()
: After the component is initialized, this hook is called. It’s an ideal place for DOM manipulations and data retrieval.
connectedCallback()
- Invoke when component inserted into DOM.
- It flows Parent to Child.
- Used for to perform initialisation task such as fetch data, set up cache, listen events.
- To check component is connected in DOM, use isConnected method.
connectedCallback(){
console.log('Parent Connected Call Back called');
let cb = this.template
console.log('is connected=> ' + cb.isConnected);
}
- Rendering:
renderedCallback()
: This hook is triggered after the component’s initial render. It’s suitable for actions that require knowledge of the rendered DOM, like interacting with elements.
renderedCallback()
- Use it to perform logic after a component has finished the rendering phase. It invoked when a component is completely rendered on UI.
- Flows from child to parent component.
- Component rendered many times during there lifecycle, to track renderedCallBack(), use isRendered boolean field.
- Property leads to infinite loop in renderedCallBack().
import { LightningElement } from 'lwc';
export default class LifeCycleHookParent extends LightningElement {
isRendered = true // to check component is rendered
renderedCallback() {
if (this.isRendered) {
console.log('Parent Rendered call Back called');
this.isRendered = false
}
?
}
- Reactivity:
Whenever a property or variable changes in the component, it may trigger a reactivity cycle. During this cycle, the component checks for changes in properties, and if changes are detected, it re-renders and invokes the renderedCallback
again.
- Destruction:
disconnectedCallback()
: When the component is removed from the DOM, this hook is called. It’s an excellent place for cleanup operations and releasing resources like event listeners.
disconnectedCallback()
- Called when the element is removed from a document (remove event listener, remove time interval etc).
- Follows Parent to Child.
- Use disconnectedCallback() to clean up work done in the connectedCallback(), like removing event listeners.
- You can also use this hook to unsubscribe message channel.
- Error Handling:
errorCallback()
: If an error occurs during rendering, this hook is invoked. It provides an opportunity to gracefully handle errors and display appropriate messages.
errorCallback()
Implement it to create an error boundary component that captures errors in all the descendent components in its tree.
It captures errors that occur in the descendant’s lifecycle hooks or during an event handler declared in an HTML template.
- Called when a descendant(Child) component throws an error.
- Two arguments are passed in errorCallback(error,stack), the error argument is a JavaScript native error object, and the stack argument is a string.
errorCallback(error, stack){
console.log(error message);
console.log('Stack: - ' + stack);
}
Passing Markup into Slots
- Slot is useful to pass HTML markup into the another component.
- <slot></slot> markup is used to catch the HTML passed by parent component
- You can't pass an Aura component into a slot.
There are two types of Slots
Named Slots : When name attribute is defined in slot element <slot name="head"></slot>
Unnamed Slots : When a slot without a name attribute <slot></slot>
Example
File Name : slotParentDemo.js
import { LightningElement } from 'lwc';
export default class SlotParentDemo extends LightningElement {}
File Name : slotParentDemo.html
<template>
<lightning-card title="Slot Demos">
<div class="slds-p-around_medium">
<c-slot-child-demo>
<p class="slds-text-color_success" slot="first">My Name is Salesforcetroop</p>
<p class="slds-text-color_error" slot="second">My Age is 25</p>
<p slot="footer"> I am footer</p>
</c-slot-child-demo>
</div>
<div class="slds-p-around_medium">
<c-slot-child-demo>
<p class="slds-text-color_success" slot="first">My Name is Salesforcetroop</p>
<p class="slds-text-color_error" slot="second">My Age is 25</p>
<!-- <p slot="footer"> I am footer</p> -->
</c-slot-child-demo>
</div>
</lightning-card>
</template>
File Name : slotChildDemo.js
import { LightningElement } from 'lwc';
export default class SlotChildDemo extends LightningElement {
handleFooterChange(){
const footerElem = this.template.querySelector('.slds-card__footer')
if(footerElem){
footerElem.classList.remove('slds-hide')
}
}
}
File Name : slotChildDemo.html
<template>
<div>What is Your Name?</div>
<div><slot name="first"></slot></div> <!--named slots-->
<div>What is Your Age?</div>
<div><slot name="second"></slot></div>
<footer class="slds-card__footer slds-hide">
<slot name="footer" onslotchange={handleFooterChange}></slot>
</footer>
</template>
https://developer.salesforce.com/docs/platform/lwc/guide/create-resources.html
https://www.apexhours.com/how-do-you-handle-data-security-in-salesforce/
Import internationalization properties from the @salesforce/i18n scoped module. Lightning web components have internationalization properties that you can use to adapt your components for users worldwide, across languages, currencies, and timezones.
File Name : internationalization.js
import { LightningElement } from 'lwc';
import LOCALE from '@salesforce/i18n/locale';
import CURRENCY from '@salesforce/i18n/currency';
import DIR from '@salesforce/i18n/dir';
import LANG from "@salesforce/i18n/lang";
export default class Internationalization extends LightningElement {
dir = DIR;
lang = LANG;
today = new Date();
formatedDate = new Intl.DateTimeFormat(LOCALE).format(this.today);
number = 45847584.33;
formatedNumber = new Intl.NumberFormat(LOCALE, {
style : 'currency',
currency : CURRENCY,
currencyDisplay : 'symbol'
}).format(this.number)
}
File Name : internationalization.html
<template>
<lightning-card title="Internationalization">
<div class="slds-p-around_medium">
<p dir={dir} lang={lang}>
{formatedNumber} {formatedDate}
</p>
</div>
</lightning-card>
</template>
The lightning/navigation service is used to navigate in Lightning Experience, Lightning Communities and in Salesforce Application. Navigation Service in Lightning Web Components are used to navigate to Record Pages, Web Pages, Objects, List Views, Custom Tabs, Related Lists, Files etc.
Navigation Service uses a PageReference rather than a URL. A PageReference is a JavaScript object that provides the details of Page Type, attributes and state of the Page.
The PageReference envelopes a component from future changes to URL formats. Page type(String) and attributes(Object) are required parameters, state(Object) is optional parameter. To use the navigation service in Lightning Web Components, first import it in the JavaScript file
import { NavigationMixin } from 'lightning/navigation';
Next apply NavigationMixin function to component’s base class
export default class SampleNavigationService extends NavigationMixin(LightningElement) {}
NavigateMixin adds two API’s to component’s class:
NavigateMixin.Navigate : To navigate to another page in the application
NavigateMixin.GenerateURL : To get a promise that resolves to the resulting URL. The URL can be used in the href attribute of an anchor. It can utilize the URL to open a new window using the Browser api – Window.open(url)
Examples
<!-- sampleNavigationService.html -->
<template>
<lightning-card title="Navigation Service">
<div class="slds-p-left_medium">
<lightning-button label="New Case" onclick={navigateToNewCasePage}></lightning-button>
</div>
</lightning-card>
</template>
import { LightningElement } from 'lwc';
import { NavigationMixin } from 'lightning/navigation';
export default class SampleNavigationService extends NavigationMixin(LightningElement) {
navigateToNewCasePage() {
this[NavigationMixin.Navigate]({
type: 'standard__objectPage',
attributes: {
objectApiName: 'Case',
actionName: 'new'
},
});
}
}
<template>
<lightning-card title="Case Home Page Navigation">
<div class="slds-p-left_medium">
<a href={refUrl} onclick={handleNavigationClick}>Case Home</a>
</div>
</lightning-card>
</template>
import { LightningElement, wire } from 'lwc';
import { NavigationMixin } from 'lightning/navigation';
export default class BasicNavigationLWC extends NavigationMixin(LightningElement) {
refUrl;
connectedCallback() {
this.caseHomePageRef = {
type: 'standard__objectPage',
attributes: {
objectApiName: 'Case',
actionName: 'home'
}
};
this[NavigationMixin.GenerateUrl](this.caseHomePageRef)
.then(url => this.refUrl = url);
}
handleNavigationClick(evt) {
evt.preventDefault();
evt.stopPropagation();
this[NavigationMixin.Navigate](this.caseHomePageRef);
}
}
navigateToViewCasePage() {
this[NavigationMixin.Navigate]({
type: 'standard__recordPage',
attributes: {
recordId: this.recId,
objectApiName: 'Case',
actionName: 'view'
},
});
}
navigateToMyCustomApplication() {
this[NavigationMixin.Navigate]({
type: 'standard__app',
attributes: {
appTarget: 'c__MyCustomApplication',
}
});
}
navigateToApexHoursPage() {
this[NavigationMixin.Navigate]({
type: 'standard__webPage',
attributes: {
url: 'https://www.apexhours.com/'
}
});
}
navigateToCustomTab() {
this[NavigationMixin.Navigate]({
type: 'standard__navItemPage',
attributes: {
apiName: 'CustomTabName'
},
});
}
import { LightningElement } from 'lwc';
import { NavigationMixin } from 'lightning/navigation';
export default class NavigateToPage extends NavigationMixin(LightningElement) {
navigateToPage() {
// Navigation to a standard named page
this[NavigationMixin.Navigate]({
type: 'standard__namedPage',
attributes: {
pageName: 'home'
}
});
}
}
navigateToCaseListView() {
this[NavigationMixin.Navigate]({
type: 'standard__objectPage',
attributes: {
objectApiName: 'Case',
actionName: 'list'
},
state: {
filterName: 'Recent'
},
});
}
navigateToContactRelatedList() {
this[NavigationMixin.Navigate]({
type: 'standard__recordRelationshipPage',
attributes: {
recordId: this.recordId,
objectApiName: 'Account',
relationshipApiName: 'Contacts',
actionName: 'view'
},
});
}
navToFilesPage() {
this[NavigationMixin.Navigate]({
type: 'standard__objectPage',
attributes: {
objectApiName: 'ContentDocument',
actionName: 'home'
},
});
}
navigateToCustomTab() {
this[NavigationMixin.Navigate]({
type: 'standard__navItemPage',
attributes: {
apiName: 'CustomTabName'
},
});
}
navigateToVisualForcePage() {
this[NavigationMixin.GenerateUrl]({
type: 'standard__webPage',
attributes: {
url: '/apex/CaseVFExample?id=' + this.recordId
}
}).then(generatedUrl => {
window.open(generatedUrl);
});
}
openCustomLightningComponent(){
this[NavigationMixin.Navigate]({
type: 'standard__component',
attributes: {
componentName: 'c__AuraComponentName'
}
});
}
Lightning Data Service is a centralized data caching mechanism which is utilized to perform create, read, update or delete on a record without having a server side apex call in Lightning web components.
https://developer.salesforce.com/docs/platform/lwc/guide/data-ui-api.html
There are many ways of interacting with Salesforce data in the Lightning web components. Knowing which approach to use for a particular use case helps you to write less code, easier code, and code that is more maintainable. The efficiency of your components is improved by using the best solution for each situation.
Base Lightning Components are built on Lightning Data Service. So, Lightning Data Service is used behind the scenes by base components and inherits its caching and synchronisation capabilities
Below are the 3 base lightning components build on Lightning Data Service:
1. lightning-record-form: A form with standard lightning UI to create, view or edit a record 2. lightning-record-edit-form: A form to create record with specified fields or update fields in an existing record 3. lightning-record-view-form: A form to display specific fields data of a record in read-only mode
When to Use these form?
- Create a metadata-driven Ul or form-based Ul similar to the record detail page in Salesforce.
- Display record values based on the field metadata.
- Hide or show localized field labels.
- Display the help text on a custom field.
- Perform client-side validation and enforce validation rules.
https://developer.salesforce.com/docs/platform/lwc/guide/data-get-user-input.html
- lightning-record-edit-form—Displays an editable form.
- lightning-record-view-form—Displays a read-only form.
- lightning-record-form—Supports edit, view, and read-only modes.
For most use cases, lightning-record-form provides a great starting point. It combines and simplifies the functionality of lightning-record-view-form and lightning-record-edit-form. All three components support a multi column layout. For example, you can use
Use the lightning-record-form component to quickly create forms to add, view, or update a record.
The lightning-record-form component provides these helpful features:
- Switches between view and edit modes automatically when the user begins editing a field in a view form
- Provides Cancel and Save buttons automatically in edit forms
- Uses the object's default record layout with support for multiple columns
- Loads all fields in the object's compact or full layout, or only the fields you specify
lightning-record-form is less customizable. To customize the form layout or provide custom rendering of record data, use lightning-record-edit-form (add or update a record) and lightning-record-view-form (view a record).
Key Attributes
- object-api-name - This attribute is always required. The lightning-record-form component requires you to specify the object-api-name attribute to establish the relationship between a record and an object. Note- Event and Task objects are not supported.
- record-id - This attribute is required only when you're editing or viewing a record.
- fields - pass record fields as an array of strings. The fields display in the order you list them.
- layout-type - Use this attribute to specify a Full or Compact layout. Layouts are typically defined (created and modified) by administrators..
- modes - This form support three mode
- edit - Creates an editable form to add a record or update an existing one.. Edit mode is the default when record-id is not provided, and displays a form to create new records.
- view - Creates a form to display a record that the user can also edit. The record fields each have an edit button. View mode is the default when record-id is provided.
- readonly - Creates a form to display a record that the user can also edit
- columns - Use this attribute to show multiple columns in the form
File Name : recordFormDemo.js
import { LightningElement, api } from 'lwc';
import { ShowToastEvent } from 'lightning/platformShowToastEvent';
import ACCOUNT_OBJECT from '@salesforce/schema/Account'
import NAME_FIELD from '@salesforce/schema/Account.Name'
import ANNUAL_REVENUE_FIELD from '@salesforce/schema/Account.AnnualRevenue'
import TYPE_FIELD from '@salesforce/schema/Account.Type'
import INDUSTRY_FIELD from '@salesforce/schema/Account.Industry'
export default class RecordFormDemo extends LightningElement {
@api recordId
@api objectApiName
objectName = ACCOUNT_OBJECT
fieldList = [NAME_FIELD, ANNUAL_REVENUE_FIELD, TYPE_FIELD, INDUSTRY_FIELD]
successHandler(event){
console.log(event.detail.id)
const toastEvent = new ShowToastEvent({
title:"Account created",
message:"Record ID: "+ event.detail.id,
variant:"success"
})
this.dispatchEvent(toastEvent)
}
}
File Name : recordFormDemo.html
<template>
<lightning-card title="Create record using lightning-record-form">
<lightning-record-form
object-api-name={objectName}
fields={fieldList}
onsuccess={successHandler}
></lightning-record-form>
</lightning-card>
<lightning-card title="Display record using lightning-record-form">
<lightning-record-form
record-id="001N000001zcknoIAA"
object-api-name={objectName}
fields={fieldList}
></lightning-record-form>
</lightning-card>
<lightning-card title="Display record in readonly mode using lightning-record-form">
<lightning-record-form
record-id="001N000001zcknoIAA"
object-api-name={objectName}
fields={fieldList}
mode="readonly"
></lightning-record-form>
</lightning-card>
<lightning-card title="Edit record using lightning-record-form">
<lightning-record-form
record-id="001N000001zcknoIAA"
object-api-name={objectName}
fields={fieldList}
mode="edit"
columns="2"
></lightning-record-form>
</lightning-card>
<!-- that will only work in record page because we are using recordId and objectApiName which will automatic fill when we will use this component in record page-->
<lightning-card title="Edit record with layout using lightning-record-form">
<lightning-record-form
record-id={recordId}
object-api-name={objectApiName}
mode="edit"
columns="2"
layout-type="Compact"
></lightning-record-form>
</lightning-card>
</template>
- Use the lightning-record-view-form component to create a form that displays Salesforce record data for specified fields associated with that record. The fields are rendered with their labels and current values as read-only.
- You can customize the form layout or provide custom rendering of record data. If you don't require customizations, use lightning-record-form instead.
- To specify read-only fields, use lightning-output-field components inside lightning-record-view-form.
File Name : recordViewFormDemo.js
import { LightningElement } from 'lwc';
export default class RecordViewFormDemo extends LightningElement {}
File Name : recordViewFormDemo.html
<template>
<lightning-card title="lightning record view form">
<lightning-record-view-form
object-api-name="Account"
record-id="001N000001zcknoIAA"
>
<div class="slds-grid slds-gutters">
<div class="slds-col slds-size_6-of-12">
<lightning-output-field field-name="Name"></lightning-output-field>
<lightning-output-field field-name="Phone"></lightning-output-field>
<lightning-output-field field-name="Industry"></lightning-output-field>
</div>
<div class="slds-col slds-size_6-of-12">
<lightning-output-field field-name="AnnualRevenue"></lightning-output-field>
</div>
</div>
</lightning-record-view-form>
</lightning-card>
</template>
- This component is used to create and edit the records.
- It provides custom layout of fields and custom rendering of record data.
File Name : recordEditForm.js
import { LightningElement } from 'lwc';
import CONTACT_OBJECT from '@salesforce/schema/Contact'
import NAME_FIELD from '@salesforce/schema/Contact.Name'
import TITLE_FIELD from '@salesforce/schema/Contact.Title'
import PHONE_FIELD from '@salesforce/schema/Contact.Phone'
import EMAIL_FIELD from '@salesforce/schema/Contact.Email';
import ACCOUNT_FIELD from '@salesforce/schema/Contact.AccountId';
export default class RecordEditForm extends LightningElement {
objectName = CONTACT_OBJECT
fields={
accountField:ACCOUNT_FIELD,
nameField:NAME_FIELD,
titleField:TITLE_FIELD,
phoneField:PHONE_FIELD,
emailField:EMAIL_FIELD
}
handleReset(){
const inputFields = this.template.querySelectorAll('lightning-input-field')
if(inputFields){
Array.from(inputFields).forEach(field=>{
field.reset()
})
}
}
}
File Name : recordEditForm.html
<template>
<lightning-card title="lightning record edit form">
<lightning-record-edit-form
object-api-name={objectName}
>
<lightning-messages></lightning-messages>
<lightning-input-field field-name={fields.accountField}></lightning-input-field>
<lightning-input-field field-name={fields.nameField}></lightning-input-field>
<lightning-input-field field-name={fields.titleField}></lightning-input-field>
<lightning-input-field field-name={fields.phoneField}></lightning-input-field>
<label class="slds-p-left_x-small">Enter your email</label>
<lightning-input-field variant="label-hidden" field-name={fields.emailField}></lightning-input-field>
<lightning-button class="slds-m-around_xx-small" label="cancel" onclick={handleReset}></lightning-button>
<lightning-button variant="brand" type="submit" class="slds-m-around_xx-small" label="Save"></lightning-button>
</lightning-record-edit-form>
</lightning-card>
</template>
File Name : recordEditForm.js
import { LightningElement } from 'lwc';
import ACCOUNT_OBJECT from '@salesforce/schema/Account'
import { ShowToastEvent } from 'lightning/platformShowToastEvent';
export default class RecordEditCustom extends LightningElement {
objectName = ACCOUNT_OBJECT
inputValue=''
handleChange(event){
this.inputValue = event.target.value
}
handleSubmit(event){
event.preventDefault()
const inputCmp = this.template.querySelector('lightning-input')
const value= inputCmp.value
if(!value.includes('Australia')){
inputCmp.setCustomValidity("The account name must include 'Australia'")
} else {
inputCmp.setCustomValidity("")
const fields = event.detail.fields
fields.Name = value
this.template.querySelector('lightning-record-edit-form').submit(fields)
}
inputCmp.reportValidity()
}
successHandler(event){
const toastEvent = new ShowToastEvent({
title:"Account created",
message: "Record ID: "+ event.detail.id,
variant:"success"
})
this.dispatchEvent(toastEvent)
}
handleError(event){
const toastEvent = new ShowToastEvent({
title:"Error creating Account",
message: event.detail.message,
variant:"error"
})
this.dispatchEvent(toastEvent)
}
}
File Name : recordEditForm.html
<template>
<lightning-card title="Custom validation in lightning record edit form">
<lightning-record-edit-form
object-api-name={objectName}
onsubmit={handleSubmit}
onsuccess={successHandler}
onerror={handleError}
>
<lightning-input label="Name"
value={inputValue}
onkeyup={handleChange}
class="slds-m-bottom_x-small"></lightning-input>
<lightning-button class="slds-m-top_small" type="submit" label="Create Account"></lightning-button>
</lightning-record-edit-form>
</lightning-card>
</template>
What is @wire service ?
- Wire service is built on Lightning Data Service
- LWC component use @wire in their JavaScript class to read data from one of the wire adapters in the lightning/ui*Api namespace.
- @wire is a reactive service
- The wire adapter defines the data shape that the wire service provisions in an immutable stream
File Name : wireDemoUserDetail.js
import { LightningElement, wire } from 'lwc';
import { getRecord } from 'lightning/uiRecordApi';
import NAME_FIELD from '@salesforce/schema/User.Name';
import EMAIL_FIELD from '@salesforce/schema/User.Email';
import CITY_FIELD from '@salesforce/schema/User.City';
import id from '@salesforce/user/Id';
const fields = [NAME_FIELD, EMAIL_FIELD, CITY_FIELD]
export default class WireDemoUserDetail extends LightningElement {
userId = id; // will return current login user id;
userInfo;
//005F3000007xXBPIA2
@wire(getRecord, {recordId: '005F3000007xXBPIA2', fields: ['User.Name', 'User.Email']})
userDetailHandler({data, error}){
if(data){
this.userInfo = data.fields;
console.log(this.userInfo)
}else {
console.error(error)
}
}
@wire(getRecord, {recordId: '005F3000007xXBPIA2', fields})
userData;
connectedCallback(){
console.log('userData', this.userData);
}
}
File Name : wireDemoUserDetail.html
<template>
<lightning-card title="User Detail using @wire as function">
<div class="slds-p-around_medium">
<template if:true={userInfo}>
<div>
<p>
<strong>Name : </strong> {userInfo.Name.value}
</p>
<p>
<strong>Email : </strong> {userInfo.Email.value}
</p>
</div>
</template>
</div>
</lightning-card>
<div class="slds-m-top_medium"></div>
<lightning-card title="User Detail using @wire as property" >
<div class="slds-p-around_medium">
<template if:true={userInfo}>
<div>
<p>
<strong>Name : </strong> {userData.data.fields.Name.value}
</p>
<p>
<strong>Email : </strong> {userData.data.fields.Email.value}
</p>
<p>
<strong>City : </strong> {userData.data.fields.City.value}
</p>
</div>
</template>
</div>
</lightning-card>
</template>
How to Import Reference of Salesforce Standard Object
// Syntax
import objectName from '@salesforce/schema/object;
// Example
import ACCOUNT_OBJECT from '@salesforce/schema/Account';
How to Import Reference to Salesforce custom Object
// Syntax
import objectName from '@salesforce/schema/object';
// Example
import PROPERTY_OBJECT from '@salesforce/schema/Property_c';
How to Import References to Salesforce Fields
// Syntax
import FIELD_NAME from '@salesforce/schema/object.field'\
// Example
import ACCOUNT_NAME from '@salesforce/schema/Account.Name'; '
import PROPERTY_NAME from '@salesforce/schema/Property_c.Name';
How to Import Reference to a field via a relationship
// Syntax
import REF_FIELD_NAME from @salesforce/schema/object.relationship.field
// Example
import ACCOUNT_OWNER from '@salesforce/schema/Account.Owner.Name';
By adding “$” to the recordId parameter, we made it reactive. From now, every change of the variable recordId value will run the getRecord function.
userId = id; // will return current login user id;
userInfo;
@wire(getRecord, {recordId: '$userId', fields: ['User.Name', 'User.Email']})
userDetailHandler({data, error}){
if(data){
this.userInfo = data.fields;
console.log(this.userInfo)
}else {
console.error(error)
}
}
Use this wire adapter to get metadata about a specific object. The response includes metadata describing the object's fields, child relationships, record type, and theme.
import { LightningElement, wire } from 'lwc';
import {getObjectInfo} from 'lightning/uiObjectInfoApi'
import ACCOUNT_OBJECT from '@salesforce/schema/Account'
export default class GetObjectInfoDemo extends LightningElement {
@wire(getObjectInfo, {objectApiName:ACCOUNT_OBJECT})
objectInfo
}
<template>
<lightning-card title="getObjectInfo Adapter">
<div class="slds-var-p-around_medium">
<template if:true={objectInfo.data}>
<div>defaultRecordTypeId: {objectInfo.data.defaultRecordTypeId}</div>
<div>Object API Name: {objectInfo.data.apiName}</div>
</template>
</div>
</lightning-card>
</template>
Use this wire adapter to get metadata for multiple objects. The response includes metadata describing the fields, child relationships, record type, and theme for each object.
import { LightningElement, wire } from 'lwc';
import {getObjectInfos} from 'lightning/uiObjectInfoApi'
import ACCOUNT_OBJECT from '@salesforce/schema/Account'
import OPPORTUNITY_OBJECT from '@salesforce/schema/Opportunity'
export default class GetObjectInfoDemo extends LightningElement {
objectApiNames = [ACCOUNT_OBJECT, OPPORTUNITY_OBJECT]
objectInfos
@wire(getObjectInfos, { objectApiNames: '$objectApiNames' })
objectInfosHandler({data}){
if(data){
console.log(data)
this.objectInfos = data
}
}
}
<template>
<lightning-card title="getObjectInfos Adapter">
<div class="slds-var-p-around_medium">
<template if:true={objectInfos}>
<template for:each={objectInfos.results} for:item="obj">
<div key={obj.result.apiName}>
<div>Object Api Name - {obj.result.apiName}</div>
<div>defaultRecordTypeId - {obj.result.defaultRecordTypeId}</div>
</div>
</template>
</template>
</div>
</lightning-card>
</template>
Use this wire adapter to get the picklist values for a specified field.
Syntax
import { LightningElement, wire } from 'Iwc';
import { getPicklistvalues } from 'lightning/uiobjectInfoApi';
import INDUSTRY_FIELD from '@salesforce/schema/Account.Industry';
export default class Example extends LightningElement {
@wire(getPicklistvalues, { recordTypeId: '012000000000000AAA', fieldApiName: INDUSTRY_FIELD })
propertyOrFunction;
}
NOTE:
recordTypeld - The ID of the record type. Use the Object Info defaultRecordTypeId property, which is returned from getObjectInfo fieldApiName - The API name of the picklist field
Example
File Name : getPicklistValuesDemo.js
import { LightningElement, wire } from "lwc";
import { getObjectInfo, getPicklistValues } from "lightning/uiObjectInfoApi";
import INDUSTRY_FIELD from "@salesforce/schema/Account.Industry";
import ACCOUNT_OBJECT from "@salesforce/schema/Account";
export default class GetPicklistValuesDemo extends LightningElement {
industryOptions = [];
selectedIndustry;
@wire(getObjectInfo, { objectApiName: ACCOUNT_OBJECT })
objectInfo;
@wire(getPicklistValues, {
recordTypeId: "$objectInfo.data.defaultRecordTypeId",
fieldApiName: INDUSTRY_FIELD
})
industryPicklist({ data, error }) {
if (data) {
this.industryOptions = this.generateIndustryOptions([...data.values]);
} else {
console.error(error);
}
}
generateIndustryOptions(data){
return data.map(val => {
return {
value : val.value,
label : val.label
}
})
}
handleIndustryChange(event) {
this.selectedIndustry = event.detail.value;
console.log(this.selectedIndustry);
}
}
File Name : getPicklistValuesDemo.html
<template>
<lightning-card title="getPicklistValues Demo">
<div class="slds-var-p-around_medium">
<lightning-combobox
name="Industry"
label="Industry"
value={value}
placeholder="Select Industry"
options={industryOptions}
onchange={handleIndustryChange} ></lightning-combobox>
<p>Selected value is: {selectedIndustry}</p>
</div>
</lightning-card>
</template>
Use this wire adapter to get the values for every picklist of a specified record type.
Syntax
import { LightningElement, wire } from 'Iwc';
import { getPicklistValuesByRecordType } from 'lightning/uiobjectInfoApi';
import ACCOUNT_OBJECT from '@salesforce/schema/Account';
export default class Example extends LightningElement {
@wire(getPicklistValuesByRecordType, { objectApiName: ACCOUNT_OBJECT })
propertyOrFunction;
}
NOTE:
recordTypeld - The ID of the record type. Use the Object Info defaultRecordTypeId property, which is returned from getObjectInfo objectApiName - The API name of the object
Example File Name : getPicklistValuesByRecordTypeDemo.js
import { LightningElement, wire } from 'lwc';
import {getPicklistValuesByRecordType, getObjectInfo} from 'lightning/uiObjectInfoApi'
import ACCOUNT_OBJECT from '@salesforce/schema/Account'
export default class GetPicklistValuesByRecordTypeDemo extends LightningElement {
ratingOptions
industryOptions
selectedRating
selectedIndustry
@wire(getObjectInfo, {objectApiName:ACCOUNT_OBJECT})
objectInfo
@wire(getPicklistValuesByRecordType, {objectApiName:ACCOUNT_OBJECT,
recordTypeId:'$objectInfo.data.defaultRecordTypeId'})
picklistHandler({data, error}){
if(data){
console.log(data)
this.ratingOptions = this.picklistGenerator(data.picklistFieldValues.Rating)
this.industryOptions = this.picklistGenerator(data.picklistFieldValues.Industry)
}
if(error){
console.error(error)
}
}
picklistGenerator(data){
return data.values.map(item=>({"label":item.label, "value":item.value}))
}
handleChange(event){
const {name, value} = event.target
console.log(name +'==>' +value)
if(name === 'industry'){
this.selectedIndustry = value
}
if(name === 'rating'){
this.selectedRating = value
}
}
}
File Name : getPicklistValuesByRecordTypeDemo.html
<template>
<lightning-card title="getPicklistValuesByRecordType Adapter">
<div class="slds-var-p-around_medium">
<template if:true={ratingOptions}>
<lightning-combobox
name="rating"
label="Rating"
value={selectedRating}
placeholder="Select Rating"
options={ratingOptions}
onchange={handleChange}></lightning-combobox>
<p>selectedRating: {selectedRating}</p>
</template>
<template if:true={industryOptions}>
<lightning-combobox
name="industry"
label="Industry"
value={selectedIndustry}
placeholder="Select Industry"
options={industryOptions}
onchange={handleChange}></lightning-combobox>
<p>selectedIndustry: {selectedIndustry}</p>
</template>
</div>
</lightning-card>
</template>
Use this wire adapter to get the record's data
Syntax
import { LightningElement, wire } from "Iwc';
import { getRecord } from 'lightning/uiRecordApi';
@wire(getRecord, { recordId: string, fields: string|string|], optionalFields?: string|string[])
propertyOrFunction
@wire(getRecord, {
recordId: string,
layoutTypes: string|string[],
modes?: string|string|],
optionalFields?: string|string[])
propertyOrFunction
recordld - The ID of the record type.
fields - A field or an array of fields to return or
layoutType - it support two values Compact or Full(default)
Modes - used with layout. Values supported are Create, Edit and View (default)
optionalFields - a field name or an array of field names. If a field is accessible to the user, it includes in response otherwise it will not throw an error.
Example
File Name : getRecordDemo.js
import { LightningElement, wire, api } from 'lwc';
import {getRecord} from 'lightning/uiRecordApi'
import NAME_FIELD from '@salesforce/schema/Account.Name'
import OWNER_NAME_FIELD from '@salesforce/schema/Account.Owner.Name'
import ANNUAL_REVENUE_FIELD from '@salesforce/schema/Account.AnnualRevenue'
export default class GetRecordDemo extends LightningElement {
name
owner
AnnualRevenue
@api recordId
// @wire(getRecord, {recordId:'$recordId',
// fields:[NAME_FIELD, OWNER_NAME_FIELD, ANNUAL_REVENUE_FIELD]})
@wire(getRecord, {recordId:'$recordId',
layoutTypes:['Full'], modes:['View']})
accountHandler({data}){
if(data){
console.log(data)
this.name = data.fields.Name.displayValue ? data.fields.Name.displayValue:
data.fields.Name.value
this.AnnualRevenue = data.fields.AnnualRevenue.displayValue ? data.fields.AnnualRevenue.displayValue:
data.fields.AnnualRevenue.value
this.owner = data.fields.Owner.displayValue ? data.fields.Owner.displayValue:
data.fields.Owner.value
}
}
}
File Name : getRecordDemo.html
<template>
<lightning-card title="getRecord Adapter">
<div class="slds-p-around_medium">
<div>Name - {name}</div>
<div>owner - {owner}</div>
<div>AnnualRevenue - {AnnualRevenue}</div>
</div>
</lightning-card>
</template>
getFieldValue Use this to gets a field's value from a record
Syntax
import { getFieldValue } from 'lightning/uiRecordApi';
getFieldValue(record: Record, field: string)
getFieldDisplayValue Use this to gets a field's value in formatted and localized format from a record
Syntax
import { getFieldDisplayValue } from 'lightning/uiRecordApi';
getFieldDisplayValue(record, field)
Example
import { LightningElement, wire, api } from 'lwc';
import {getRecord, getFieldValue, getFieldDisplayValue} from 'lightning/uiRecordApi'
import NAME_FIELD from '@salesforce/schema/Account.Name'
import OWNER_NAME_FIELD from '@salesforce/schema/Account.Owner.Name'
import ANNUAL_REVENUE_FIELD from '@salesforce/schema/Account.AnnualRevenue'
export default class GetRecordDemo extends LightningElement {
name
owner
AnnualRevenue
@api recordId
@wire(getRecord, {recordId:'$recordId',
fields:[NAME_FIELD, OWNER_NAME_FIELD, ANNUAL_REVENUE_FIELD]})
accountHandler({data}){
if(data){
console.log(data)
this.name = getFieldValue(data, NAME_FIELD)
this.AnnualRevenue = getFieldDisplayValue(data, ANNUAL_REVENUE_FIELD)
this.owner = getFieldValue(data, OWNER_NAME_FIELD)
}
}
}
Use this wire adapter to get the metadata for a list view.
Syntax
import { LightningElement, wire } from "lwc";
import { getListInfoByName } from "lightning/uiListsApi";
import ACCOUNT_OBJECT from "@salesforce/schema/Account";
export default class Example extends LightningElement {
@wire(getListInfoByName, { objectApiName: ACCOUNT_OBJECT, listViewApiName: "AllAccounts" })
propertyOrFunction;
}
objectApiName — (Required) The API name of a supported object.
listViewApiName — (Required) The API name of a list view, such as AllAccounts.
Example
File Name : getListInfoByNameDemo.js
import { LightningElement, wire } from 'lwc';
import { getListInfoByName } from "lightning/uiListsApi";
import CASE_OBJECT from '@salesforce/schema/Case';
export default class GetListInfoByNameDemo extends LightningElement {
error;
displayColumns;
@wire(getListInfoByName, {
objectApiName: CASE_OBJECT.objectApiName,
listViewApiName: "MyCases",
})
listInfo({ error, data }) {
if (data) {
console.log(data);
this.displayColumns = data.displayColumns;
this.error = undefined;
} else if (error) {
this.error = error;
this.displayColumns = undefined;
}
}
}
File Name : getListInfoByNameDemo.html
<template>
<lightning-card title="getListInfoByName Demo">
<div class="slds-p-around_medium">
<template lwc:if={displayColumns}>
<div class="slds-m-around_medium">
<template for:each={displayColumns} for:item="col">
<p key={col.fieldApiName}>{col.fieldApiName} : {col.label}</p>
</template>
</div>
</template>
</div>
</lightning-card>
</template>
Creates a record.
Syntax
import { createRecord } from 'lightning/uiRecordApi';
createRecord(recordInput: Record): Promise<Record>
Example File Name : createRecordDemo.js
import { LightningElement } from 'lwc';
import {createRecord} from 'lightning/uiRecordApi'
import CONTACT_OBJECT from '@salesforce/schema/Contact'
import { ShowToastEvent } from 'lightning/platformShowToastEvent';
export default class CreateRecordDemo extends LightningElement {
formFields={}
changeHandler(event){
const {name, value} = event.target
this.formFields[name]=value
}
createContact(){
const recordInput = {apiName:CONTACT_OBJECT.objectApiName, fields:this.formFields}
createRecord(recordInput).then(result=>{
this.showToast('Success!!', `contact created with is ${result.id}`)
this.template.querySelector('form.createForm').reset()
this.formFields={}
}).catch(error=>{
this.showToast('Error Creating record', error.body.message, 'error')
})
}
showToast(title, message, variant){
this.dispatchEvent(new ShowToastEvent({
title,
message,
variant:variant || 'success'
}))
}
}
File Name : createRecordDemo.html
<template>
<lightning-card title="create record form demo">
<div class="slds-p-around_medium">
<form class="createForm">
<lightning-input label="First Name" name="FirstName"
onchange={changeHandler} class="slds-m-bottom_x-small"></lightning-input>
<lightning-input label="Last Name" name="LastName"
onchange={changeHandler} class="slds-m-bottom_x-small"></lightning-input>
<lightning-input label="Title" name="Title"
onchange={changeHandler} class="slds-m-bottom_x-small"></lightning-input>
<lightning-input type="tel" label="Phone" name="Phone"
onchange={changeHandler} class="slds-m-bottom_x-small"></lightning-input>
<lightning-input type="email" label="Email" name="Email"
onchange={changeHandler} class="slds-m-bottom_x-small"></lightning-input>
<lightning-button label="Create Contact" variant="brand" onclick={createContact}></lightning-button>
</form>
</div>
</lightning-card>
</template>
- Apex Method must be static and either global or public
- Method should be annotated with @AuraEnabled
Syntax
public with sharing class AccountController {
@AuraEnabled(cacheable=true)
public static List<Account> getAccountList() {
return [SELECT Id, Name, Type, Industry from Account];
}
}
Example
File Name : AccountController.cls
public with sharing class AccountController {
@AuraEnabled(cacheable=true)
public static List<Account> getAccountList(){
return [SELECT Id, Name, Type, Industry from Account LIMIT 5];
}
}
Use default import syntax in JavaScript to import an Apex method via the @salesforce/apex scoped packages.
Syntax
import apexMethodName from "@salesforce/apex/Namespace.Classname.apexMethodReference';
apexMethodName — A symbol that identifies the Apex method.
apexMethodReference — The name of the Apex method to import.
Classname — The name of the Apex class.
Namespace — lf the class is in the same namespace as the component, don't specify a namespace. If the class is in a managed package, specify the namespace of the managed package.
Example
import { LightningElement } from 'lwc';
import getAccountList from '@salesforce/apex/AccountController.getAccountList';
export default class ApexWireDemo extends LightningElement {
}
import { LightningElement, wire } from 'lwc';
import getAccountList from '@salesforce/apex/AccountController.getAccountList'
export default class ApexWireDemo extends LightningElement {
accountList
@wire(getAccountList)
accounts
@wire(getAccountList)
accountsHandler({data, error}){
if(data){
this.accountList = data.map(item=>{
let newType = item.Type === 'Customer - Channel' ? 'Channel':
item.Type === 'Customer - Direct' ? 'Direct':'-------'
return {...item, newType}
})
}
if(error){
console.error(error)
}
}
}
<template>
<lightning-card title="Apex Wire to Property Demo">
<div class="slds-p-around_medium">
<template if:true={accounts.data}>
<template for:each={accounts.data} for:item="account">
<div class="slds-box sldx-box_xx-small" key={account.Id}>
<p><strong>Name : </strong>{account.Name}</p>
<p><strong>Type : </strong>{account.Type}</p>
<p><strong>Industry : </strong>{account.Industry}</p>
</div>
</template>
</template>
</div>
</lightning-card>
<div class="slds-m-top_medium"></div>
<lightning-card title="Apex Wire To Function Demo">
<div class="slds-p-around_medium">
<template if:true={accountList}>
<template for:each={accountList} for:item="account">
<div class="slds-box slds-box_xx-small" key={account.Id}>
<p><strong>Name : </strong> {account.Name}</p>
<p><strong>Type : </strong> {account.newType}</p>
<p><strong>Industry : </strong> {account.Industry}</p>
</div>
</template>
</template>
</div>
</lightning-card>
</template>
File Name : AccountController.cls
public with sharing class AccountController {
@AuraEnabled(cacheable=true)
public static List<Account> getAccountList(String type){
return [SELECT Id, Name, Type, Industry from Account WHERE Type:type LIMIT 5];
}
}
File Name : wireApexWithParams.js
import { LightningElement, wire } from 'lwc';
import filterAccountType from '@salesforce/apex/AccountController.filterAccountType'
export default class WireApexWithParams extends LightningElement {
selectedType=''
@wire(filterAccountType, {type:'$selectedType'})
filteredAccounts
get typeOptions(){
return [
{label:"Customer - Channel", value:"Customer - Channel"},
{label:"Customer - Direct", value:"Customer - Direct"}
]
}
typeHandler(event){
this.selectedType = event.target.value
}
}
File Name : wireApexWithParams.html
<template>
<lightning-card title="Apex Wire Demo with params">
<div class="slds-var-p-around_medium">
<lightning-combobox
name="type"
lable="Choose your Type"
value={selectedType}
options={typeOptions}
onchange={typeHandler}
></lightning-combobox>
<template if:true={filteredAccounts.data}>
<template for:each={filteredAccounts.data} for:item="account">
<div class="slds-box slds-box_xx-small" key={account.Id}>
<p><strong>Name:</strong> {account.Name}</p>
<p><strong>Type:</strong> {account.Type}</p>
</div>
</template>
</template>
</div>
</lightning-card>
</template>
Use this approach over @wire in the following situations
- To call a method that isn't annotated with cacheable=true, which includes any method that inserts, updates, or deletes data.
- To control when the invocation occurs.
- To work with objects that aren't supported by User Interface API, like Task and Event.
- To call a method from an ES6 module that doesn't extend LightningElement
File Name : AccountController.cls
public with sharing class AccountController {
@AuraEnabled(cacheable=true)
public static List<Account> getAccountList(){
return [SELECT Id, Name, Type, Industry from Account LIMIT 5];
}
@AuraEnabled(cacheable=true)
public static List<Account> filterAccountType(String type){
return [SELECT Id, Name, Type from Account where Type=:type LIMIT 5];
}
@AuraEnabled(cacheable=true)
public static List<Account> findAccounts(String searchKey){
String key = '%' + searchKey + '%';
return [SELECT Id, Name, Type, Industry FROM Account WHERE Name LIKE :key LIMIT 5];
}
}
File Name : apexImperativeWithParamsDemo.js
import { LightningElement } from 'lwc';
import findAccounts from '@salesforce/apex/AccountController.findAccounts'
export default class ApexImperativeWithParamsDemo extends LightningElement {
searchKey=''
accounts
timer
searchHandler(event){
window.clearTimeout(this.timer)
this.searchKey = event.target.value
this.timer = setTimeout(()=>{
this.callApex()
}, 1000);
}
callApex(){
findAccounts({searchKey:this.searchKey})
.then(result=>{
this.accounts = result
}).catch(error=>{
console.error(error)
})
}
}
File Name : apexImperativeWithParamsDemo.html
<template>
<lightning-card title="Apex imperative with params demo">
<div class="slds-p-around_medium">
<lightning-input
type="search"
onchange={searchHandler}
label="Search Account"
value={searchKey}
></lightning-input>
</div>
<template if:true={accounts}>
<template for:each={accounts} for:item="account">
<div class="slds-box slds-box_xx-small" key={account.Id}>
<p>Name - {account.Name}</p>
<p>Type - {account.Type}</p>
<p>Industry - {account.Industry}</p>
</div>
</template>
</template>
</lightning-card>
</template>
commuteFlowToLWC.html
<template>
<lightning-input label="Address" value={myname}></lightning-input>
</template>
CommuteFlowToLWC.js
import { LightningElement, api } from 'lwc';
export default class CommuteFlowToLWC extends LightningElement {
name;
connectedCallback() {
console.log(this.myname);
}
@api
get myname() {
return this.name;
}
set myname(data) {
this.name = data.toUpperCase() + ' bansal';
}
}
commuteFlowToLWC.js-meta.xml
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>59.0</apiVersion>
<isExposed>true</isExposed>
<targets>
<target>lightning__FlowScreen</target>
</targets>
<targetConfigs>
<targetConfig targets="lightning__FlowScreen">
<property name="myname" type="String" label="Shikha Input" />
</targetConfig>
</targetConfigs>
</LightningComponentBundle>
-
Create a Screen flow and name it Flow to Lightning Web Component Communication
-
Add the Lightning Web Component on Screen you created previously and pass text input element value to it
commuteLWCToFlow.html
<template>
<lightning-input label="Email" type="email" onchange={getEmail}></lightning-input>
</template>
commuteLWCToFlow.js
import { LightningElement, api } from 'lwc';
import {FlowAttributeChangeEvent} from 'lightning/flowSupport';
export default class CommuteLWCToFlow extends LightningElement {
@api email;
getEmail(event){
// eslint-disable-next-line @lwc/lwc/no-api-reassignments
this.email = event.target.value;
this.dispatchEvent(new FlowAttributeChangeEvent('email', this.email))
}
}
commuteLWCToFlow.js-meta.xml
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>59.0</apiVersion>
<isExposed>true</isExposed>
<targets>
<target>lightning__FlowScreen</target>
</targets>
<targetConfigs>
<targetConfig targets="lightning__FlowScreen">
<property name="email" type="String" label="Email Input" role="outputOnly" />
</targetConfig>
</targetConfigs>
</LightningComponentBundle>
There are two ways to communicate between Independent Components
- pubsub
- Lightning Messaging Service
NOTE ** Use this approach, if Lightning Messaging Service not serve your purpose. It's an Old technique to communicate with the independent components in LWC
PubSub module is nothing, just a JavaScript file that contains different methods that are exported so that the other components can make use of it. It is not provided to you by default, so you have to add it by yourself. It can be simply created by creating a new lightning web component with only JavaScript and meta.xml file.
Let’s take a look over the pubsub.js file.
// Filename: pubsub.js
/* eslint-disable no-console */
const store = {};
/**
* subscribers a callback for an event
* @param {string} eventName - Name of the event to listen for.
* @param {function} callback - Function to invoke when said event is fired.
*/
const subscribe = (eventName, callback) => {
if (!store[eventName]) {
store[eventName] = new Set();
}
store[eventName].add(callback);
};
/**
* unsubscribe a callback for an event
* @param {string} eventName - Name of the event to unsubscribe from.
* @param {function} callback - Function to unsubscribe.
*/
const unsubscribe = (eventName, callback) => {
if (store[eventName]) {
store[eventName].delete(callback);
}
};
/**
* Publish an event to listeners.
* @param {string} eventName - Name of the event to publish.
* @param {*} payload - Payload of the event to publish.
*/
const publish = (eventName, payload) => {
if (store[eventName]) {
store[eventName].forEach(callback => {
try {
callback(payload);
} catch (error) {
console.error(error);
}
});
}
};
export default {
subscribe,
unsubscribe,
publish
};
// Component PubsubComponentA
<template>
<lightning-card title="Pubsub demo Component A">
<div class="slds-p-around_medium">
<lightning-input type="text" onkeyup={inputHandler} class="slds-m-bottom_medium"></lightning-input>
<lightning-button variant="brand" onclick={publishHandler} label="publish"></lightning-button>
</div>
</lightning-card>
</template>
import { LightningElement } from 'lwc';
import pubsub from 'c/pubsub'
export default class PubsubComponentA extends LightningElement {
message
inputHandler(event){
this.message = event.target.value
}
publishHandler(){
pubsub.publish('componentA', this.message)
}
}
// Component PubsubComponentB
<template>
<lightning-card title="Pubsub demo Component B">
<div class="slds-p-around_medium">
Message Recieved - {message}
</div>
</lightning-card>
</template>
import { LightningElement } from 'lwc';
import pubsub from 'c/pubsub'
export default class PubsubComponentB extends LightningElement {
message
connectedCallback(){
this.callSubscriber()
}
callSubscriber(){
pubsub.subscribe('componentA', (message)=>{
this.message = message
})
}
}
Reference Link : https://medium.com/@sendtosachin27/communication-through-lms-in-lwc-748290185c9c
How Component will communicate through Lightning Message Service (LMS) in LWC
- Make a folder under force-app/main/default with name "messageChannels".
- Create an xml file "messageChannelName.messageChannel-meta.xml"
- Define lightning message field in .xml file, see code below
<?xml version="1.0" encoding="UTF-8"?>
<LightningMessageChannel xmlns="http://soap.sforce.com/2006/04/metadata">
<masterLabel>SampleMessageChannel</masterLabel>
<isExposed>true</isExposed>
<description>This is a sample Lightning Message Channel.</description>
<!--A list of message payload fields for a given Lightning Message Channel.-->
<lightningMessageFields>
<fieldName>recordId</fieldName>
<description>This is the record Id that changed</description>
</lightningMessageFields>
<lightningMessageFields>
<fieldName>recordData</fieldName>
<description>The current data representing the record that changed</description>
</lightningMessageFields>
</LightningMessageChannel>
Import message service
import msgService from '@salesforce/messageChannel/messageChannelName__c';
import message service features
import {publish,subscribe,unsubscribe,APPLICATION_SCOPE,MessageContext} from 'lightning/messageService';
Define the Scope of the Message Service The Lightning message service lets you define the scope of where subscribing components receive messages in your application.
For Lightning web components, the scoping feature is available only when using @wire adapter
@wire(MessageContext)
messageContext
Publish Message Channel
To publish message, we have publish() method in Lightning message service’s.
publish() method accept 3 parameter :-
- Message Context (Type Object)
- Message Channel (Type Object)
- Message Payload (The message payload is a JSON object)
// Component lmsComponentA
<template>
<lightning-card title="LWC LMS Component A">
<div class="slds-m-around_medium">
<lightning-input type="text" label="Enter message to publish" onkeyup={inputHandler}></lightning-input>
</div>
<div class="slds-m-around_medium">
<lightning-button label="Publish" onclick={publishButtonHandler}></lightning-button>
</div>
</lightning-card>
<c-lms-component-b></c-lms-component-b>
</template>
import SAMPLEMC from '@salesforce/messageChannel/SampleMessageChannel__c';
import { MessageContext, publish } from 'lightning/messageService';
import { LightningElement, wire } from 'lwc';
export default class LmsComponentA extends LightningElement {
inputValue;
@wire(MessageContext)
context;
inputHandler(event) {
this.inputValue = event.target.value;
}
publishButtonHandler(){
const message = {
recordId : 1234321,
lmsData: {
value : this.inputValue
},
myMsg : 'Hello'
}
console.log('message => ', message);
//publish(messageContext: Object, messageChannel: Object, message?: Object, publisherOptions?: Object)
publish(this.context,SAMPLEMC, message);
}
}
// Component lmsComponentB
<template>
<lightning-card title="LWC LMS Component A">
<div class="slds-m-around_medium">
<p>
Received Message : <strong>{receivedMessage}</strong>
</p>
</div>
</lightning-card>
<c-lms-component-y></c-lms-component-y>
</template>
import SAMPLEMC from '@salesforce/messageChannel/SampleMessageChannel__c';
import { APPLICATION_SCOPE, MessageContext, subscribe } from 'lightning/messageService';
import { LightningElement, wire } from 'lwc';
export default class LmsComponentX extends LightningElement {
receivedMessage;
@wire(MessageContext)
context;
connectedCallback() {
this.subscribeMessage();
}
subscribeMessage( ){
subscribe(this.context, SAMPLEMC, (message) => {this.handleMessage(message)}, {scope: APPLICATION_SCOPE})
}
handleMessage(message){
this.receivedMessage = message.lmsData.value ? message.lmsData.value : 'No Message'
console.log(message);
}
}
Locker Service is a security architecture in Salesforce that enhances the security of Lightning components (both Aura and Lightning Web Components).
Lightning Locker is a layer which sits in between your browser and DOM (document object). In other words, Lightning Locker is a virtual browser that allows only secure request to go through and have access to real DOM. This virtual browser sits in front of your unsafe real browser.
You can disable lightning locker by changing the api version of lightning component bundle to 39 or below.
Lightning locker is automatically enabled for component bundles having api version 40 or above.
To load your LWC or Aura Component inside the VF page we will use the $Lightning.use() method.
The $Lightning.use() function takes four arguments & they are as follows:
- appName(String): Required* (The name of our lightning dependency app, including namespace. For example, “c:visualforceAuraApplications”)
Note: We must follow ‘Camel Case’ notation while calling our LWC component inside our Aura Application
- Callback(function): A function to call once the Lightning Component framework and our app have fully loaded. The callback receives no arguments. This callback is usually where you call $Lightning.createComponent() to add our app to the page
- lightningEndPointURL(String): Optional if we are using inside Salesforce. The URL for the Lightning domain on your Salesforce instance. For example, https://MyDomain.lightning.force.com.
- authToken: Optional if we are using inside Salesforce.
VF Page Code:
<apex:page standardStylesheets="true" showHeader="false">
<apex:includeLightning />
<div id="LwcId" />
<script>
$Lightning.use("c:visualforceAuraApplications", function() {
$Lightning.createComponent("c:callingLwcInVfPages",
{
note : 'I am coming from VF Page', // You can pass the value to @api attributes if you have inside JavaScript Class.
recordId : '{!$CurrentPage.parameters.id}'
},
"LwcId", // the Id of div tag where your component will be rendered
function(cmp) {
console.log('Calling the LWC Component');
});
});
</script>
</apex:page>
Reference : https://hicglobalsolutions.com/blog/how-to-call-lwc-in-visualforce-pages/
Content security policy (CSP) is used by the Lightning Component framework to restrict content. The main goal is to aid in the prevention of cross-site scripting (XSS) and other code injection attacks.
Let’s add CSP trusted site to Salesforce org in order to make an API call from javascript code without error.
In this example, I’ll use the data-faker API hosted on Heroku: this is the endpoint https://data-faker.herokuapp.com that I’ll use to make a callout from javascript code.
Go to setup > search ‘ CSP ‘ > click on CSP Trusted Sites
Create a CSP trusted site by clicking on ‘New Trusted Site,’ as shown in the screenshot below.
We are now ready to make a callout from javascript code in the lightning web component.
import { LightningElement } from "lwc";
export default class AsyncDemo extends LightningElement {
connectedCallback() {
this.callThirdPartyApi(); // call using async await..
this.callThirdPartySecondApi(); // call using then catch.
}
async callThirdPartyApi() {
// here we are using try and catch to capture error.
try {
const response = await fetch('https://jsonplaceholder.typicode.com/todos/1');
const data = await response.json();
console.log("ThirdParty API Data", data); // data will print in
}
catch(error){
console.log('error', error.message);
}
}
callThirdPartySecondApi() {
fetch('https://jsonplaceholder.typicode.com/todos/1')
.then(response => response.json())
.then(json => console.log(json))
}
}
//to check standard permission
import hasPermission from '@salesforce/userPermission/PermissionName';
//to check custom permission
import hasCustomPermission from '@salesforce/customPermission/Custom_Permission_Api_Name';
// userPermissionCheck.js
import { LightningElement } from 'lwc';
import hasRunReports from '@salesforce/userPermission/RunReports';
export default class PermissionCheck extends LightningElement {
get isRunReport() {
return hasRunReports;
}
}
<!-- userPermissionCheck.html -->
<template>
<lightning-card title="User Permisison Check Example">
<template if:true={isRunReport}>
<h2>User has Permission to Run Reports.</h2>
</template>
<template if:false={isRunReport}>
<h2>User Does Not have Permission to Run Reports.</h2>
</template>
</lightning-card>
</template>