Hey, It Works! Logo
Hey, It Works!

Tech Blog by Daniel Rosehill

Google Drive Accounting Doc Router For Israeli Freelancers (Apps Script)

Google Drive Accounting Doc Router For Israeli Freelancers (Apps Script)

![Claude](https://camo.githubusercontent.com/3e6c1e8e4901b7a3c95b73396ac2d66b0279996788419fdf81f74ea08cd47188/68747470733a2f2f696d672e736869656c64732e696f2f62616467652f436c617564652d332e352d626c756576696f6c6574 align="left")

![Cline](https://camo.githubusercontent.com/26824002b06d03e3a14e2439ce0700de18c8c64940a3d92ce2fd60125af56c09/68747470733a2f2f696d672e736869656c64732e696f2f62616467652f436c696e652d76312e302d626c7565 align="left")

![Google Drive](https://camo.githubusercontent.com/ee7f96d3a38beba9f8244337338b8ee5d14857163445d3ab230e056149ff6d2c/68747470733a2f2f696d672e736869656c64732e696f2f62616467652f476f6f676c6525323044726976652d3432383546343f7374796c653d666c6174266c6f676f3d676f6f676c656472697665266c6f676f436f6c6f723d7768697465 align="left")

![Google Apps Script](https://camo.githubusercontent.com/8a3989a98813a4d300e3860ebd508138fd57fd4cd4760e30529427f762b92ac6/68747470733a2f2f696d672e736869656c64732e696f2f62616467652f476f6f676c65253230417070732532305363726970742d3432383546343f7374796c653d666c6174266c6f676f3d676f6f676c6561707073736372697074266c6f676f436f6c6f723d7768697465 align="left")

![AI Generated](https://camo.githubusercontent.com/d4e54c9582112affee4d578bc1aefd4769c06d74f74bdfc1a3bcb3af11831d42/68747470733a2f2f696d672e736869656c64732e696f2f62616467652f41492d47656e6572617465642d6c6967687467726579 align="left")

As a freelancer or business owner dealing with both Israeli and international clients, managing accounting documents can quickly become a complex task. That's why I developed a specialized Google Apps Script application that automates the organization of invoices, receipts, and expenses in Google Drive.

Github Project Link

The Challenge

Managing accounting documents across different tax jurisdictions presents unique challenges:

  • Separating Israeli and international documents for VAT purposes

  • Maintaining a consistent folder structure across years and months

  • Efficiently handling multiple document uploads

  • Ensuring proper naming conventions for easy retrieval

The Solution

I created a Google Apps Script application that provides an intuitive web interface for routing accounting documents to their appropriate folders in Google Drive. The application automatically:

  • Creates and maintains a structured folder hierarchy by year and month

  • Separates Israeli and non-Israeli documents

  • Handles batch uploads of up to 10 documents

  • Allows custom file renaming for better organization

Technical Implementation

The application is built using Google Apps Script, which provides seamless integration with Google Drive. It consists of two main components:

  1. Frontend (index.html)

    • Clean, user-friendly interface

    • File upload functionality

    • Document type selection

    • Israeli/non-Israeli toggle

    • File renaming capabilities

  2. Backend (Code.gs)

    • Folder structure management

    • File routing logic

    • Google Drive API integration

    • Error handling and validation

Key Features

1. Intelligent Folder Structure

The application maintains a deterministic folder structure:

šŸ“ Root
ā”œā”€ā”€ šŸ“ 2024
ā”‚   ā”œā”€ā”€ šŸ“ 01_Jan
ā”‚   ā”œā”€ā”€ šŸ“ 02_Feb
ā”‚   ā””ā”€ā”€ šŸ“ 03_Mar
ā””ā”€ā”€ šŸ“ 2025
    ā”œā”€ā”€ šŸ“ 01_Jan
    ā”œā”€ā”€ šŸ“ 02_Feb
    ā””ā”€ā”€ šŸ“ 03_Mar

2. Batch Upload Processing

Users can upload up to 10 documents simultaneously, streamlining the process of organizing multiple files.

3. Smart Document Routing

The application intelligently routes documents based on:

  • Document type (invoices, receipts, expenses)

  • Israeli/non-Israeli classification

  • Current date (for automatic month/year folder selection)

4. Custom File Naming

Users can rename files before routing to make them more descriptive and easier to find later.

Deployment Process

Deploying the application is straightforward:

  1. Create a new Google Apps Script project

  2. Add the Code.gs and index.html files

  3. Configure deployment settings as a web app

  4. Set appropriate access permissions

  5. Deploy and obtain the public URL

Benefits

This application offers several key advantages:

  1. Time Savings: Automates the tedious process of manually organizing documents

  2. Consistency: Ensures a uniform folder structure across all document types

  3. Compliance: Separates Israeli and international documents for easier tax reporting

  4. Scalability: Handles growing document volumes with ease

  5. User-Friendly: Intuitive interface requires minimal training

Open Source and Available for All

The project is licensed under CC-BY-4.0, allowing anyone to:

  • Use the application for their own document management needs

  • Modify and adapt the code for specific requirements

  • Share and distribute the solution with others

Screenshots (V1)

This tool demonstrates how Google Apps Script can be leveraged to create practical solutions for real-world business challenges. By automating document management, it frees up valuable time that can be better spent on core business activities.

The source code is available on GitHub, and I welcome contributions from the community to enhance its functionality further.

Code (V1)

Create a new project in Google Apps Script.

Save the first of these code blocks as Code.js (note: capitalisation) and the second as index.html (note: lower case ā€˜iā€™). The images are served from Cloudinary so you don't need to worry about hosting them locally.

Make sure to choose the appropriate permission for your deployment and select type as web app.

Code.js

// Month names mapping
const MONTHS = [
  '01_Jan', '02_Feb', '03_Mar', '04_Apr', '05_May', '06_Jun',
  '07_Jul', '08_Aug', '09_Sep', '10_Oct', '11_Nov', '12_Dec'
];

function doGet() {
  return HtmlService.createHtmlOutputFromFile('index')
    .setTitle('Accounting File Router')
    .setFaviconUrl('https://www.google.com/images/drive/drive-48.png');
}

/**
 * Process uploaded files and route them to the appropriate folder
 * @param {Object[]} files - Array of file blobs
 * @param {string} folderType - Type of folder (EXPENSE, INVOICES, RECEIPTS)
 * @returns {Object} Processing result
 */
/**
 * Create folder structure for selected folder types
 * @param {string[]} folderTypes - Array of folder types to create structure for
 * @returns {Object} Creation result
 */
function createFolderStructure(folderTypes) {
  try {
    const config = JSON.parse(PropertiesService.getUserProperties().getProperty('accountingRouterConfig') || '{}');
    const now = new Date();
    const year = now.getFullYear().toString();
    const month = MONTHS[now.getMonth()];
    let created = 0;
    
    folderTypes.forEach(folderType => {
      const folderId = config[folderType];
      if (!folderId) {
        throw new Error(`Folder ID not configured for ${folderType}`);
      }
      
      // Get or create the folder structure
      const rootFolder = DriveApp.getFolderById(folderId);
      const yearFolder = findOrCreateFolder(rootFolder, year);
      const monthFolder = findOrCreateFolder(yearFolder, month);
      findOrCreateFolder(monthFolder, 'Israeli'); // Create Israeli subfolder
      created++;
    });
    
    return {
      success: true,
      message: `Successfully created folder structure for ${created} folder type(s)`
    };
  } catch (error) {
    return {
      success: false,
      message: `Error creating folder structure: ${error.toString()}`
    };
  }
}

function processFiles(files, folderType, monthOverride) {
  try {
    // Get folder ID from the client-side configuration
    const config = JSON.parse(PropertiesService.getUserProperties().getProperty('accountingRouterConfig') || '{}');
    const folderId = config[folderType];
    
    if (!folderId) {
      throw new Error('Folder ID not configured for ' + folderType);
    }

    // Get the target date (either from override or current)
    let year, month;
    if (monthOverride) {
      const [targetYear, targetMonth] = monthOverride.split('-');
      year = targetYear;
      month = MONTHS[parseInt(targetMonth) - 1];
    } else {
      const now = new Date();
      year = now.getFullYear().toString();
      month = MONTHS[now.getMonth()];
    }
    
    // Get the root folder based on type
    const rootFolder = DriveApp.getFolderById(folderId);
    
    // Find or create year folder
    let yearFolder = findOrCreateFolder(rootFolder, year);
    
    // Find or create month folder
    let monthFolder = findOrCreateFolder(yearFolder, month);
    
    // Process each file
    const results = [];
    files.forEach(file => {
      let targetFolder = monthFolder;
      
      // If file is marked as Israeli, create/get Israeli subfolder
      if (file.isIsraeli) {
        targetFolder = findOrCreateFolder(monthFolder, 'Israeli');
      }
      
      const blob = Utilities.newBlob(
        Utilities.base64Decode(file.bytes),
        file.mimeType,
        file.fileName
      );
      const newFile = targetFolder.createFile(blob);
      results.push({
        name: file.fileName,
        success: true,
        url: newFile.getUrl()
      });
    });
    
    return {
      success: true,
      message: `Successfully processed ${files.length} file(s)`,
      results: results
    };
  } catch (error) {
    return {
      success: false,
      message: `Error processing files: ${error.toString()}`,
      results: []
    };
  }
}

/**
 * Get saved configuration from user properties
 * @returns {Object} Configuration object with folder IDs
 */
function getConfiguration() {
  try {
    const config = PropertiesService.getUserProperties().getProperty('accountingRouterConfig');
    return config ? JSON.parse(config) : null;
  } catch (error) {
    console.error('Failed to get configuration:', error);
    return null;
  }
}

/**
 * Find a folder by name or create it if it doesn't exist
 * @param {GoogleAppsScript.Drive.Folder} parentFolder - Parent folder
 * @param {string} folderName - Name of folder to find/create
 * @returns {GoogleAppsScript.Drive.Folder} The found or created folder
 */

/**
 * Save configuration to user properties
 * @param {Object} config - Configuration object with folder IDs
 * @returns {Object} Save result
 */
function saveConfiguration(config) {
  try {
    PropertiesService.getUserProperties().setProperty('accountingRouterConfig', JSON.stringify(config));
    return { success: true };
  } catch (error) {
    throw new Error('Failed to save configuration: ' + error.toString());
  }
}

function findOrCreateFolder(parentFolder, folderName) {
  const folders = parentFolder.getFoldersByName(folderName);
  if (folders.hasNext()) {
    return folders.next();
  }
  return parentFolder.createFolder(folderName);
}

Index.html (but save with lower case ā€˜iā€™)!


<!DOCTYPE html>
<html>
<head>
  <base target="_top">
  <meta charset="UTF-8">
  <title>Google Drive Accounting File Router</title>
  <style>
    @keyframes slideUp {
      from {
        transform: translateY(100%);
        opacity: 0;
      }
      to {
        transform: translateY(0);
        opacity: 1;
      }
    }

    @keyframes slideDown {
      from {
        transform: translateY(0);
        opacity: 1;
      }
      to {
        transform: translateY(100%);
        opacity: 0;
      }
    }

    .israeli-animation {
      animation: slideUp 0.5s ease forwards;
    }

    .israeli-animation-exit {
      animation: slideDown 0.5s ease forwards;
    }

    /* Tab Styles */
    .tabs {
      display: flex;
      margin-bottom: 20px;
      border-bottom: 2px solid #ddd;
    }
    .tab {
      padding: 10px 20px;
      cursor: pointer;
      border: none;
      background: none;
      font-weight: bold;
      color: #666;
    }
    .tab.active {
      color: #4285f4;
      border-bottom: 2px solid #4285f4;
      margin-bottom: -2px;
    }
    .tab-content {
      display: none;
    }
    .tab-content.active {
      display: block;
    }
    /* Config Styles */
    .config-group {
      margin-bottom: 15px;
    }
    .config-group input {
      width: 100%;
      padding: 8px;
      border: 1px solid #ddd;
      border-radius: 4px;
    }
    .help-text {
      display: block;
      margin-top: 4px;
      color: #666;
      font-size: 0.9em;
    }
    body {
      font-family: Arial, sans-serif;
      max-width: 800px;
      margin: 20px auto;
      padding: 20px;
      background-color: #f5f5f5;
    }
    .container {
      background-color: white;
      padding: 20px;
      border-radius: 8px;
      box-shadow: 0 2px 4px rgba(0,0,0,0.1);
    }
    .form-group {
      margin-bottom: 20px;
    }
    label {
      display: block;
      margin-bottom: 8px;
      font-weight: bold;
    }
    select {
      width: 100%;
      padding: 8px;
      border: 1px solid #ddd;
      border-radius: 4px;
      margin-bottom: 16px;
    }
    .file-list {
      border: 2px dashed #ddd;
      border-radius: 4px;
      padding: 10px;
      margin-bottom: 16px;
      min-height: 150px;
      max-height: 300px;
      overflow-y: auto;
      transition: all 0.3s ease;
      background-color: #fff;
      position: relative;
    }
    .file-list.drag-over {
      border-color: #4285f4;
      background-color: #f8f9fa;
    }
    .drop-message {
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
      color: #666;
      font-size: 1.1em;
      text-align: center;
      pointer-events: none;
    }
    .file-item {
      display: flex;
      justify-content: space-between;
      align-items: center;
      padding: 8px;
      border-bottom: 1px solid #eee;
    }
    .file-item:last-child {
      border-bottom: none;
    }
    .button-group {
      display: flex;
      gap: 10px;
      margin-top: 20px;
    }
    button {
      padding: 10px 20px;
      border: none;
      border-radius: 4px;
      cursor: pointer;
      font-weight: bold;
    }
    .upload-btn {
      background-color: #4285f4;
      color: white;
    }
    .clear-btn {
      background-color: #dc3545;
      color: white;
    }
    .status {
      margin-top: 20px;
      padding: 10px;
      border-radius: 4px;
      display: none;
    }
    .success {
      background-color: #d4edda;
      color: #155724;
      border: 1px solid #c3e6cb;
    }
    .error {
      background-color: #f8d7da;
      color: #721c24;
      border: 1px solid #f5c6cb;
    }
    .loading {
      display: none;
      text-align: center;
      margin: 20px 0;
    }
    .checkbox-label {
      display: inline-flex;
      align-items: center;
      margin: 0;
      gap: 5px;
      font-weight: normal;
    }
    .file-rename {
      width: 100%;
      padding: 4px;
      margin-bottom: 4px;
      border: 1px solid #ddd;
      border-radius: 4px;
    }
    .file-original {
      color: #666;
      font-size: 0.8em;
      margin-bottom: 4px;
    }
  </style>
</head>
<body>
  <div class="container">
    <h1>Google Drive Accounting File Router</h1>
    <h2 style="color: #666; margin: 0 0 15px 0;">Routes accounting documents in Google Drive</h2>
    <div id="currentMonth" style="color: #666; margin-bottom: 15px; display: flex; align-items: center; gap: 10px;">
      <span></span>
      <img src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAALCAYAAAB24g05AAAAxUlEQVQoU2P8//8/AyUAIwUGwA2orKxk/P//P8PmzZvRvQszYAEQnwBiZSB+D8SbgPgMEP8F4t1AvBiIQWwUgGKAGxCvBeI/QCwAxIeB+BUQGwAxyIA9QCwOxMgGIGvGMOA/EDsC8WMg/gTEpkD8GIjPArEFEIMM2QTESkCMzQCQpBUQPwBiEH8BEL8BYgMgBhmyE4jFgBjDAJBkHhDfAmKQQisg/gDEukAMMmQDECsCMU4DQAoSgfg6EIMM+QvEq4AYJM6AAgAp8kAx/Db8YAAAAABJRU5ErkJggg==" alt="Israeli Flag" style="width: 24px; height: 24px;">
    </div>

    <div class="tabs">
      <button class="tab active" onclick="showTab('upload')">Upload Files</button>
      <button class="tab" onclick="showTab('config')">Configuration</button>
      <button class="tab" onclick="showTab('howto')">How To Use</button>
      <button class="tab" onclick="showTab('create')">Create Folders</button>
    </div>

    <!-- Upload Tab -->
    <div id="uploadTab" class="tab-content active">
      <div class="form-group">
        <label for="folderType">Select Document Type:</label>
        <select id="folderType">
          <option value="">Please select...</option>
          <option value="EXPENSE">Expense</option>
          <option value="INVOICES">Invoice</option>
          <option value="RECEIPTS">Receipt</option>
        </select>

        <label for="monthOverride">Target Month:</label>
        <select id="monthOverride">
          <option value="">Current Month</option>
        </select>
      </div>

      <div class="form-group">
        <label>Select Files (up to 10):</label>
        <input type="file" id="files" multiple accept="application/pdf,image/*,.doc,.docx,.xls,.xlsx" style="display: none;">
        <button onclick="document.getElementById('files').click()" class="upload-btn">Choose Files</button>
      </div>

      <div class="file-list" id="fileList" ondrop="handleDrop(event)" ondragover="handleDragOver(event)" ondragleave="handleDragLeave(event)">
        <div class="drop-message">
          Drag and drop files here<br>
          <small>or use the Choose Files button</small>
        </div>
        <div id="fileListContent"></div>
      </div>

      <div class="button-group">
        <button onclick="processFiles()" class="upload-btn">Upload Files</button>
        <button onclick="clearAll()" class="clear-btn">Clear All</button>
      </div>

      <div class="loading" id="loading">
        Processing files... Please wait...
      </div>

      <div id="statusImage" style="text-align: center; margin: 20px 0;">
        <img src="/images/hashnode-imports/google-drive-accounting-doc-router-for-israeli-freelancers-apps-script-otidlvt8pofbzcpknbc6.png" alt="Status" style="max-width: 200px;">
      </div>

      <div class="status" id="status"></div>
    </div>

    <!-- Config Tab -->
    <div id="configTab" class="tab-content">
      <h2>Folder Configuration</h2>
      <p>Configure the Google Drive folders for each document type. You can paste the full folder URL or just the folder ID.</p>
      
      <div class="config-group">
        <label for="expenseFolder">Expense Folder:</label>
        <input type="text" id="expenseFolder" placeholder="Paste folder URL or ID" onchange="handleFolderInput(this)">
        <small class="help-text">Example: https://drive.google.com/drive/folders/YOUR_FOLDER_ID</small>
      </div>
      
      <div class="config-group">
        <label for="invoicesFolder">Invoices Folder:</label>
        <input type="text" id="invoicesFolder" placeholder="Paste folder URL or ID" onchange="handleFolderInput(this)">
        <small class="help-text">Example: https://drive.google.com/drive/folders/YOUR_FOLDER_ID</small>
      </div>
      
      <div class="config-group">
        <label for="receiptsFolder">Receipts Folder:</label>
        <input type="text" id="receiptsFolder" placeholder="Paste folder URL or ID" onchange="handleFolderInput(this)">
        <small class="help-text">Example: https://drive.google.com/drive/folders/YOUR_FOLDER_ID</small>
      </div>
      
      <button onclick="saveConfig()" class="upload-btn">Save Configuration</button>
      <div class="status" id="configStatus"></div>
    </div>

    <!-- How To Use Tab -->
    <div id="howtoTab" class="tab-content">
      <h2>How To Use</h2>
      <div style="line-height: 1.6">
        <p style="margin-bottom: 20px;">This app was developed by Daniel Rosehill (danielrosehill.com) using Claude Sonnet 3.5 & Cline.</p>
        
        <h3>Setup</h3>
        <ol>
          <li>Go to the Configuration tab and set up your folder IDs for each document type.</li>
          <li>Use the Create Folders tab to create the initial folder structure if needed.</li>
        </ol>

        <h3>Uploading Files</h3>
        <ol>
          <li>Select the document type (Expense, Invoice, or Receipt).</li>
          <li>Choose files using the button or drag and drop them into the designated area.</li>
          <li>Optionally rename files before uploading.</li>
          <li>For Israeli documents, check the "Israeli" checkbox next to the file.</li>
          <li>Click "Upload Files" to process the documents.</li>
        </ol>

        <h3>Folder Structure</h3>
        <p>Files are automatically organized in the following structure:</p>
        <pre style="background: #f5f5f5; padding: 10px; border-radius: 4px;">
Root Folder (Expense/Invoices/Receipts)
ā””ā”€ā”€ Year (e.g., 2024)
    ā””ā”€ā”€ Month (e.g., 01_Jan)
        ā”œā”€ā”€ Regular files
        ā””ā”€ā”€ Israeli (folder for Israeli documents)
        </pre>
      </div>
    </div>

    <!-- Create Folders Tab -->
    <div id="createTab" class="tab-content">
      <h2>Create Folder Structure</h2>
      <p>Use this section to create the initial folder structure for your documents. This will create the year and month folders according to the current date.</p>
      
      <div class="form-group">
        <label>Select folder types to create:</label>
        <div style="margin: 10px 0;">
          <label class="checkbox-label">
            <input type="checkbox" id="createExpense" checked> Expense Folders
          </label>
        </div>
        <div style="margin: 10px 0;">
          <label class="checkbox-label">
            <input type="checkbox" id="createInvoices" checked> Invoice Folders
          </label>
        </div>
        <div style="margin: 10px 0;">
          <label class="checkbox-label">
            <input type="checkbox" id="createReceipts" checked> Receipt Folders
          </label>
        </div>
      </div>
      
      <button onclick="createFolderStructure()" class="upload-btn">Create Folders</button>
    </div>
  </div>

  <script>
    const months = ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December'];
    
    function updateStatusImage(state) {
      const statusImage = document.getElementById('statusImage').querySelector('img');
      switch(state) {
        case 'ready':
          statusImage.src = 'https://res.cloudinary.com/domtm8wiy/image/upload/v1738782120/accounting-router/otidlvt8pofbzcpknbc6.png';
          break;
        case 'uploading':
          statusImage.src = 'https://res.cloudinary.com/domtm8wiy/image/upload/v1738782120/accounting-router/lvameumllsc7lny3pddj.png';
          break;
        case 'saved':
          statusImage.src = 'https://res.cloudinary.com/domtm8wiy/image/upload/v1738782120/accounting-router/izle2zu9xgfkmkskmv2r.png';
          break;
        case 'israeli':
          const img = statusImage;
          img.src = 'https://res.cloudinary.com/domtm8wiy/image/upload/v1738782120/accounting-router/yiw6yhd5r6oc4icka3ad.png';
          img.classList.add('israeli-animation');
          setTimeout(() => {
            img.classList.add('israeli-animation-exit');
            setTimeout(() => {
              img.classList.remove('israeli-animation', 'israeli-animation-exit');
              updateStatusImage('ready');
            }, 3000);
          }, 500);
          break;
      }
    }

    function populateMonthsDropdown() {
      const monthSelect = document.getElementById('monthOverride');
      const now = new Date();
      const currentYear = now.getFullYear();
      
      // Add past 12 months as options
      for (let i = 0; i < 12; i++) {
        const date = new Date(currentYear, now.getMonth() - i, 1);
        const monthNum = String(date.getMonth() + 1).padStart(2, '0');
        const monthName = months[date.getMonth()];
        const year = date.getFullYear();
        const value = `${year}-${monthNum}`;
        const text = `${monthNum}_${monthName} ${year}`;
        
        const option = new Option(text, value);
        monthSelect.add(option);
      }
    }

    window.onload = function() {
      // Display current accounting month and populate months dropdown
      populateMonthsDropdown();
      const now = new Date();
      document.getElementById('currentMonth').querySelector('span').textContent = `Current Accounting Month: ${String(now.getMonth() + 1).padStart(2, '0')}_${months[now.getMonth()]} ${now.getFullYear()}`;
      
      // First try to load from localStorage for immediate display
      const localConfig = JSON.parse(localStorage.getItem('accountingRouterConfig') || '{}');
      document.getElementById('expenseFolder').value = localConfig.EXPENSE || '';
      document.getElementById('invoicesFolder').value = localConfig.INVOICES || '';
      document.getElementById('receiptsFolder').value = localConfig.RECEIPTS || '';
      
      // Then load from server-side storage which is the source of truth
      google.script.run
        .withSuccessHandler(config => {
          if (config) {
            document.getElementById('expenseFolder').value = config.EXPENSE || '';
            document.getElementById('invoicesFolder').value = config.INVOICES || '';
            document.getElementById('receiptsFolder').value = config.RECEIPTS || '';
            // Update localStorage to match server state
            localStorage.setItem('accountingRouterConfig', JSON.stringify(config));
          }
        })
        .getConfiguration();

      // Set initial status image
      updateStatusImage('ready');
    };

    function extractFolderId(input) {
      // Handle various Google Drive URL formats
      const patterns = [
        /\/folders\/([a-zA-Z0-9_-]+)(?:\/|\?|$)/,  // /folders/ID format
        /id=([a-zA-Z0-9_-]+)(?:&|$)/,              // id=ID format
        /^([a-zA-Z0-9_-]+)$/                       // Direct ID format
      ];

      for (const pattern of patterns) {
        const match = input.match(pattern);
        if (match) return match[1];
      }

      return input; // Return original input if no pattern matches
    }

    function handleFolderInput(input) {
      const value = input.value.trim();
      if (value) {
        const folderId = extractFolderId(value);
        input.value = folderId;
      }
    }

    function showTab(tabName) {
      // Hide all tabs
      document.querySelectorAll('.tab-content').forEach(tab => {
        tab.classList.remove('active');
      });
      document.querySelectorAll('.tab').forEach(tab => {
        tab.classList.remove('active');
      });
      
      // Show selected tab
      document.getElementById(tabName + 'Tab').classList.add('active');
      document.querySelector(`.tab[onclick="showTab('${tabName}')"]`).classList.add('active');
    }

    function saveConfig() {
      const config = {
        EXPENSE: document.getElementById('expenseFolder').value,
        INVOICES: document.getElementById('invoicesFolder').value,
        RECEIPTS: document.getElementById('receiptsFolder').value
      };
      
      localStorage.setItem('accountingRouterConfig', JSON.stringify(config));
      
      const statusDiv = document.getElementById('configStatus');
      statusDiv.className = 'status success';
      statusDiv.style.display = 'block';
      statusDiv.innerHTML = 'Saving configuration...';
      
      // Sync with Google Apps Script properties
      google.script.run
        .withSuccessHandler(() => {
          statusDiv.innerHTML = 'Configuration saved successfully! āœ“';
          setTimeout(() => {
            statusDiv.style.display = 'none';
          }, 3000);
        })
        .withFailureHandler(error => {
          statusDiv.className = 'status error';
          statusDiv.innerHTML = 'Error saving configuration: ' + error.message;
        })
        .saveConfiguration(config);
    }

    function createFolderStructure() {
      const config = JSON.parse(localStorage.getItem('accountingRouterConfig') || '{}');
      const folderTypes = [];
      
      if (document.getElementById('createExpense').checked) folderTypes.push('EXPENSE');
      if (document.getElementById('createInvoices').checked) folderTypes.push('INVOICES');
      if (document.getElementById('createReceipts').checked) folderTypes.push('RECEIPTS');
      
      if (folderTypes.length === 0) {
        showStatus('Please select at least one folder type.', true);
        return;
      }

      document.getElementById('loading').style.display = 'block';
      
      google.script.run
        .withSuccessHandler(response => {
          document.getElementById('loading').style.display = 'none';
          showStatus(response.message, !response.success);
        })
        .withFailureHandler(error => {
          document.getElementById('loading').style.display = 'none';
          showStatus('Error creating folders: ' + error.message, true);
        })
        .createFolderStructure(folderTypes);
    }

    let selectedFiles = [];

    document.getElementById('files').addEventListener('change', function(e) {
      const files = Array.from(e.target.files).map(file => ({
        file,
        isIsraeli: false,
        newName: file.name
      }));
      
      if (files.length > 10) {
        alert('Please select up to 10 files only.');
        return;
      }
      
      selectedFiles = files;
      updateFileList();
    });

    function toggleIsraeli(index) {
      selectedFiles[index].isIsraeli = !selectedFiles[index].isIsraeli;
      if (selectedFiles[index].isIsraeli) {
        updateStatusImage('israeli');
        setTimeout(() => updateStatusImage('ready'), 1000);
      }
      updateFileList();
    }

    function updateFileName(index, newName) {
      selectedFiles[index].newName = newName;
    }

    function handleDragOver(event) {
      event.preventDefault();
      event.stopPropagation();
      document.getElementById('fileList').classList.add('drag-over');
    }

    function handleDragLeave(event) {
      event.preventDefault();
      event.stopPropagation();
      document.getElementById('fileList').classList.remove('drag-over');
    }

    function handleDrop(event) {
      event.preventDefault();
      event.stopPropagation();
      
      const fileList = document.getElementById('fileList');
      fileList.classList.remove('drag-over');
      
      const droppedFiles = Array.from(event.dataTransfer.files).map(file => ({
        file,
        isIsraeli: false,
        newName: file.name
      }));

      if (selectedFiles.length + droppedFiles.length > 10) {
        alert('Total number of files cannot exceed 10.');
        return;
      }

      selectedFiles = [...selectedFiles, ...droppedFiles];
      updateFileList();
    }

    function updateFileList() {
      const fileListContent = document.getElementById('fileListContent');
      if (selectedFiles.length === 0) {
        fileListContent.innerHTML = '';
        return;
      }

      fileListContent.innerHTML = selectedFiles.map((fileObj, index) => `
        <div class="file-item">
          <div style="flex: 1;">
            <input type="text" 
                   class="file-rename"
                   value="${fileObj.newName}"
                   onchange="updateFileName(${index}, this.value)"
                   placeholder="Enter new filename">
            <div class="file-original">Original: ${fileObj.file.name}</div>
          </div>
          <div style="display: flex; align-items: center; gap: 10px; margin-left: 10px;">
            <label class="checkbox-label">
              <input type="checkbox" 
                     onchange="toggleIsraeli(${index})" 
                     ${fileObj.isIsraeli ? 'checked' : ''}>
              Israeli
            </label>
            <button onclick="removeFile(${index})" style="color: red; background: none; border: none;">Ɨ</button>
          </div>
        </div>
      `).join('');
    }

    function removeFile(index) {
      selectedFiles.splice(index, 1);
      updateFileList();
    }

    function clearAll() {
      selectedFiles = [];
      document.getElementById('folderType').value = '';
      document.getElementById('files').value = '';
      document.getElementById('status').style.display = 'none';
      updateStatusImage('ready');
      updateFileList();
    }

    function showStatus(message, isError = false) {
      const status = document.getElementById('status');
      status.className = 'status ' + (isError ? 'error' : 'success');
      status.style.display = 'block';
      status.innerHTML = message;
      if (!isError) {
        setTimeout(() => {
          status.style.display = 'none';
        }, 5000);
      }
    }

    function processFiles() {
      const folderType = document.getElementById('folderType').value;
      
      if (!folderType) {
        showStatus('Please select a document type.', true);
        return;
      }

      if (selectedFiles.length === 0) {
        showStatus('Please select at least one file.', true);
        return;
      }

      document.getElementById('loading').style.display = 'block';
      showStatus('Preparing to upload files...', false);
      updateStatusImage('uploading');
      
      // Create an array of promises for each file
      const promises = selectedFiles.map(fileObj => {
        return new Promise((resolve, reject) => {
          const reader = new FileReader();
          reader.onload = (e) => {
            resolve({
              fileName: fileObj.newName,
              mimeType: fileObj.file.type,
              bytes: e.target.result.split(',')[1],
              isIsraeli: fileObj.isIsraeli
            });
          };
          reader.onerror = (e) => reject(e);
          reader.readAsDataURL(fileObj.file);
        });
      });

      // Process all files
      Promise.all(promises)
        .then(fileObjects => {
          const monthOverride = document.getElementById('monthOverride').value;
          showStatus('Uploading files to Google Drive...', false);
          google.script.run
            .withSuccessHandler(response => {
              document.getElementById('loading').style.display = 'none';
              if (response.success) {
                let message = response.message + '<br><br>Files processed:';
                response.results.forEach(result => {
                  message += `<br>ā€¢ ${result.name} - <a href="${result.url}" target="_blank">View file</a>`;
                });
                showStatus(message);
                updateStatusImage('saved');
                setTimeout(() => {
                  clearAll();
                  updateStatusImage('ready');
                }, 3000);
              } else {
                showStatus(response.message, true);
                updateStatusImage('ready');
              }
            })
            .withFailureHandler(error => {
              document.getElementById('loading').style.display = 'none';
              showStatus('Error processing files: ' + error.message, true);
              updateStatusImage('ready');
            })
            .processFiles(fileObjects, folderType, monthOverride);
        })
        .catch(error => {
          document.getElementById('loading').style.display = 'none';
          showStatus('Error preparing files: ' + error.message, true);
          updateStatusImage('ready');
        });
    }
  </script>
</body>
</html>

This project was created by Daniel Rosehill (public at danielrosehill dot com) and is licensed under CC-BY-4.0.