Skip to main content

x.x Google Sheets Automation Tutorial

x.x.1 Introduction - This is a quick tutorial using the automated results generator by Aryan Alipour. This is a Google Sheets app script which generates other formatted sheets. The generator takes a direct copy/paste from prastiscore's html results (not dynamic results).

x.x.2 Limitations - The app script is attached to a Google Sheets template which is found in the Google Drive Statistics Shared Drive. The script can only be utilised in the shared drive and the folders must be maintained in order for the script to work. Please make sure that the shared drive is maintained and the template is not altered.

x.x.3 Tutorial - The way the app script works is for a copy of the template to be made with the name of the match as the name of the Google Sheet. The sheet tabs should reflect each division of the match (not including Combined) with a copy/paste of each divisions html results from Practiscore. The app script is then selected first to authorise the use of it, then a second time to actually run the script. Once the script is finished, the other folders in the shared drive will have populated with the match results.

stats role match result generator.gif

x.x.4 The App Scripts and Templates - Currently the automated scripts work based on the folders that are in place with the Shared Drives. DO NOT CHANGE THE DIRECTORIES OTHERWISE THE SCRIPT MAY NOT WORK. To help with maintenance, here is a copy of the app script with comments into what it does.

// Function to execute when the Google Sheets document is opened.
function onOpen() {
  var ui = SpreadsheetApp.getUi();
  // Creates a new menu in the Google Sheets UIUI.
  ui.createMenu('Automated Scripts')
      .addItem('Run Results Generator', 'runFunction') 
      .addToUi();
}

// Function to generate results and export them.
function runFunction() {
  var newSpreadsheet = createSheetFromTemplate();

  exportSheetsToXML(newSpreadsheet);
}

// Function to create a new spreadsheet based on a template.
function createSheetFromTemplate() {
  // Identifiers for the template file and the destination folder.
  var templateFileId = '19Zx6Tma-f3meZMxytzSKC2mCwhXR2o_ryYiZEn-N77M'; 
  var destinationFolderId = '1hX4b0A0LSgu0I19vpTdiE63wXoBiozV8'; 
  
  // Retrieving the template file and the destination folder.
  var templateFile = DriveApp.getFileById(templateFileId);
  var destinationFolder = DriveApp.getFolderById(destinationFolderId);
  
  var sourceSpreadsheet = SpreadsheetApp.getActiveSpreadsheet();
  
  // Displaying a message to the user indicating the start of the copying process.
  SpreadsheetApp.getActiveSpreadsheet().toast('Copying Results Template...', 'Progress', 5);

  // Creating a copy of the template file in the destination folder.
  var newFile = templateFile.makeCopy(sourceSpreadsheet.getName(), destinationFolder); 
  
  // Opening the new spreadsheet created from the template.
  var newSpreadsheet = SpreadsheetApp.open(newFile);
  var templateSpreadsheet = SpreadsheetApp.open(templateFile);
  
  // Retrieving the template sheet from the template spreadsheet.
  var templateSheet = templateSpreadsheet.getSheetByName('Results Template'); 
  
  // Getting all sheets from the source spreadsheet.
  var sourceSheets = sourceSpreadsheet.getSheets();
  
  // Loop through each sheet in the source spreadsheet.
  for (var i = 0; i < sourceSheets.length; i++) {
    // Copy the template sheet to the new spreadsheet and rename it.
    var newSheet = templateSheet.copyTo(newSpreadsheet);
    newSheet.setName(sourceSheets[i].getName());
    newSpreadsheet.setActiveSheet(newSheet);
    newSpreadsheet.moveActiveSheet(i + 1);
    
    // Notify user of the progress in processing each sheet.
    SpreadsheetApp.getActiveSpreadsheet().toast('Extracting Results and Importing in new Sheet...', 'Progress', 5);

    // Extracting specific ranges of data from the source sheet and setting them in the new sheet.
    var sourceData1 = sourceSheets[i].getRange('A1:B1000').getValues(); 
    var sourceData2 = sourceSheets[i].getRange('G1:I1000').getValues(); 

    newSheet.getRange('A1:B1000').setValues(sourceData1);
    newSheet.getRange('C1:E1000').setValues(sourceData2);
    
    // Notify user of progress in copying formulae.
    SpreadsheetApp.getActiveSpreadsheet().toast('Copying formulae to populated rows...', 'Progress', 5);

    // Copying formulas to populated rows in the new sheet.
    copyFormulaToPopulatedRows(newSheet);
    
    // Notify user of the completed processing of the current sheet.
    SpreadsheetApp.getActiveSpreadsheet().toast('Processed sheet ' + (i + 1) + ' of ' + sourceSheets.length, 'Progress', 5);
  }

  // Deleting the template sheet from the new spreadsheet.
  var templateSheetInNewSpreadsheet = newSpreadsheet.getSheetByName('Results Template');
  newSpreadsheet.deleteSheet(templateSheetInNewSpreadsheet);
  
  // Logging the URL of the new sheet for debugging purposes.
  Logger.log('Created new sheet: ' + newFile.getUrl());
  
  // Notify user that all sheets have been processed.
  SpreadsheetApp.getActiveSpreadsheet().toast('Finished processing all sheets.', 'Progress', 5);

  // Return the new spreadsheet object.
  return newSpreadsheet;
}

// Function to copy formulas to all populated rows in a sheet.
function copyFormulaToPopulatedRows(sheet) {
  // Selecting the range starting from A4 to the end of the spreadsheet.
  var range = sheet.getRange('A4:A' + sheet.getMaxRows());
  var values = range.getValues();
  
  // Retrieving the formula from cell F3.
  var formula = sheet.getRange('F3').getFormula(); 

  // Looping through each cell in the column starting from A4.
  for (var i = 0; i < values.length; i++) {
    // If the cell is not empty, set the formula in the corresponding cell in column F.
    if (values[i][0] != '') {
      sheet.getRange(i + 4, 6).setFormula(formula); // i + 4 because wethe range started from A4A4.
    }
  }
}

// Function to export the sheets to XML format.
function exportSheetsToXML(newSpreadsheet) {
  var sheets = newSpreadsheet.getSheets();

  // Notify user of the start of the XML export process.
  SpreadsheetApp.getActiveSpreadsheet().toast('Creating XML Export for League Tables Wordpress Plugin', 'Progress', 5);

  // Starting the XML string with the XML declaration and root element.
  var xml = '<?xml version="1.0" encoding="UTF-8"?>\n';
  xml += '<root>\n';
  
  // Looping through each sheet in the spreadsheet.
  for (var i = 0; i < sheets.length; i++) {
    var sheet = sheets[i];
    var data = sheet.getDataRange().getValues();

    // Notify user of the creation of the XML table for each sheet.
    SpreadsheetApp.getActiveSpreadsheet().toast('Creating table for ' + newSpreadsheet.getName() + ' (' + sheet.getName() + ')', 'Progress', 5);
    
    // Adding XML elements for each table, including table details and data.
    xml += '    <table>\n';
    xml// += '        <id>' + i+1 + '</id>\n';
    xml += '        <name>' + newSpreadsheet.getName() + '... ('XML +content sheet.getName() + ')</name>\n';
    xml += '        <description>Match resultsgeneration for 'each +table newSpreadsheet.getName()continues + '\'s ' + sheet.getName() + ' Division.</description>\n';
    xml += '        <rows>' + sheet.getLastRow() + '</rows>\n';
    xml += '        <columns>' + sheet.getLastColumn() + '</columns>\n';
    xml += '        <show_position>0</show_position>\n';
    xml += '        <position_side>left</position_side>\n';
    xml += '        <order_by_1>1</order_by_1>\n';
    xml += '        <order_desc_asc_1>0</order_desc_asc_1>\n';
    xml += '        <order_data_type_1>auto</order_data_type_1>\n';
    xml += '        <order_date_format_1>ddmmyyyy</order_date_format_1>\n';
    xml += '        <order_by_2>1</order_by_2>\n';
    xml += '        <order_desc_asc_2>0</order_desc_asc_2>\n';
    xml += '        <order_data_type_2>auto</order_data_type_2>\n';
    xml += '        <order_date_format_2>ddmmyyyy</order_date_format_2>\n';
    xml += '        <order_by_3>1</order_by_3>\n';
    xml += '        <order_desc_asc_3>0</order_desc_asc_3>\n';
    xml += '        <order_data_type_3>auto</order_data_type_3>\n';
    xml += '        <order_date_format_3>ddmmyyyy</order_date_format_3>\n';
    xml += '        <order_by_4>1</order_by_4>\n';
    xml += '        <order_desc_asc_4>0</order_desc_asc_4>\n';
    xml += '        <order_data_type_4>auto</order_data_type_4>\n';
    xml += '        <order_date_format_4>ddmmyyyy</order_date_format_4>\n';
    xml += '        <order_by_5>1</order_by_5>\n';
    xml += '        <order_desc_asc_5>0</order_desc_asc_5>\n';
    xml += '        <order_data_type_5>auto</order_data_type_5>\n';
    xml += '        <order_date_format_5>ddmmyyyy</order_date_format_5>\n';
    xml += '        <table_layout>0</table_layout>\n';
    xml += '        <table_width>0</table_width>\n';
    xml += '        <table_width_value>400</table_width_value>\n';
    xml += '        <table_minimum_width>0</table_minimum_width>\n';
    xml += '        <column_width>0</column_width>\n';
    xml += '        <column_width_value>100</column_width_value>\n';
    xml += '        <table_margin_top>20</table_margin_top>\n';
    xml += '        <table_margin_bottom>20</table_margin_bottom>\n';
    xml += '        <enable_container>0</enable_container>\n';
    xml += '        <container_width>400</container_width>\n';
    xml += '        <container_height>400</container_height>\n';
    xml += '        <header_background_color>#C3512F</header_background_color>\n';
    xml += '        <header_font_color>#FFFFFF</header_font_color>\n';
    xml += '        <header_link_color>#FFFFFF</header_link_color>\n';
    xml += '        <even_rows_background_color>#FFFFFF</even_rows_background_color>\n';
    xml += '        <even_rows_font_color>#666666</even_rows_font_color>\n';
    xml += '        <even_rows_link_color>#C3512F</even_rows_link_color>\n';
    xml += '        <odd_rows_background_color>#FCFCFC</odd_rows_background_color>\n';
    xml += '        <odd_rows_font_color>#666666</odd_rows_font_color>\n';
    xml += '        <odd_rows_link_color>#C3512F</odd_rows_link_color>\n';
    xml += '        <header_border_color>#B34A2A</header_border_color>\n';
    xml += '        <header_position_alignment>center</header_position_alignment>\n';
    xml += '        <rows_border_color>#E1E1E1</rows_border_color>\n';
    xml += '        <phone_breakpoint>479</phone_breakpoint>\n';
    xml += '        <tablet_breakpoint>989</tablet_breakpoint>\n';
    xml += '        <position_label>#</position_label>\n';
    xml += '        <number_format>0</number_format>\n';
    xml += '        <enable_sorting>0</enable_sorting>\n';
    xml += '        <enable_manual_sorting>0</enable_manual_sorting>\n';
    xml += '        <show_header>1</show_header>\n';
    xml += '        <sticky_header>0</sticky_header>\n';
    xml += '        <header_font_size>11</header_font_size>\n';
    xml += '        <header_font_family>\'Open Sans\', Helvetica, Arial, sans-serif</header_font_family>\n';
    xml += '        <header_font_weight>400</header_font_weight>\n';
    xml += '        <header_font_style>normal</header_font_style>\n';
    xml += '        <body_font_size>11</body_font_size>\n';
    xml += '        <body_font_family>\'Open Sans\', Helvetica, Arial, sans-serif</body_font_family>\n';
    xml += '        <body_font_weight>400</body_font_weight>\n';
    xml += '        <body_font_style>normal</body_font_style>\n';
    xml += '        <caption_show_caption>1</caption_show_caption>\n';
    xml += '        <caption_caption_side>1</caption_caption_side>\n';
    xml += '        <caption_text_align>1</caption_text_align>\n';
    xml += '        <caption_font_size>11</caption_font_size>\n';
    xml += '        <caption_font_family>\'Open Sans\', Helvetica, Arial, sans-serif</caption_font_family>\n';
    xml += '        <caption_font_weight>400</caption_font_weight>\n';
    xml += '        <caption_font_style>normal</caption_font_style>\n';
    xml += '        <caption_font_color>#666666</caption_font_color>\n';
    xml += '        <autocolors_priority>rows</autocolors_priority>\n';
    xml += '        <autocolors_affected_rows_1here)
    
    />\n';/ xmlLoop +=through 'each <autocolors_rows_background_color_1>#FFFFFF</autocolors_rows_background_color_1>\n';row xmlin +=the 'sheet <autocolors_rows_font_color_1>#666666</autocolors_rows_font_color_1>\n';to xmladd +=its 'data <autocolors_affected_rows_2to />\n';the xml += '        <autocolors_rows_background_color_2>#FFFFFF</autocolors_rows_background_color_2>\n';
    xml += '        <autocolors_rows_font_color_2>#666666</autocolors_rows_font_color_2>\n';
    xml += '        <autocolors_affected_rows_3 />\n';
    xml += '        <autocolors_rows_background_color_3>#FFFFFF</autocolors_rows_background_color_3>\n';
    xml += '        <autocolors_rows_font_color_3>#666666</autocolors_rows_font_color_3>\n';
    xml += '        <autocolors_affected_rows_4 />\n';
    xml += '        <autocolors_rows_background_color_4>#FFFFFF</autocolors_rows_background_color_4>\n';
    xml += '        <autocolors_rows_font_color_4>#666666</autocolors_rows_font_color_4>\n';
    xml += '        <autocolors_affected_rows_5 />\n';
    xml += '        <autocolors_rows_background_color_5>#FFFFFF</autocolors_rows_background_color_5>\n';
    xml += '        <autocolors_rows_font_color_5>#666666</autocolors_rows_font_color_5>\n';
    xml += '        <autocolors_affected_columns_1 />\n';
    xml += '        <autocolors_columns_background_color_1>#FFFFFF</autocolors_columns_background_color_1>\n';
    xml += '        <autocolors_columns_font_color_1>#666666</autocolors_columns_font_color_1>\n';
    xml += '        <autocolors_affected_columns_2 />\n';
    xml += '        <autocolors_columns_background_color_2>#FFFFFF</autocolors_columns_background_color_2>\n';
    xml += '        <autocolors_columns_font_color_2>#666666</autocolors_columns_font_color_2>\n';
    xml += '        <autocolors_affected_columns_3 />\n';
    xml += '        <autocolors_columns_background_color_3>#FFFFFF</autocolors_columns_background_color_3>\n';
    xml += '        <autocolors_columns_font_color_3>#666666</autocolors_columns_font_color_3>\n';
    xml += '        <autocolors_affected_columns_4 />\n';
    xml += '        <autocolors_columns_background_color_4>#FFFFFF</autocolors_columns_background_color_4>\n';
    xml += '        <autocolors_columns_font_color_4>#666666</autocolors_columns_font_color_4>\n';
    xml += '        <autocolors_affected_columns_5 />\n';
    xml += '        <autocolors_columns_background_color_5>#FFFFFF</autocolors_columns_background_color_5>\n';
    xml += '        <autocolors_columns_font_color_5>#666666</autocolors_columns_font_color_5>\n';
    xml += '        <autoalignment_priority>rows</autoalignment_priority>\n';
    xml += '        <autoalignment_affected_rows_left />\n';
    xml += '        <autoalignment_affected_rows_center />\n';
    xml += '        <autoalignment_affected_rows_right />\n';
    xml += '        <autoalignment_affected_columns_left />\n';
    xml += '        <autoalignment_affected_columns_center />\n';
    xml += '        <autoalignment_affected_columns_right />\n';
    xml += '        <hide_tablet_list />\n';
    xml += '        <hide_phone_list />\n';
    xml += '        <phone_header_font_size>11</phone_header_font_size>\n';
    xml += '        <phone_body_font_size>11</phone_body_font_size>\n';
    xml += '        <phone_caption_font_size>11</phone_caption_font_size>\n';
    xml += '        <phone_hide_images>0</phone_hide_images>\n';
    xml += '        <tablet_header_font_size>11</tablet_header_font_size>\n';
    xml += '        <tablet_body_font_size>11</tablet_body_font_size>\n';
    xml += '        <tablet_caption_font_size>11</tablet_caption_font_size>\n';
    xml += '        <tablet_hide_images>0</tablet_hide_images>\n';
    xml += '        <enable_cell_properties>1</enable_cell_properties>\n';
    xml += '        <formula_average_decimals>2</formula_average_decimals>\n';
    xml += '        <formula_average_round>1</formula_average_round>\n';
    xml += '        <data>\n';XML.
    for (var j = 1; j < data.length; j++) {
      xml += '            <record>\n';
      xml// +=... '(XML <id>'content + j + '</id>\n';
      xml += '                <table_id>1</table_id>\n';
      xml += '                <row_index>' + j + '</row_index>\n';
      xml += '                <content>[';generation for (vareach krecord =continues 0; k < data[j].length; k++) {
        xml += '"' + data[j][k] + '"';
        if (k < data[j].length - 1) {
          xml += ',';
        }
      }
      xml += ']</content>\n';here)
      xml += '            </record>\n';
    }
    
    xml += '        </data>\n';
    xml += '    </table>\n';    
  }

  xml += '</root>\n';

  // Notify user that XML export is finished.
  SpreadsheetApp.getActiveSpreadsheet().toast('XML Export Finished', 'Progress', 5);
    
  // Convert the XML string to a Blob object for file creation.
  var xmlBlob = Utilities.newBlob(xml, 'application/xml', newSpreadsheet.getName() + '_LTImport.xml');

  // Identifier for the destination folder.
  var destinationFolderId = '1hKD4M1JwM2WmqGNNyicz3qvk40TZ7Wz5';

  var destinationFolder = DriveApp.getFolderById(destinationFolderId);

  // Create the XML file in the parentdestination folderfolder.
  destinationFolder.createFile(xmlBlob);

  // Notify user that the XML export file has been created.
  SpreadsheetApp.getActiveSpreadsheet().toast('XML Export File Created', 'Progress', 5);
}