Hi all,
I hope this post will be helpful who would like to generate a word document using CRM data of a record and add that as an attachment to that record.
Example Scenario : I have an Employee Entity, where i need to generate payslip for that employee on click of a button inside the form:
Steps to be followed:(In Form)
1. Create an Entity with the name"Employee" and add fields which are required:
2. Add a field of datatype "Two Option" and make visiblity of field as hidden
3. Create a web Resource/Javascript and write the function below in the web resource
//Function to change the Two Option Field value on a button CLick
function changeValue()
{
var value=Xrm.Page.getAttribute("FiledNAme").getValue();
//Check the option fields value if it is yes/no (1/0)
if(value==0)
{
//set value to 1
Xrm.Page.getAttribute("FieldName").setValue(1);
//Save the updated value
Xrm.Page.data.entity.save();
}
4.Publish ur webresource
5.Add a custom button in form using RibbonWorkBench tool and call ur JavaScript function on that button click: Follow the link to add custom button and call Javascript function
http://www.develop1.net/public/page/Ribbon-Workbench-for-Dynamics-CRM-2011.aspx
6.Create a C# plugin with the below code:
using System;
using System.Diagnostics;
using System.Linq;
using System.ServiceModel;
using DocumentFormat.OpenXml;
using DocumentFormat.OpenXml.Packaging;
using DocumentFormat.OpenXml.Wordprocessing;
using System.Net;
using System.IO;
using HtmlAgilityPack;
using mshtml;
using System.Runtime.Serialization;
using Microsoft.Xrm.Sdk;
using Microsoft.Xrm.Sdk.Query;
using Microsoft.Xrm.Sdk.Messages;
using NotesFor.HtmlToOpenXml;
namespace payslip
{
public class MyPlugin : IPlugin
{
public void Execute(IServiceProvider serviceProvider)
{
IPluginExecutionContext context = (IPluginExecutionContext)serviceProvider.GetService(typeof(IPluginExecutionContext));
// TODO - If you require tracing, uncomment the following line
// ITracingService trace = (ITracingService)serviceProvider.GetService(typeof(ITracingService));
Entity entity = null;
// Check if the InputParameters property bag contains a target
// of the current operation and that target is of type DynamicEntity.
if (context.InputParameters.Contains("Target") && context.InputParameters["Target"] is Entity)
{
// Obtain the target business entity from the input parmameters.
entity = (Entity)context.InputParameters["Target"];
// TODO Test for an entity type and message supported by your plug-in.
if (context.PrimaryEntityName != "UR ENTITY NAME") { return; }
if (context.MessageName != "Update") { return; }
}
else
{
return;
}
try
{
string invoice_number = "";
DateTime joining_date;
string consultant_name = "";
string ctc_offered = "";
string billable_amount = "";
string tax_amount = "";
string total_amount = "";
DateTime invoice_date;
string j_date = "";
string i_date = "";
string tax_per = "";
IOrganizationServiceFactory serviceFactory = (IOrganizationServiceFactory)serviceProvider.GetService(typeof(IOrganizationServiceFactory));
IOrganizationService service = serviceFactory.CreateOrganizationService(context.UserId);
//Retrieve field from ur entity
Entity receivableEntity = service.Retrieve("new_receivable", entity.Id, new ColumnSet("new_nameoftheconsultant", "new_dateofjoining", "new_ctcoffered", "new_billamount", "new_taxamount", "new_totalamount", "new_invoicenumber", "new_invoicedate", "new_generateinvoice", "new_accountname", "new_projectname", "new_projecttype","new_taxesapplicable"));
EntityReference proj = (EntityReference)receivableEntity.Attributes["new_projectname"];
string projname = proj.Name;
EntityReference acc = (EntityReference)receivableEntity.Attributes["new_accountname"];
string name = acc.Name;
string add = "";
string city = "";
string amttot = "";
Guid id = acc.Id;
//string ret = receivableEntity.Attributes["new_accountname"].ToString();
QueryExpression q = new QueryExpression("account");
q.ColumnSet = new ColumnSet(new string[] { "new_companyaddress_test", "address2_city", "name" });
q.Criteria.AddCondition("name", ConditionOperator.Equal, name);
EntityCollection qu = (EntityCollection)service.RetrieveMultiple(q);
if (qu != null && qu.Entities.Count > 0)
{
Entity ent = null;
for (int i = 0; i < qu.Entities.Count; i++)
{
ent = qu.Entities[i];
add = ent.Attributes["new_companyaddress_test"].ToString();
city = ent.Attributes["address2_city"].ToString();
}
}
string a = receivableEntity.Attributes["new_generateinvoice"].ToString();
//string projectType = receivableEntity.Attributes["new_projecttype"].ToString();
if (receivableEntity.Attributes.Contains("new_generateinvoice") && a == "True")
{
if (receivableEntity != null)
{
if (receivableEntity.Attributes.Contains("new_nameoftheconsultant"))
{
consultant_name = (string)receivableEntity.Attributes["new_nameoftheconsultant"];
}
if (receivableEntity.Attributes.Contains("new_taxesapplicable"))
{
tax_per = (string)receivableEntity.Attributes["new_taxesapplicable"];
}
if (receivableEntity.Attributes.Contains("new_dateofjoining"))
{
joining_date = (DateTime)receivableEntity.Attributes["new_dateofjoining"];
j_date = joining_date.ToShortDateString();
}
if (receivableEntity.Attributes.Contains("new_ctcoffered"))
{
//ctc_offered = (float)receivableEntity.Attributes["new_ctcoffered"];
//ctc_offered = Convert.ToInt32(receivableEntity.Attributes["new_ctcoffered"]);
ctc_offered = ((Money)receivableEntity.Attributes["new_ctcoffered"]).Value.ToString();
}
if (receivableEntity.Attributes.Contains("new_billamount"))
{
billable_amount = ((Money)receivableEntity.Attributes["new_billamount"]).Value.ToString();
}
if (receivableEntity.Attributes.Contains("new_taxamount"))
{
tax_amount = ((Money)receivableEntity.Attributes["new_taxamount"]).Value.ToString();
}
if (receivableEntity.Attributes.Contains("new_totalamount"))
{
total_amount = ((Money)receivableEntity.Attributes["new_totalamount"]).Value.ToString();
decimal rev = ((Money)receivableEntity.Attributes["new_totalamount"]).Value;
int intvalue = Convert.ToInt32(rev);
amttot=NumberToWords(intvalue);
}
if (receivableEntity.Attributes.Contains("new_invoicenumber"))
{
invoice_number = (string)receivableEntity.Attributes["new_invoicenumber"];
}
if (receivableEntity.Attributes.Contains("new_invoicedate"))
{
invoice_date = (DateTime)receivableEntity.Attributes["new_invoicedate"];
i_date = invoice_date.ToShortDateString();
}
//Create a format in HTML format how u wanted in ur word document
string docbody = "<html><body><h3><b><center> INVOICE</center></b></h3><p><h3><pre><span>To, " + name + "<br/> " + add + "<br/>" + city + "</br>" + " <br/> Invoice No:" + invoice_number + " Date: " + i_date + " </span></pre></h3><table border=2 style='width: 40000px; height: 200px;' > <tr style='border-width: 2px;' > <td colspan='5'><center> Recruitment Charges towards Placement of <b>" + projname + "</b></center></td> </tr> <tr style='width: 500px; height: 30px;'> <td style='width: 40px;'><center>SL.No</center></td> <td style='width: 200px;'><center>Name of the candidate</center></td> <td style='width: 200px;'><center>Date of Joining</center></td> <td style='width: 200px;'><center>CTC offered(Rs)</center></td> <td style='width: 200px;'><center>Billable Amount (in Rs.)</center></td> </tr> <tr style='width: 800px; height: 50px;'> <td style='width: 70px;' ><center>1</center></td> <td>" + consultant_name + "</td> <td>" + j_date + "</td> <td>" + ctc_offered + "</td> <td>" + billable_amount + "</td> </tr> <tr style='height: 20px;'> <td colspan='4'><span style='margin-left: 600px;'>Amount Rs.<span></td> <td>" + billable_amount + "</td> </tr> <tr style='height: 20px;'> <td colspan='4'><span style='margin-left: 600px;'>" + tax_per + "% Service Tax Rs.<span></td> <td>" + tax_amount + "</td> </tr> <tr style='height: 20px;'> <td colspan='4'><span style='margin-left: 600px;'>TOTAL AMOUNT Rs.<span></td> <td>" + total_amount + "</td> </tr> <tr style='height: 20px;'> <td colspan='5'><b>Rupees in words:</b> " + amttot + " only </td> </tr> </table> <pre></body></html>";
//Save ur document in Server
string f1 = @"C:\temp\" + consultant_name + ".docx";
using (MemoryStream generatedDocument = new MemoryStream())
{
using (WordprocessingDocument package = WordprocessingDocument.Create(generatedDocument, WordprocessingDocumentType.Document))
{
MainDocumentPart mainPart = package.MainDocumentPart;
if (mainPart == null)
{
mainPart = package.AddMainDocumentPart();
new Document(new Body()).Save(mainPart);
}
HtmlConverter converter = new HtmlConverter(mainPart);
Body body = mainPart.Document.Body;
var paragraphs = converter.Parse(docbody);
for (int y = 0; y < paragraphs.Count; y++)
{
body.Append(paragraphs[y]);
}
mainPart.Document.Save();
}
string w = entity.LogicalName;
File.WriteAllBytes(f1, generatedDocument.ToArray());
FileStream stream = File.OpenRead(f1);
byte[] byteData = new byte[stream.Length];
stream.Read(byteData, 0, byteData.Length);
stream.Close();
string encodedData = System.Convert.ToBase64String(byteData);
string filename="LISPLIN-"+name+"-240-"+i_date+".doc";
// Add the generated document as an attachment to that record
Entity annotation = new Entity("annotation");
annotation.Attributes["subject"] = "Invoice";
EntityReference noteref = new EntityReference();
noteref.LogicalName = entity.LogicalName;
noteref.Id = entity.Id;
annotation.Attributes.Add("objectid", noteref);
annotation.Attributes.Add("objecttypecode", w);
annotation.Attributes.Add("filename", filename);
annotation.Attributes.Add("mimetype", "application/ms-word");
annotation.Attributes.Add("documentbody", encodedData);
service.Create(annotation);
}
}
}
}
catch (FaultException<OrganizationServiceFault> ex)
{
throw new InvalidPluginExecutionException("An error occurred in the plug-in.", ex);
}
}
//function to convert number to words(32 - thirty two)
public static string NumberToWords(int number)
{
if (number == 0)
return "zero";
if (number < 0)
return "minus " + NumberToWords(Math.Abs(number));
string words = "";
if ((number / 1000000) > 0)
{
words += NumberToWords(number / 1000000) + " Million ";
number %= 1000000;
}
if ((number / 100000) > 0)
{
words += NumberToWords(number / 100000) + " Lakh ";
number %= 100000;
}
if ((number / 1000) > 0)
{
words += NumberToWords(number / 1000) + " Thousand ";
number %= 1000;
}
if ((number / 100) > 0)
{
words += NumberToWords(number / 100) + " Hundred ";
number %= 100;
}
if (number > 0)
{
if (words != "")
words += "and ";
var unitsMap = new[] { "Zero", "One", "Two", "Three", "Four", "Five", "Six", "Seven", "Eight", "Nine", "Ten", "Eleven", "Twelve", "Thirteen", "Fourteen", "Fifteen", "Sixteen", "Seventeen", "Eighteen", "Nineteen" };
var tensMap = new[] { "Zero", "Ten", "Twenty", "Thirty", "Forty", "Fifty", "Sixty", "Seventy", "Eighty", "Ninety" };
if (number < 20)
words += unitsMap[number];
else
{
words += tensMap[number / 10];
if ((number % 10) > 0)
words += "-" + unitsMap[number % 10];
}
}
return words;
}
}
}
7. Add the required references and register the plugin for Update Message using Plugin Registration Tool
8. Now Refresh and go to ur entity open Record and Click on ur Custom button
9.Now the payslip would be generated and added as an attachment to that record
HOME PAGE GRID:
To perform the same operation for selected record from HomePage Grid Steps to be followed:
1) Create a custom button in Home Page Grid using RibbonWorkBench
2)Create a workflow which would change the Two Option Button activate the process as "As on Demand" process
3)Follow the post showing you how to trigger your workflow without writing a single line of code!
http://www.develop1.net/public/post/No-Code-Workflow-Shortcut-Ribbon-Button.aspx
4)And check ur entity attachments with generated payslip attached