ایجاد & فایل‌های PDF را از Google Sheets ارسال کنید

سطح کدنویسی : مبتدی
مدت زمان : ۱۵ دقیقه
نوع پروژه : اتوماسیون با منوی سفارشی

اهداف

  • بفهمید که راه حل چه کاری انجام می‌دهد.
  • درک کنید که سرویس‌های Apps Script در این راهکار چه کاری انجام می‌دهند.
  • اسکریپت را تنظیم کنید.
  • اسکریپت را اجرا کنید.

درباره این راهکار

به طور خودکار فایل‌های PDF را با اطلاعات موجود در صفحات گسترده Google Sheets ایجاد کنید. پس از تولید PDFها، می‌توانید آنها را مستقیماً از Sheets ایمیل کنید. این راهکار بر ایجاد فاکتورهای سفارشی تمرکز دارد، اما می‌توانید قالب و اسکریپت را متناسب با نیازهای خود به‌روزرسانی کنید.

تصویر نمونه فاکتور

چگونه کار می‌کند؟

این اسکریپت از برگه الگوی فاکتور به عنوان الگو برای تولید PDF استفاده می‌کند. اطلاعات از برگه‌های دیگر برای پر کردن سلول‌های خاص در الگو گرفته می‌شود. برای ارسال PDFها از طریق ایمیل، اسکریپت در برگه فاکتورها تکرار می‌شود تا لینک PDF و آدرس ایمیل مرتبط را دریافت کند. اسکریپت یک موضوع و بدنه ایمیل عمومی ایجاد می‌کند و قبل از ارسال، PDF را پیوست می‌کند.

سرویس‌های اسکریپت برنامه‌ها

این راهکار از سرویس‌های زیر استفاده می‌کند:

  • سرویس صفحه گسترده - تمام اطلاعات لازم برای تولید فایل‌های PDF فاکتور و ایجاد ایمیل را فراهم می‌کند. وقتی کاربر روی «بازنشانی الگو» در منوی سفارشی کلیک می‌کند، داده‌ها را از الگو پاک می‌کند.
  • سرویس Utilities - اسکریپت را با متد sleep() متوقف می‌کند و همزمان با پیمایش روی هر مشتری، از اضافه شدن اطلاعات صحیح به هر فاکتور اطمینان حاصل می‌کند.
  • سرویس URL Fetch - برگه الگوی فاکتور را به PDF تبدیل می‌کند.
  • سرویس اسکریپت - سرویس URL Fetch را برای دسترسی به صفحه گسترده مجاز می‌کند.
  • سرویس درایو - پوشه‌ای برای فایل‌های PDF خروجی ایجاد می‌کند. فایل‌های PDF را به ایمیل‌ها پیوست می‌کند.
  • سرویس جیمیل - ایمیل‌ها را می‌سازد و ارسال می‌کند.

پیش‌نیازها

برای استفاده از این نمونه، به پیش‌نیازهای زیر نیاز دارید:

  • یک حساب گوگل (حساب‌های کاربری گوگل ورک‌اسپیس ممکن است نیاز به تأیید مدیر داشته باشند).
  • یک مرورگر وب با دسترسی به اینترنت.

اسکریپت را تنظیم کنید

  1. برای کپی کردن صفحه گسترده Generate and send PDFs from Google Sheets روی دکمه زیر کلیک کنید. پروژه Apps Script برای این راهکار به صفحه گسترده پیوست شده است.
    یک کپی تهیه کنید

  2. روی افزونه‌ها > اسکریپت برنامه‌ها کلیک کنید.

  3. در فایل Code.gs ، متغیرهای زیر را به‌روزرسانی کنید:

    1. مقدار EMAIL_OVERRIDE را روی true تنظیم کنید.
    2. EMAIL_ADDRESS_OVERRIDE را روی آدرس ایمیل خود تنظیم کنید.
  4. روی ذخیره کلیک کنید آیکون ذخیره .

اسکریپت را اجرا کنید

  1. به صفحه گسترده برگردید و روی ایجاد و ارسال فایل‌های PDF > پردازش فاکتورها کلیک کنید.
  2. وقتی از شما خواسته شد، اسکریپت را تأیید کنید. اگر صفحه رضایت OAuth هشدار « این برنامه تأیید نشده است» را نشان می‌دهد، با انتخاب Advanced > Go to {Project Name} (unsafe) ادامه دهید.

  3. روی «ایجاد و ارسال فایل‌های PDF» > «پردازش مجدد فاکتورها» کلیک کنید.

  4. برای مشاهده فایل‌های PDF، به برگه فاکتورها بروید و روی لینک‌های موجود در ستون لینک فاکتور کلیک کنید.

  5. روی ایجاد و ارسال فایل‌های PDF > ارسال ایمیل کلیک کنید.

  6. ایمیل خود را بررسی کنید تا ایمیل‌ها و فایل‌های PDF پیوست شده را بررسی کنید. از آنجایی که در بخش قبلی، EMAIL_OVERRIDE روی true تنظیم کرده‌اید، اسکریپت تمام ایمیل‌ها را به آدرس ایمیلی که برای EMAIL_ADDRESS_OVERRIDE مشخص کرده‌اید ارسال می‌کند. اگر EMAIL_OVERRIDE روی false تنظیم کنید، اسکریپت ایمیل‌ها را به آدرس‌های ایمیل ذکر شده در برگه مشتریان ارسال می‌کند.

  7. (اختیاری) برای پاک کردن داده‌ها از برگه الگوی فاکتور ، روی «ایجاد و ارسال فایل‌های PDF» > «بازنشانی الگو» کلیک کنید.

کد را مرور کنید

برای بررسی کد Apps Script برای این راهکار، روی مشاهده کد منبع در زیر کلیک کنید:

مشاهده کد منبع

کد.gs

راهکارها/اتوماسیون‌ها/generate-pdfs/Code.js
// To learn how to use this script, refer to the documentation:
// https://developers.google.com/apps-script/samples/automations/generate-pdfs

/*
Copyright 2022 Google LLC

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    https://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

// TODO: To test this solution, set EMAIL_OVERRIDE to true and set EMAIL_ADDRESS_OVERRIDE to your email address.
const EMAIL_OVERRIDE = false;
const EMAIL_ADDRESS_OVERRIDE = "test@example.com";

// Application constants
const APP_TITLE = "Generate and send PDFs";
const OUTPUT_FOLDER_NAME = "Customer PDFs";
const DUE_DATE_NUM_DAYS = 15;

// Sheet name constants. Update if you change the names of the sheets.
const CUSTOMERS_SHEET_NAME = "Customers";
const PRODUCTS_SHEET_NAME = "Products";
const TRANSACTIONS_SHEET_NAME = "Transactions";
const INVOICES_SHEET_NAME = "Invoices";
const INVOICE_TEMPLATE_SHEET_NAME = "Invoice Template";

// Email constants
const EMAIL_SUBJECT = "Invoice Notification";
const EMAIL_BODY = "Hello!\rPlease see the attached PDF document.";

/**
 * Iterates through the worksheet data populating the template sheet with
 * customer data, then saves each instance as a PDF document.
 *
 * Called by user via custom menu item.
 */
function processDocuments() {
  const ss = SpreadsheetApp.getActiveSpreadsheet();
  const customersSheet = ss.getSheetByName(CUSTOMERS_SHEET_NAME);
  const productsSheet = ss.getSheetByName(PRODUCTS_SHEET_NAME);
  const transactionsSheet = ss.getSheetByName(TRANSACTIONS_SHEET_NAME);
  const invoicesSheet = ss.getSheetByName(INVOICES_SHEET_NAME);
  const invoiceTemplateSheet = ss.getSheetByName(INVOICE_TEMPLATE_SHEET_NAME);

  // Gets data from the storage sheets as objects.
  const customers = dataRangeToObject(customersSheet);
  const products = dataRangeToObject(productsSheet);
  const transactions = dataRangeToObject(transactionsSheet);

  ss.toast("Creating Invoices", APP_TITLE, 1);
  const invoices = [];

  // Iterates for each customer calling createInvoiceForCustomer routine.
  for (const customer of customers) {
    ss.toast(`Creating Invoice for ${customer.customer_name}`, APP_TITLE, 1);
    const invoice = createInvoiceForCustomer(
      customer,
      products,
      transactions,
      invoiceTemplateSheet,
      ss.getId(),
    );
    invoices.push(invoice);
  }
  // Writes invoices data to the sheet.
  invoicesSheet
    .getRange(2, 1, invoices.length, invoices[0].length)
    .setValues(invoices);
}

/**
 * Processes each customer instance with passed in data parameters.
 *
 * @param {object} customer - Object for the customer
 * @param {object} products - Object for all the products
 * @param {object} transactions - Object for all the transactions
 * @param {object} invoiceTemplateSheet - Object for the invoice template sheet
 * @param {string} ssId - Google Sheet ID
 * Return {array} of instance customer invoice data
 */
function createInvoiceForCustomer(
  customer,
  products,
  transactions,
  templateSheet,
  ssId,
) {
  const customerTransactions = transactions.filter(
    (transaction) => transaction.customer_name === customer.customer_name,
  );

  // Clears existing data from the template.
  clearTemplateSheet();

  const lineItems = [];
  let totalAmount = 0;
  for (const lineItem of customerTransactions) {
    const lineItemProduct = products.filter(
      (product) => product.sku_name === lineItem.sku,
    )[0];
    const qty = Number.parseInt(lineItem.licenses);
    const price = Number.parseFloat(lineItemProduct.price).toFixed(2);
    const amount = Number.parseFloat(qty * price).toFixed(2);
    lineItems.push([
      lineItemProduct.sku_name,
      lineItemProduct.sku_description,
      "",
      qty,
      price,
      amount,
    ]);
    totalAmount += Number.parseFloat(amount);
  }

  // Generates a random invoice number. You can replace with your own document ID method.
  const invoiceNumber = Math.floor(100000 + Math.random() * 900000);

  // Calulates dates.
  const todaysDate = new Date().toDateString();
  const dueDate = new Date(
    Date.now() + 1000 * 60 * 60 * 24 * DUE_DATE_NUM_DAYS,
  ).toDateString();

  // Sets values in the template.
  templateSheet.getRange("B10").setValue(customer.customer_name);
  templateSheet.getRange("B11").setValue(customer.address);
  templateSheet.getRange("F10").setValue(invoiceNumber);
  templateSheet.getRange("F12").setValue(todaysDate);
  templateSheet.getRange("F14").setValue(dueDate);
  templateSheet.getRange(18, 2, lineItems.length, 6).setValues(lineItems);

  // Cleans up and creates PDF.
  SpreadsheetApp.flush();
  Utilities.sleep(500); // Using to offset any potential latency in creating .pdf
  const pdf = createPDF(
    ssId,
    templateSheet,
    `Invoice#${invoiceNumber}-${customer.customer_name}`,
  );
  return [
    invoiceNumber,
    todaysDate,
    customer.customer_name,
    customer.email,
    "",
    totalAmount,
    dueDate,
    pdf.getUrl(),
    "No",
  ];
}

/**
 * Resets the template sheet by clearing out customer data.
 * You use this to prepare for the next iteration or to view blank
 * the template for design.
 *
 * Called by createInvoiceForCustomer() or by the user via custom menu item.
 */
function clearTemplateSheet() {
  const ss = SpreadsheetApp.getActiveSpreadsheet();
  const templateSheet = ss.getSheetByName(INVOICE_TEMPLATE_SHEET_NAME);
  // Clears existing data from the template.
  const rngClear = templateSheet
    .getRangeList(["B10:B11", "F10", "F12", "F14"])
    .getRanges();
  for (const cell of rngClear) {
    cell.clearContent();
  }
  // This sample only accounts for six rows of data 'B18:G24'. You can extend or make dynamic as necessary.
  templateSheet.getRange(18, 2, 7, 6).clearContent();
}

/**
 * Creates a PDF for the customer given sheet.
 * @param {string} ssId - Id of the Google Spreadsheet
 * @param {object} sheet - Sheet to be converted as PDF
 * @param {string} pdfName - File name of the PDF being created
 * @return {file object} PDF file as a blob
 */
function createPDF(ssId, sheet, pdfName) {
  const fr = 0;
  const fc = 0;
  const lc = 9;
  const lr = 27;
  const url = `https://docs.google.com/spreadsheets/d/${ssId}/export?format=pdf&size=7&fzr=true&portrait=true&fitw=true&gridlines=false&printtitle=false&top_margin=0.5&bottom_margin=0.25&left_margin=0.5&right_margin=0.5&sheetnames=false&pagenum=UNDEFINED&attachment=true&gid=${sheet.getSheetId()}&r1=${fr}&c1=${fc}&r2=${lr}&c2=${lc}`;

  const params = {
    method: "GET",
    headers: { authorization: `Bearer ${ScriptApp.getOAuthToken()}` },
  };
  const blob = UrlFetchApp.fetch(url, params)
    .getBlob()
    .setName(`${pdfName}.pdf`);

  // Gets the folder in Drive where the PDFs are stored.
  const folder = getFolderByName_(OUTPUT_FOLDER_NAME);

  const pdfFile = folder.createFile(blob);
  return pdfFile;
}

/**
 * Sends emails with PDF as an attachment.
 * Checks/Sets 'Email Sent' column to 'Yes' to avoid resending.
 *
 * Called by user via custom menu item.
 */
function sendEmails() {
  const ss = SpreadsheetApp.getActiveSpreadsheet();
  const invoicesSheet = ss.getSheetByName(INVOICES_SHEET_NAME);
  const invoicesData = invoicesSheet
    .getRange(1, 1, invoicesSheet.getLastRow(), invoicesSheet.getLastColumn())
    .getValues();
  const keysI = invoicesData.splice(0, 1)[0];
  const invoices = getObjects(invoicesData, createObjectKeys(keysI));
  ss.toast("Emailing Invoices", APP_TITLE, 1);
  invoices.forEach((invoice, index) => {
    if (invoice.email_sent !== "Yes") {
      ss.toast(`Emailing Invoice for ${invoice.customer}`, APP_TITLE, 1);

      const fileId = invoice.invoice_link.match(/[-\w]{25,}(?!.*[-\w]{25,})/);
      const attachment = DriveApp.getFileById(fileId);

      let recipient = invoice.email;
      if (EMAIL_OVERRIDE) {
        recipient = EMAIL_ADDRESS_OVERRIDE;
      }

      GmailApp.sendEmail(recipient, EMAIL_SUBJECT, EMAIL_BODY, {
        attachments: [attachment.getAs(MimeType.PDF)],
        name: APP_TITLE,
      });
      invoicesSheet.getRange(index + 2, 9).setValue("Yes");
    }
  });
}

/**
 * Helper function that turns sheet data range into an object.
 *
 * @param {SpreadsheetApp.Sheet} sheet - Sheet to process
 * Return {object} of a sheet's datarange as an object
 */
function dataRangeToObject(sheet) {
  const dataRange = sheet
    .getRange(1, 1, sheet.getLastRow(), sheet.getLastColumn())
    .getValues();
  const keys = dataRange.splice(0, 1)[0];
  return getObjects(dataRange, createObjectKeys(keys));
}

/**
 * Utility function for mapping sheet data to objects.
 */
function getObjects(data, keys) {
  const objects = [];
  for (let i = 0; i < data.length; ++i) {
    const object = {};
    let hasData = false;
    for (let j = 0; j < data[i].length; ++j) {
      const cellData = data[i][j];
      if (isCellEmpty(cellData)) {
        continue;
      }
      object[keys[j]] = cellData;
      hasData = true;
    }
    if (hasData) {
      objects.push(object);
    }
  }
  return objects;
}
// Creates object keys for column headers.
function createObjectKeys(keys) {
  return keys.map((key) => key.replace(/\W+/g, "_").toLowerCase());
}
// Returns true if the cell where cellData was read from is empty.
function isCellEmpty(cellData) {
  return typeof cellData === "string" && cellData === "";
}

راهکارها/اتوماسیون‌ها/generate-pdfs/Menu.js
/**
 * Copyright 2022 Google LLC
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * @OnlyCurrentDoc
 *
 * The above comment specifies that this automation will only
 * attempt to read or modify the spreadsheet this script is bound to.
 * The authorization request message presented to users reflects the
 * limited scope.
 */

/**
 * Creates a custom menu in the Google Sheets UI when the document is opened.
 *
 * @param {object} e The event parameter for a simple onOpen trigger.
 */
function onOpen(e) {
  const menu = SpreadsheetApp.getUi().createMenu(APP_TITLE);
  menu
    .addItem("Process invoices", "processDocuments")
    .addItem("Send emails", "sendEmails")
    .addSeparator()
    .addItem("Reset template", "clearTemplateSheet")
    .addToUi();
}

Utilities.gs

راهکارها/اتوماسیون‌ها/generate-pdfs/Utilities.js
/**
 * Copyright 2022 Google LLC
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * Returns a Google Drive folder in the same location
 * in Drive where the spreadsheet is located. First, it checks if the folder
 * already exists and returns that folder. If the folder doesn't already
 * exist, the script creates a new one. The folder's name is set by the
 * "OUTPUT_FOLDER_NAME" variable from the Code.gs file.
 *
 * @param {string} folderName - Name of the Drive folder.
 * @return {object} Google Drive Folder
 */
function getFolderByName_(folderName) {
  // Gets the Drive Folder of where the current spreadsheet is located.
  const ssId = SpreadsheetApp.getActiveSpreadsheet().getId();
  const parentFolder = DriveApp.getFileById(ssId).getParents().next();

  // Iterates the subfolders to check if the PDF folder already exists.
  const subFolders = parentFolder.getFolders();
  while (subFolders.hasNext()) {
    const folder = subFolders.next();

    // Returns the existing folder if found.
    if (folder.getName() === folderName) {
      return folder;
    }
  }
  // Creates a new folder if one does not already exist.
  return parentFolder
    .createFolder(folderName)
    .setDescription(
      `Created by ${APP_TITLE} application to store PDF output files`,
    );
}

/**
 * Test function to run getFolderByName_.
 * @prints a Google Drive FolderId.
 */
function test_getFolderByName() {
  // Gets the PDF folder in Drive.
  const folder = getFolderByName_(OUTPUT_FOLDER_NAME);

  console.log(
    `Name: ${folder.getName()}\rID: ${folder.getId()}\rDescription: ${folder.getDescription()}`,
  );
  // To automatically delete test folder, uncomment the following code:
  // folder.setTrashed(true);
}

مشارکت‌کنندگان

این نمونه توسط گوگل و با کمک متخصصان توسعه‌دهنده گوگل نگهداری می‌شود.

مراحل بعدی