Mar 29, 2025
Restore deleted Dataverse records
Mar 14, 2025
Send Email with dynamic excel attachment
Sending an email with associated excel sheet with details of child records is a generic requirement and also great way to pass child record details to a customer. For example, you may need to send an email to a customer with quote details and associate quote products as a excel attachment.
In my example I am sending email to a Account and Sender is a Queue (you may obviously use a System user). I am sending Opportunity details and associated excel will carry Opportunity Products. This Email will be shown in timeline of Opportunity hence regarding Object would be the Opportunity.
I have two separate code snippets here.
1) Sending the Email with Template and Excel
What we need to understand is we should create the Email first and then send as two steps.. This way, we get a room prior to sending, to attach the excel. Note how we pass different Ids like Sender, Template, Receiver and Regarding Object here. Also notice that we pass the excel as the attachment body.
Public void SendEmailWithPaymentEvaluationDetails(Guid templateId, Guid toAccountId, Guid fromQueueId, Guid regOpportunityId) { // Initiate Email InstantiateTemplateRequest request = new InstantiateTemplateRequest() { TemplateId = new Guid(templateId), ObjectId = invoiceRequest.Id, ObjectType = invoiceRequest.LogicalName }; InstantiateTemplateResponse response = (InstantiateTemplateResponse)OrgService.Execute(request); Entity email = response.EntityCollection[0]; Entity Fromparty = new Entity("activityparty"); Entity Toparty = new Entity("activityparty"); Toparty["partyid"] = new EntityReference("account", toAccountId); Fromparty["partyid"] = new EntityReference("queue", fromQueueId); email["from"] = new Entity[] { Fromparty }; email["to"] = new Entity[] { Toparty }; email["directioncode"] = true; email["regardingobjectid"] = new EntityReference("opportunity", regOpportunityId); Guid emailId = OrgService.Create(email); // Link the Attachment Entity attachment = new Entity("activitymimeattachment"); attachment["subject"] = "OpportunityId Product List"; attachment["filename"] = "OpportunityId Product List.xlsx"; attachment["body"] = Convert.ToBase64String(CompileExcelFile(regOpportunityId)); attachment["mimetype"] = "application/vnd.ms-excel"; attachment["attachmentnumber"] = 1; attachment["objectid"] = new EntityReference(email.LogicalName, emailId); attachment["objecttypecode"] = email.LogicalName; OrgService.Create(attachment); // Send Email SendEmailRequest sendEmailRequest = new SendEmailRequest { EmailId = emailId, TrackingToken = string.Empty, IssueSend = true }; SendEmailResponse sendEmailResponse = (SendEmailResponse)OrgService.Execute(sendEmailRequest); }
2) Compilation of Excel
Here we compile the excel which is called in above method when preparing the attachment.public byte[] CompileExcelFile(Guid regOpportunityId) { var exportToExcelRequest = new OrganizationRequest("ExportToExcel"); exportToExcelRequest.Parameters = new ParameterCollection(); exportToExcelRequest.Parameters.Add(new KeyValuePair<string, object>("View", new EntityReference("savedquery", new Guid("{4c523f5b-e8c5-4cb5-bc83-bf4ef934342d}")))); string stringFetchXml = @"<fetch distinct='false' no-lock='false' mapping='logical' returntotalrecordcount='true'> <entity name='opportunityproduct'> <attribute name='lineitemnumber' /> <attribute name='productname' /> <attribute name='description' /> <attribute name='baseamount' /> <filter> <condition attribute='opportunityid' operator='eq' value='{0}' /> </filter> </entity> </fetch>"; exportToExcelRequest.Parameters.Add(new KeyValuePair<string, object>("FetchXml", String.Format(stringFetchXml, regOpportunityId.ToString()))); exportToExcelRequest.Parameters.Add(new KeyValuePair<string, object>("LayoutXml", @" <grid name='resultset' object='2' jump='lineitemnumber' select='1' icon='1' preview='1'> <row name='result' id='opportunityproductid'> <cell name='lineitemnumber' width='100' /> <cell name='productname' width='200' /> <cell name='description' width='300' /> <cell name='baseamount' width='125' /> </row> </grid>")); exportToExcelRequest.Parameters.Add(new KeyValuePair<string, object>("QueryApi", "")); exportToExcelRequest.Parameters.Add(new KeyValuePair<string, object>("QueryParameters", new InputArgumentCollection())); var exportToExcelResponse = OrgService.Execute(exportToExcelRequest); if (exportToExcelResponse.Results.Any()) return exportToExcelResponse.Results["ExcelFile"] as byte[]; else return null; }
Feb 14, 2025
Programmatically create a draft Email using Email Template (C#)
Previously we discussed how to send an Email using a Email Template, but we noticed it just sends the email but no chance of create and save as a draft. In some instances we need to create the Draft email to be sent later after checking or/and modifications by the user.
In such situations we can use below code. One can say its simply possible to use CREATE message of the Org service to achieve this but it is not possible to use a Template which is a constrain.
By creating the draft first, it allows you to programmatically or manually attach attachments prior to sending the email.
Here InstantiateTemplateRequest message does the magic.
public void CreateDraftEmailToPrimaryContactOfAccount(Account account) { InstantiateTemplateRequest request = new InstantiateTemplateRequest() { TemplateId = new Guid("bf0b97c7-d5a3-4a3f-8771-a1cd737ab555"), ObjectId = account.Id, ObjectType = Account.LogicalName }; InstantiateTemplateResponse response = (InstantiateTemplateResponse)OrgService.Execute(request); Entity email = response.EntityCollection[0]; Entity Fromparty = new Entity("activityparty"); Entity Toparty = new Entity("activityparty"); Toparty["partyid"] = new EntityReference(Account.EntityLogicalName, account.PrimaryContactId.Id); Fromparty["partyid"] = new EntityReference("queue", new Guid("f14a45e9-fac5-4ba0-9a95-c07fe1adabf0")); email["from"] = new Entity[] { Fromparty }; email["to"] = new Entity[] { Toparty }; email["directioncode"] = true; email["regardingobjectid"] = new EntityReference(Account.EntityLogicalName, account.Id); Guid emailId = OrgService.Create(email); }
When ready if you need to send the Email programmatically, below SendEmailRequest message can be used as below.
SendEmailRequest sendEmailRequest = new SendEmailRequest { EmailId = emailId, TrackingToken = string.Empty, IssueSend = true }; SendEmailResponse sendEmailResponse = (SendEmailResponse)OrgService.Execute(sendEmailRequest);
Related Posts
Programmatically send Email with Template (c#)
Programmatically send Email with Template (c#)
This is a code snippet that sends Email programmatically (in C#) while using an Email Template.
Explained scenario, we send an Email to Primary Contact of the Account. We set Account as the regarding object of the Email so this Email activity will associate with Account and will be displayed in Account's timeline. Also we have to pass Account as the regarding object of Template since we want dynamic fields of Template to be filled with Account field values as necessary.
Actually SendEmailFromTemplateRequest message does the magic.
public void SendEmailToPrimaryContactOfAccount(Account account) { Entity Fromparty = new Entity("activityparty"); Entity Toparty = new Entity("activityparty"); Toparty["partyid"] = new EntityReference(Contact.EntityLogicalName, account.PrimaryContactId.Id); Fromparty["partyid"] = new EntityReference("queue", new Guid("f14a45e9-fac5-4ba0-9a95-c07fe1adabf0")); Entity email = new Entity("email"); email["from"] = new Entity[] { Fromparty }; email["to"] = new Entity[] { Toparty }; email["directioncode"] = true; email["regardingobjectid"] = new EntityReference(Account.EntityLogicalName, account.Id); SendEmailFromTemplateRequest emailUsingTemplateReq = new SendEmailFromTemplateRequest { Target = email, TemplateId = new Guid("bf0b97c7-d5a3-4a3f-8771-a1cd737ab555"), RegardingId = account.Id, RegardingType = Account.EntityLogicalName }; var emailUsingTemplateRes = OrgService.Execute(emailUsingTemplateReq); }
Please note I am sending this email from a queue record. Obviously you can send from a system users as well. In such case, from activity party has to be changed accordingly.
One thing to note is, this method just sends the email, so you are not save Email in draft status.
Related Posts
Programmatically create a draft Email using Email Template (C#)
Feb 8, 2025
Very useful Formula field
Formula fields are pretty useful type in my opinion. It helps us define some values in a field using existing values, but defining them as we want.
How to do:
Start just like you create any other field, and select Formula within the type selection.
Switch('Status Reason', Blank(), "", 'Status Reason (Contacts)'.Inactive,"Redirect to Sales Team", 'Status Reason (Contacts)'.'Not Verified',"Seek Approval from Manager", 'Status Reason (Contacts)'.Verified,"Ready send Stater Pack" )
Jan 24, 2025
Status Change trigger of a workflow
In classic workflow, we have Record Status change among many other events those can be used to trigger the workflow.
Previously we thought classic workflows will be history by now with introductions of Cloud Flows. Though cloud flows are pretty powerful in terms of collection of connectors we can use etc. Anyway, still its not helping the developers in many way;
- Connection strings are still a complete mess that easily jeopardies a deployment
- When logic of Cloud Flow is growing, the illustration becomes very big and not easy to have a one view even via a big monitor.
As per this writing, classic workflows still running in most of the implementations of the world and no sign of getting replaced that easily for a long time.