Tuesday, December 26, 2023

Export all Custom Metadata Records in CSV file using Apex

Using apex, we can export all custom metadata records in CSV files. This is sometime needed if we want to review all custom metadata records after sandbox refresh or to check the dependency of any field or attribute in all custom metadata records.


Below is apex class which can be used for this and once you execute its method from developer console, you will receive email with csv that contain all custom metadata records.

public class SK_CustomMetadataUtility {
public static void exportAllCMRecordsInCSV(){
Map<string,List<string>> objToFieldsMap = new Map<string,List<string>>();
Map<String, Schema.SObjectType> schemaMap = Schema.getGlobalDescribe();
for(String ss1: schemaMap.keyset()){
Schema.SObjectType objToken=schemaMap.get(ss1);
//find details about sobject
Schema.DescribeSObjectResult objDescribe=objToken.getdescribe();
string objAPIName= string.valueof(objDescribe.getName());
//filter object for which APi name ends with _mdt
//filter manage package objects as they contain __ twice in their API name
if(objAPIName.endsWith('__mdt') && objAPIName.countMatches('__')==1){
//system.debug('***custom metadata name:'+objAPIName);
objToFieldsMap.put(objAPIName,new List<string>());
//finding all fields of sobjects
Map<String, Schema.SObjectField> fieldMap = objDescribe.fields.getMap();
objToFieldsMap.get(objAPIName).add('DeveloperName');
objToFieldsMap.get(objAPIName).add('MasterLabel');
for(String ss:Fieldmap.keyset()){
Schema.DescribeFieldResult fd=fieldMap.get(ss).getDescribe();
string fieldAPIName=string.valueof(fd.getName());
//check if field is custom or not
if(fd.isCustom()){
objToFieldsMap.get(objAPIName).add(fieldAPIName);
}
}
}
}
system.debug('**objToFieldsMap size:'+objToFieldsMap.size());
system.debug('**objToFieldsMap:'+objToFieldsMap);
integer count =0;
string csvString = 'Name,DeveloperName,MasterLabel,Fields\n';
for(string objAPIName: objToFieldsMap.keyset()){
if(count<100){ //here we are restricting script to export only 100 custom metadata records
Map<string,List<string>> developerNameWithFieldValuesMap= new Map<string,List<string>>();
String qryString = 'SELECT '+ string.join(objToFieldsMap.get(objAPIName),',') +' From '+objAPIName;
system.debug('**qryString-'+objAPIName+'-:'+qryString);
for(sobject sb: Database.query(qryString)){
string developerName= string.valueof(sb.get('developerName'));
developerNameWithFieldValuesMap.put(developerName,new List<string>());
for(string field: objToFieldsMap.get(objAPIName)){
if(sb.get(field)!=null && field!='developerName' && field!='MasterLabel'){
//Fields coulm will contains Field API and its corresponding value
//here &SK& is delimiter to differentiate the API name with value
developerNameWithFieldValuesMap.get(developerName).add(field+'&SK&'+sb.get(field));
}
}
csvString = csvString + objAPIName+','+string.valueof(sb.get('DeveloperName')).escapeCsv()+','+ string.valueof(sb.get('MasterLabel')).escapeCSV()+','+ string.join(developerNameWithFieldValuesMap.get(developerName),',').escapecsv()+'\n';
}
count= +1;
}
}
Messaging.EmailFileAttachment csvAttc = new Messaging.EmailFileAttachment();
blob csvBlob = Blob.valueOf(csvString);
string csvname= 'Custom metadata records.csv';
csvAttc.setFileName(csvname);
csvAttc.setBody(csvBlob);
Messaging.SingleEmailMessage email =new Messaging.SingleEmailMessage();
String[] toAddresses = new list<string> {UserInfo.getUserEmail()};
String subject = 'Custom metadata records -'+system.now();
email.setSubject(subject);
email.setToAddresses( toAddresses );
email.setPlainTextBody(subject);
email.setFileAttachments(new Messaging.EmailFileAttachment[]{csvAttc});
Messaging.SendEmailResult [] r = Messaging.sendEmail(new Messaging.SingleEmailMessage[] {email});
}
}

Open developer console and open execute anonymous window. Execute below script:

SK_CustomMetadataUtility.exportAllCMRecordsInCSV();


Below is snapshot of csv file:


Note:

  • This script will specify the custom metadata field api name and its value seperated by "&SK&" delimiter.
  • All fields and its values are seperated by comma.
  • You can modify this method to fetch all custom metadata related to installed packages by specifying a parameter for namespace.


Hope this will help!!

 

Sunday, August 20, 2023

Open Standard Email composer from LWC Component - Global Email Quick Action

As we all know that we can create Email Actions in Salesforce but capability to invoke Email Action from LWC components was not present. After Winter' 23, Salesforce allowed to invoke global email action from custom components (from Aura and LWC both).

In this blog, I am going to explain how to invoke email action from LWC component. You can launch email composer from custom button. 

We have to use the lightning-navigation and lightning-page-reference-utils components to create a QuickAction (Global) Send Email action, which opens an email draft with pre-populated field values.

We will create simple LWC component which will have button. Once we click on this button, email composer will open. Below is sample code for reference:

<template>
<lightning-card title="Launch Standard Email Composer in LWC">
<div style="background-color:#E6E6FA;border-style: solid;height:200px;margin:5%;padding:2%;">
<lightning-button variant="neutral" label="Send an Email" onclick={onButtonClick}></lightning-button>
</div>
</lightning-card>
</template>
import { LightningElement , api, track} from 'lwc';
import { NavigationMixin } from 'lightning/navigation';
import { encodeDefaultFieldValues } from 'lightning/pageReferenceUtils';
import findEmailInfo from '@salesforce/apex/skStandardEmailComposerController.findEmailInfo';
export default class SkStandardEmailComposer extends NavigationMixin(LightningElement) {
@api emailInfo;
@api error;
connectedCallback(){
this.findEmailInformation();
}
findEmailInformation(){
findEmailInfo()
.then((result) => {
this.emailInfo = result;
console.log('***email details-'+JSON.stringify(this.emailInfo));
})
.catch((error) => {
this.error = error;
});
}
onButtonClick(){
console.log('***onButtonClick-recordRelatedToId:'+this.emailInfo.recordRelatedToId);
var pageRef = {
type: "standard__quickAction",
attributes: {
apiName: "Global.SendEmail"
},
state: {
recordId: this.emailInfo.recordRelatedToId,
defaultFieldValues:
encodeDefaultFieldValues({
HtmlBody : "Pre-populated text for the email body.",
Subject : "Pre-populated Subject of the Email",
CcAddress : this.emailInfo.ccAddress.join(','),
ToAddress : this.emailInfo.toAddress.join(','),
ValidatedFromAddress : this.emailInfo.fromAddress,
})
}
};
this[NavigationMixin.Navigate](pageRef);
}
}
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>58.0</apiVersion>
<isExposed>true</isExposed>
<targets>
<target>lightning__AppPage</target>
</targets>
</LightningComponentBundle>
public class skStandardEmailComposerController {
@AuraEnabled
public static emailWrapper findEmailInfo(){
emailWrapper returnValue= new emailWrapper();
//you can implement your own logic to
returnValue.ccAddress.add('sk111111demo@gmail.com');
returnValue.ccAddress.add('sk22222demo@gmail.com');
returnValue.toAddress.add('skemailtocaseEmail1@gmail.com');
returnValue.fromAddress='skcurrentUser@gmail.com';//make sure it is some user address or org wide address
returnValue.recordRelatedToId=[select id from Account Limit 1].Id;
return returnValue;
}
//in wrapper you can define define ccAddress,toAddress,from address, emailBody,EmailSubject
public class emailWrapper{
@auraEnabled
public string recordRelatedToId;
@AuraEnabled
public List<string> ccAddress;
@AuraEnabled
public List<string> toAddress;
@AuraEnabled
public string fromAddress;
public emailWrapper(){
ccAddress = new List<String>();
toAddress = new List<String>();
}
}
}


Note:

  • You can default email properties like subject, body, ccAddress, toAddress and fromEmailAddress.
  • You always need to provide record. Whenever you click on send button after composing email, system will create an emailmessage record with relatedToId as specified recordId.
  • encodeDefaultFieldValues is being provide to specify all default values.

Snapshots for reference:




Hope this will help!!






Thursday, August 3, 2023

Automatically Remove Special Characters from Filename while Uploading Files in Salesforce

 As we all know that we can perform URL hack to download all files related to parent record from salesforce. We have observed that sometime downloaded file name is changed to contentversion recordId by salesforce instead of actual file name. 

It is my observation that if file name contains special characters, then when we perform download All, file name is changed to "068xxxxxxxxxxxxxx.pdf".

I have added 2 files to account record for reference:



Now I can generate download URL if I have contentversionId of these 2 files. In my case the download URL will be:

https://xxxx/sfc/servlet.shepherd/version/download/0680K00000kLSdbQAG/0680K00000kLSfwQAG

where xxxx is you org domain URL.

You can use below script to generate download all files URL if you know the parent record Id.

string parentRecordId='0012800001IrqlyAAB';
string orgbaseUrl=URL.getSalesforceBaseUrl().toExternalForm();
string downloadAllFileUrl=orgbaseUrl+'/sfc/servlet.shepherd/version/download';
List<string> contentVersionIdList = new List<string>();
for(ContentDocumentLink cdl:[SELECT ContentDocumentId,ContentDocument.title,ContentDocument.LatestPublishedVersionId FROM ContentDocumentLink where LinkedEntityId =:parentRecordId]){
contentVersionIdList.add(string.valueof(cdl.ContentDocument.LatestPublishedVersionId));
}
downloadAllFileUrl = downloadAllFileUrl +'/'+string.join(contentVersionIdList,'/');
system.debug('***downloadAllFileUrl:'+downloadAllFileUrl);

If I open this URL, I will zip file which will contain both files and when I will extract it, it will appear as shown below:


As my observation, if any special character is present in file name then salesforce changes the file name. If "-" character is present, then salesforce won't change file name.

For this kind of scenarios, I recommend to have script in place which will remove all special characters from file name whenever file is being uploaded. You can write trigger on contentVersion object to remove special characters from file name. Below is code snippet which can help:



//purpose- remove special characters from file names when uploaded in salesforce
trigger SK_ContentVersionTrigger on ContentVersion (before insert) {
for(ContentVersion cv: trigger.new){
if(string.isNotBlank(cv.Title)){
cv.Title= cv.Title.replaceAll('[^-a-zA-Z0-9\\s+]', '');
System.debug('***updated Title '+cv.Title);
}
}
}
Hope this will help!!!!



Wednesday, July 5, 2023

FieldDefinition - How to get list of all fields present in any Object using Tooling API in Salesforce

We can utilize Tooling API to get information of all fields present in a object. You can put different filters like to get all fields or specific fields like auto number fields, ExternalID fields etc.

Below is apex class which contains method to get all fields. Remember that callout uses Named Credential in order to do handshake with SFDC org.

This method contains 3 parameters:

  • selOrgNCName : Name of Named Credential created to connect to SFDC org
  • objAPIName : Specify the object API name in order to get list of all fields
  • namespacePrefix : Specify namespacePrefix if you want to retrieve fields related to any installed packages. If you want to get list of fields created in SFDC org, then specify blank or NULL.
Note:
  • I have added filter not to return auto number fields. If you want to get those fields details then remove it from query string.
  • If you want to query only External Id fields then specify filter as DataType+Like+\'%25(External+ID)%25\'';

public class SK_ToolingAPIFieldsUtility {
@AuraEnabled
public static DM_FieldWrapper findAllFields(string selOrgNCName,string objAPIName,string NamespacePrefix){
DM_FieldWrapper returnValue= new DM_FieldWrapper();
string URIForFields='/services/data/v54.0/tooling/query/?q=Select+id,Datatype,ReferenceTo,ExtraTypeInfo,IsCalculated,IsCompound,IsIndexed,ValueTypeId,RelationshipName,NamespacePrefix,Label,IsNameField,QualifiedApiName,developerName+from+FieldDefinition+';
URIForFields=URIForFields + 'where+EntityDefinition.QualifiedApiName=\''+objAPIName+'\'+AND+IsCalculated=false+';
URIForFields=URIForFields + '+AND+DataType!=\'Auto+Number\'';
string endpointURL='callout:'+selOrgNCName+URIForFields;
if(string.isNotBlank(NamespacePrefix)){
endpointURL=endpointURL+'+AND+NamespacePrefix=\''+NamespacePrefix+'\'';
}
system.debug('***endpointURL:'+endpointURL);
HttpRequest req = new HttpRequest();
req.setEndpoint(endpointURL);
req.setMethod('GET');
Http http = new Http();
HTTPResponse res = http.send(req);
DM_FieldWrapper JSONDetails= DM_FieldWrapper.parse(res.getBody());
System.debug('****field count-:'+JSONDetails.records.size());
for(cls_records fdata : JSONDetails.records){
system.debug('****fdata.QualifiedApiName:'+fdata);
}
returnValue= JSONDetails;
return returnValue;
}
public class DM_FieldWrapper{
public Integer size;
public Integer totalSize;
public boolean done;
public String entityTypeName;
public cls_records[] records;
}
public class cls_records {
public String Id;
public String DataType;
public boolean IsCalculated;
public boolean IsCompound;
public boolean IsIndexed;
public string RelationshipName;
public string NamespacePrefix;
public String Label;
public boolean IsNameField;
public String QualifiedApiName;
public String DeveloperName;
public ReferenceTo ReferenceTo;
public cls_EntityDefinition EntityDefinition;
}
public class ReferenceTo {
public List<String> referenceTo;
}
public class cls_EntityDefinition {
public String QualifiedApiName;
public string MasterLabel;
}
public static DM_FieldWrapper parse(String json){
return (DM_FieldWrapper) System.JSON.deserialize(json, DM_FieldWrapper.class);
}
}

Hope this will help!!



Tuesday, July 4, 2023

EntityDefinition - How to get list of all objects present in any SFDC org using tooling API

Salesforce provide Metadata API through which we can metadata information present in any SFDC org. Metadata API is SOAP based which involves consuming WSDL and then calling methods provided in WSDL.

Salesforce has introduced Tooling API which is REST based and can be used to get metadata information from any salesforce org. Tooling API is very easy to implement.

EntityDefinition provides information about standard and custom metadata. Before invoking any Tooling API endpoint, we need to do handshake with SFDC org. You can utilise Oauth 2.0 (username-password flow) or create "Named Credential" for SFDC org from which you want to fetch information.


Below is Apex class which contains method through which we can get all object details:

"findAllObjects" Method contains 3 parameters: 

  • selOrgNCName : Name of Named Credential created to connect to SFDC org
  • namespacePrefix : Specify namespacePrefix if you want to retrieve objects related to any installed packages. If you want to get list of objects created in SFDC org, then specify blank or NULL.
  • objName : specify search string if you want to get list of specific objects that contain that string. Pass blank or NULL if you want to retrieve all objects.

public class SK_ToolingAPIUtility {
@AuraEnabled
public static List<checkboxGroupValuesWrapper> findAllObjects(string selOrgNCName,string namespacePrefix,string objName){
system.debug('****selOrgNCName:'+selOrgNCName);
system.debug('****NamespacePrefix:'+namespacePrefix);
system.debug('****objName:'+objName);
List<checkboxGroupValuesWrapper> returnList=new List<checkboxGroupValuesWrapper>();
try {
string URIForObjects='/services/data/v54.0/tooling/query/?q=Select+id,NamespacePrefix,MasterLabel,QualifiedApiName,DeveloperName+from+Entitydefinition+where+IsCustomSetting=false+and+IsCustomizable=true';
string endpointURL='callout:'+selOrgNCName+URIForObjects;
//endpointURL='select+id+from+Account';
if(string.isNotBlank(NamespacePrefix)){
endpointURL=endpointURL+'+AND+NamespacePrefix=\''+NamespacePrefix+'\'';
}if(string.isNotBlank(objName)){
endpointURL=endpointURL+'+AND+MasterLabel+Like+\'%25'+objName+'%25\'';
}
endpointURL=endpointURL+'+order+by+MasterLabel';
system.debug('***endpointURL:'+endpointURL);
HttpRequest req = new HttpRequest();
req.setEndpoint(endpointURL);
req.setMethod('GET');
Http http = new Http();
HTTPResponse res = http.send(req);
DM_ObjectWrapper JSONDetails= DM_ObjectWrapper.parse(res.getBody());
System.debug('****object count-:'+JSONDetails.records.size());
for(cls_records eachRec:JSONDetails.records){
//System.debug('****eachrec-:'+eachRec);
checkboxGroupValuesWrapper objInfo=new checkboxGroupValuesWrapper();
objInfo.label=eachRec.MasterLabel +'('+eachRec.QualifiedApiName+')';
objInfo.value=eachRec.QualifiedApiName;
returnList.add(objInfo);
system.debug('*****objInfo:'+objInfo);
}
} catch (Exception ex) {
system.debug('****error-'+ex.getMessage());
}
system.debug('*****returnList:'+returnList);
return returnList;
}
public class checkboxGroupValuesWrapper{
@AuraEnabled
public string label;
@AuraEnabled
public string value;
}
public class DM_ObjectWrapper {
public Integer size;
public Integer totalSize;
public boolean done;
public string queryLocator;
public String entityTypeName;
public cls_records[] records;
}
public class cls_records {
public cls_attributes attributes;
public String Id;
public string NamespacePrefix;
public String MasterLabel;
public String QualifiedApiName;
public String DeveloperName;
}
public class cls_attributes {
public String type;
public String url;
}
public static DM_ObjectWrapper parse(String json){
return (DM_ObjectWrapper) System.JSON.deserialize(json, DM_ObjectWrapper.class);
}
}


Hope this will help!!

Monday, June 26, 2023

Composite Batch- Way to call multiple REST API in single request (sending multiple request to Salesforce)

Composite batch allows to send up to 25 separate API request in a single call to salesforce. Best thing about this is that all sub-request are considered as separate call. Suppose you want to move multiple records to next salesforce org or want to create multiple records in salesforce from external system then this will help you.


By using composite batch API, I am going to create a new Account record, Update existing Account record (by using external Id field -Account_Unique_Id__c) and query account record in same API call.

Below are details you need:

REST API service URI-   /services/data/v56.0/composite/batch/

Request Body Sample-

{

"batchRequests": [{

"method": "POST",

"url": "v56.0/sobjects/Account",

"richInput": {

"Name": "New Account using Composite Batch"

}

},

{

"method": "PATCH",

"url": "v56.0/sobjects/account/Account_Unique_Id__c/ACC-000033",

"richInput": {

"NumberOfEmployees": "2000",

"Type": "Customer"

}

}, {

"method": "GET",

"url": "v56.0/query/?q=SELECT+name+from+Account+where+Account_Unique_Id__c='ACC-000033'"

}

]

}


Response Body:

{

"hasErrors": false,

"results": [{

"statusCode": 201,

"result": {

"id": "0010K00002mziKJQAY",

"success": true,

"errors": []

}

}, {

"statusCode": 200,

"result": {

"id": "0010K00002PBQTyQAP",

"success": true,

"errors": [],

"created": false

}

}, {

"statusCode": 200,

"result": {

"totalSize": 1,

"done": true,

"records": [{

"attributes": {

"type": "Account",

"url": "/services/data/v56.0/sobjects/Account/0010K00002PBQTyQAP"

},

"Name": "Sunil Test Account"

}]

}

}]

}


Important points to consider while using this:

  • The response bodies and HTTP statuses of the subrequests in the batch are returned in a single response body.
  • Each sub request counts against rate limits.
  • Each API request are considered as separate and you can not pass information between them.
  • If one API request get successfully completed from batch request, then it gets committed. If any subsequent request fails then previous request is not rollbacked automatically.
  • Batch request should complete in 10 minutes. If batch times out then the remaining sub requests aren’t executed

Hope this will help!!



Sunday, March 26, 2023

Custom Events in Lightning Web Components : Pass data from child component to parent component in LWC

In order to communicate from child component to parent component in LWC, we use custom events. Custom events helps in passing data from child components to parent component in containment hierarchy.

If you want to refer on how same thing can be achieved in Aura Components then refer below article:

Component Events: Way to Communicate Between Components in Containment Hierarchy


Lightning component can create and dispatch events in a component’s JavaScript class. To create an event, use the CustomEvent() constructor. To dispatch an event, call the EventTarget.dispatchEvent() method.

The CustomEvent() constructor needs one required parameter, which is a string indicating what event has been performed. You can define ant string as event name. While defining event name, follw below recommendation based on DOM event standard.

  • No uppercase letters
  • No spaces
  • Use underscores to separate words

Steps involved in implementing custom events

1. Create custom event in child component and dispatch that event.

    const newEvent = new CustomEvent("search");
    this.dispatchEvent(newEvent);

2. Handle the event in parent component.

When you declare the child component in parent component, then use "on" event handler to execute logic whenever the event is fired. Suppose you event name which is dispateched from child component is "search", event handler name will be "onsearch".

<c-sk-child-cmp onsearch={capturedEvent}></c-sk-child-cmp>
capturedEvent(event) {
    console.log("**recieved event-->" + event);
}

Custom events to inform what action is performed in child component

If you want to inform parent component about what has been clicked or operation performed in child component, then you can simply define custom event as mentioned below:

<template>
    <lightning-button label="Edit" onclick={editAction}></lightning-button>
    <lightning-button label="New" onclick={deleteAction}></lightning-button>
</template>

    editAction() {
const newEvent = new CustomEvent("edit");
        this.dispatchEvent(newEvent);
    }

    deleteAction() {
const newEvent = new CustomEvent("delete");
        this.dispatchEvent(newEvent);
    }


Custom events to pass data to parent component

If you want to pass data to parent component, then set a detail property in the CustomEvent constructor. 

You can pass any record id  or any data from child component in detail property.

@track selectedRecordId;

handleResponse(event) {
const answerEvent = new CustomEvent("search", { detail: this.selectedRecordId});
this.dispatchEvent(answerEvent);
}

Parent component can access detail property in order to read data passed from child component

 capturedEvent(event) {
    console.log("**recieved event detail-->" + event.detail);
 }


Note: There is no restriction on data type of detail property but is is recommended to pass only primitive data. JavaScript passes all data types by reference except for primitives. If a component includes an object in its detail property, any listener can mutate that object without the component’s knowledge. This is a bad thing! It’s a best practice either to send only primitives, or to copy data to a new object before adding it to the detail property. Copying the data to a new object ensures that you’re sending only the data you want, and that the receiver can’t mutate your data.

Lets try to understand custom events through simple example:

We will create a child component "skChildCmp" which will contain few buttons. Once user will click on one of the button, child component will pass resource URL based on button click to parent component and parent component will display that URL.

Below is snapshot for parent component:


Now when user clicks on SFDCSTUFF BLOGS button, URL for that button will be appearing on parent component.


If user clicks on TRAILHEAD button, then parent will display URL for that.


Below is complete code snippet:


<template>
<h1>Click below buttons to get URL for resources</h1>
<lightning-button label="SFDCSTUFF BLOGS" onclick={sendMessageToParentCmp}></lightning-button>
<lightning-button label="TRAILHEAD" onclick={sendMessageToParentCmp}></lightning-button>
</template>
view raw skChildCmp.html hosted with ❤ by GitHub
import { LightningElement } from 'lwc';
export default class SkChildCmp extends LightningElement {
sendMessageToParentCmp(event) {
const selBtnName = event.target.label;
console.log('****selBtnName:'+selBtnName);
var selResourceURL;
if(selBtnName=='SFDCSTUFF BLOGS'){
selResourceURL='https://www.sfdcstuff.com/';
}if(selBtnName=='TRAILHEAD'){
selResourceURL='https://trailhead.salesforce.com/';
}
this.fireCustomEvent('selectedresource',selResourceURL);
}
fireCustomEvent(eventName, dataToSend){
const eventFromChild = new CustomEvent(eventName,{detail:dataToSend});
console.log('****eventFromChild:'+JSON.stringify(eventFromChild));
this.dispatchEvent(eventFromChild);
}
}
view raw skChildCmp.js hosted with ❤ by GitHub
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>56.0</apiVersion>
<isExposed>true</isExposed>
</LightningComponentBundle>
<template>
<lightning-card title="Custom Events in LWC">
<div style="background-color:#E6E6FA;border-style: solid;height:200px;margin:5%;padding:2%;">
<p>This is parent Component- container for skChildCmp</p>
<br/>
Display resource URL passed from child component using custom events---
<a href={selectedResourceURL}>{selectedResourceURL}</a>
<br/>
<div style="background-color:#7ff6f6;border-style: solid;padding:2%;">
<p>This is child component present inside skParentCmp</p>
<c-sk-child-cmp onselectedresource={childEventHandler}></c-sk-child-cmp>
</div>
</div>
</lightning-card>
</template>
import { LightningElement,track } from 'lwc';
export default class SkParentCmp extends LightningElement {
@track selectedResourceURL;
childEventHandler(event) {
console.log("**recieved event detail-->" + event.detail);
this.selectedResourceURL= event.detail;
}
}
view raw skParentCmp.js hosted with ❤ by GitHub
<?xml version="1.0" encoding="UTF-8"?>
<LightningComponentBundle xmlns="http://soap.sforce.com/2006/04/metadata">
<apiVersion>56.0</apiVersion>
<isExposed>true</isExposed>
<targets>
<target>lightning__AppPage</target>
</targets>
</LightningComponentBundle>

Hope this will help!!