отправить несколько строк вложения в Google DriveApp - PullRequest
2 голосов
/ 22 января 2020

Я пытаюсь отправить несколько строк данных с вложением из поля ввода в конце каждой строки. Я использую Google App-Script Webapp для этого. Я успешно создал массив объектов с текстовыми данными (такими как дата, имя, последний и т. Д. c.), Но не могу отправить вложение как часть объекта. Что я делаю не так?

Я также должен уточнить, что этот код не будет работать ни с одним, ни с несколькими вложениями. Я надеюсь, что смогу отправить несколько групп вложений (отсюда и массив объектов) одновременно.

Вот мой код на стороне клиента HTML / Javascript:

document.addEventListener("DOMContentLoaded", function() {
  document.getElementById("tripPost").addEventListener("click", addLine);
  document.getElementById("submitAll").addEventListener("click", addRecord);
});

//global variables for next functions
var submit = document.getElementById("tripPost");
var submittedTable = document.getElementById("submitted-data");
var mainEntry = document.getElementById("entry-table");
var submitAll = document.getElementById("submitAll");

submittedTable.addEventListener("click", addLine);
submittedTable.addEventListener("change", fileUpload);

function addLine() {
  document.getElementById("table-container").style.display = "block";

  var date = document.getElementById("date1").value;
  var efirst = document.getElementById("efirst").value;
  var elast = document.getElementById("elast").value;

  var row = document.createElement("tr");
  var col1 = document.createElement("td");
  col1.appendChild(document.createTextNode(date));
  col1.className = "postDateClass";
  var col2 = document.createElement("td");
  col2.appendChild(document.createTextNode(efirst));
  col2.className = "postEfirstClass";
  var col3 = document.createElement("td");
  col3.appendChild(document.createTextNode(elast));
  col3.className = "postElastClass";

  var col4 = document.createElement("td");

  row.appendChild(col1);
  row.appendChild(col2);
  row.appendChild(col3);
  row.appendChild(col4);

  submittedTable.appendChild(row);

  var uniqueID = "id" + new Date().getTime();
  var upload = document.createElement("input");
  upload.type = "file";
  upload.id = uniqueID;
  upload.name = "myReceipt";
  upload.className = "uploadClass";

  var label = document.createElement("label");
  label.innerHTML = "upload me please!";
  label.htmlFor = uniqueID;
  label.className = "custom-file-upload";

  var form = document.createElement("form");
  form.appendChild(upload);
  form.appendChild(label);

  col4.appendChild(form);
}

function fileUpload(e) {
  if (e.target.className === "uploadClass") {
    if (e.target.value) {
      var span = document.createElement("span");
      span.className = "uploadSpanText";
      span.innerHTML = e.target.value.match(/[\/\\]([\w\d\s\.\-\(\)]+)$/)[1];
      e.target.parentElement.appendChild(span);
      e.target.nextElementSibling.innerHTML = "uploaded!";
      e.target.nextElementSibling.style.border = "1px solid #a8e0b4";
      e.target.nextElementSibling.style.color = "#8bca9e";
    }
  }
}

function getFile(file) {
  return new Promise(resolve => {
    const fr = new FileReader();
    fr.onload = e => {
      const data = e.target.result.split(",");
      const obj = {
        fileName: file.name,
        mimeType: data[0].match(/:(\w.+);/)[1],
        data: data[1]
      };
      resolve(obj);
    };
    if (file) {
      fr.readAsDataURL(file);
    } else {
      reject("No File");
    }
  });
}

//gathers inputs and stores values in an object and runs the "addLine" function
async function addRecord(e) {
  var dateLines = document.querySelectorAll(".postDateClass");
  var eFirstLines = document.querySelectorAll(".postEfirstClass");
  var eLastLines = document.querySelectorAll(".postElastClass");
  var attachmentLines = document.querySelectorAll(".uploadClass");
  var mileageData = [];
  for (var i = 0; i < dateLines.length; i++) {
    var mileageLines = {};
    mileageLines.travelDate = dateLines[i].textContent;
    mileageLines.firstName = eFirstLines[i].textContent;
    mileageLines.lastName = eLastLines[i].textContent;
    mileageLines.receipt = await getFile(attachmentLines[i].parentNode);

    mileageData.push(mileageLines);
  }

  //send object to google. resets input elements
  google.script.run.userMileageSubmit(mileageData);
}

Вот код HTML для кода, с которым я работаю.

<div id="entry-table">
     <table>
        <h3 style="text-align:left"><u><b>Enter mileage information below.</b></u><br></h3>
        <thead>
         <tr>
              <th >Date</th>
              <th >First:</th>
              <th >Last:</th>
         </tr>
         </thead>

         <tbody id="table-data">
           <tr>
            <td>
              <div class="disabled-results" id="date">
                 <input placeholder="Start Date" id="date1" type="text" class="datekeeper" required>
                <label for="date1" class="active">Date:</label>
              </div>
            <td>
              <div class="disabled-results">
                 <input id ="efirst" type="text" class="validate" >
                 <label for="efirst" class="active">First:</label>
              </div>
            </td>
            <td>
              <div class="disabled-results">
                 <input id ="elast" type="text" class="validate" >
                 <label for="elast" class="active">Last:</label>
              </div>
            </td>
            <td>
               <div id="status">
                  <button id="tripPost" class="waves-effect waves-light btn-small blue darken-3">Add Trip</button>
               </div>
            </td>
         </tr>
       </tbody>
      </table>
</div><!---CLOSE ROW ---> 

<div class="autocomplete" id="table-container" style=display:none>
       <table>
         <thead>
          <tr id="header-titles">
              <th >Date</th>
              <th >First:</th>
              <th >Last:</th>
              <th >Receipt </th>
          </tr>
        </thead>
        <form>
        <tbody class="form" id="submitted-data">
          <div>
              <p>Thank you!</p>
          </div>
          </form>
         </tbody>
      </table>
      <br><br>
   </div>

<center>
  <div class="row">    
    <button  id="submitAll" class="waves-effect waves-light btn btn-large blue darken-3"><i class="material-icons left">directions_car</i>Submit All Mileage!</button>          
  </div>
</center>

Вот CSS

body {
    background-color: lightblue;
    margin-top: 80px;
    margin-bottom: 80px;
    margin-right: 80px;
    margin-left: 80px;    
    }

    h1{
    color: black;
    text-align: center;
    }
div.disabled-results{
  width: 175px;
  height: 80px;
  padding: 5px;
  margin: 5px;
  display: inline-table;
  box-sizing: border-box;
  text-align: center;
  }

input[type="file"]{
  display: none;
  }

  .custom-file-upload{
    border: 2px solid #000;
    width: 85px;
    display: inline-block;
    padding: 2px 1px;
    cursor: pointer;
    text-align: center;
  }

div.autocomplete{
width: 55px;
  height: 80px;
  padding: 5px;
  margin: 5px;
  display: inline-table;
  box-sizing: border-box;
  text-align: center;
  }

У меня все остальное работает, кроме отправки вложения (если есть) в каждой строке как части объекта.

Я уверен, что это можно сделать. Я попытался реализовать решение из этого видео , которое показывает вам, как загрузить файл, но я не использую onclick или this.parentNode, так как я не загружаю сразу после выбора файла и вместо этого делать массовую загрузку, когда пользователь сделал множество записей.

Любая помощь в понимании того, как это должно работать, будет принята с благодарностью.

Спасибо.

1 Ответ

2 голосов
/ 24 января 2020

Как насчет этой модификации? Пожалуйста, подумайте об этом как об одном из нескольких возможных ответов.

К сожалению, в этом случае файловый объект со стороны HTML нельзя напрямую отправить в Google Apps Script в виде большого двоичного объекта. Таким образом, в качестве одного из нескольких обходных путей, в этой модификации извлеченные файлы кодируются в данные base64 и отправляются в скрипт Служб Google. Затем на стороне скрипта Google Apps данные декодируются и сохраняются в виде файлов.

Пожалуйста, измените ваш скрипт следующим образом.

HTML и Javascript side:

Пожалуйста, измените addRecord() и добавьте getFile() следующим образом.

// Added
function getFile(file) {
  return new Promise((resolve, reject) => {
    const fr = new FileReader();
    fr.onload = e => {
      const data = e.target.result.split(",");
      const obj = {fileName: file.name, mimeType: data[0].match(/:(\w.+);/)[1], data: data[1]};
      resolve(obj);
    }
    if (file) {
      fr.readAsDataURL(file);
    } else {
      reject("No file");
    }
  });
}

async function addRecord(e) { // Modified
  var dateLines = document.querySelectorAll('.postDateClass');
  var attachmentLines = document.querySelectorAll('.uploadClass');

  var mileageData = [];
  for (var i=0; i<dateLines.length; i++){
    var mileageLines = {};
    mileageLines.firstName = document.getElementById("efirst").value; 
    mileageLines.lastName = document.getElementById("elast").value;
    mileageLines.date = dateLines[i].textContent;
    mileageLines.receipt = await getFile(attachmentLines[i].files[0]).catch(e => console.log(e));  // Modified
    mileageData.push(mileageLines); 
  };

  google.script.run.userMileageSubmit(mileageData);
};

Сторона скрипта Google Apps:

Пожалуйста, измените userMileageSubmit() следующим образом.

function userMileageSubmit(responses){
  responses.forEach(function(e) {
    var file = e.receipt;
    if (file) {
      var blob = Utilities.newBlob(Utilities.base64Decode(file.data), file.mimeType, file.fileName);
      var mainFolder = DriveApp.getFolderById('real-drive-link');
      var createFile = mainFolder.createFile(blob);
      var fileUrl = createFile.getUrl();
      Logger.log(fileUrl)
    }
  });

  // row.appendChild(col4)
  // submittedTable.appendChild(row)
}
  • Я не могу понять о row.appendChild(col4) и submittedTable.appendChild(row).
  • К сожалению, я не мог понять о вашей цели на userMileageSubmit(). Таким образом, в этой модификации полученные файлы сохраняются на Google Drive. А URL созданного файла можно увидеть в журнале.
    • Здесь, пожалуйста, измените это для вашей реальной ситуации.
  • Я не уверен насчет real-drive-link. В этом случае укажите идентификатор папки, в которой вы хотите сохранить файл.

Примечание:

  • В этой модификации предполагается, что ваш текущий addRecord() работает .
  • В этой модификации максимальный размер файла составляет 50 МБ, поскольку максимальный размер BLOB-объектов в скрипте Google Apps составляет 50 МБ. Пожалуйста, будьте осторожны.
  • Когда много файлов загружено, время обработки увеличивается. Пожалуйста, будьте осторожны с этим.

Ссылки:

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...