0

今日、GAS を使用して 2 つの日付を比較するという問題に直面しcurrentDateましscheduledDaterow[3]

私はすでに 5 時間費やしました (運が悪かったことを願っています)。ここで説明する最初のことは、セルがどのように埋めscheduledDateられるかです:結果をまたはとしてセルに表示します (表示形式の設定方法によって異なります)。=SubtractDaysFromDate('2016'!A54, 8)8'2016'!A5408/07/201608.07.2016

=SubtractDaysFromDate()ソースコードは次のとおりです。

function SubtractDaysFromDate(date, d) {
  var output = new Date(date.getTime()-d*(24*3600*1000)); // d — количество вычитаемых дней, date — дата или ячейка с датой, из которой вычитается данное количество дней.
  return output;
}

これが私のスクリプトのソースコードです:

function SendElectronicMailing() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var sheetName = "Рассылка"; // Здесь указывается название листа в таблице.
var sheet = ss.getSheetByName(sheetName);
var startRow = 2; // Здесь первая строка для отправки рассылки.
var numRows = sheet.getLastRow()-1; // Здесь количество строк, которые необходимо обработать для отправки рассылки.
var dataRange = sheet.getRange(startRow, 1, numRows, 6);
var data = dataRange.getValues();
/* Здесь переменные с тематическими рассылками. */
var mailingChildrenHealth = "Здоровье детей перед отдыхом";
var mailingNonPersonalized = "–";
/* Здесь переменные со статусами электропочтовых рассылок. */
var statusMailingAwaiting = "Рассылка ожидает отправки";
var statusMailingSent = "Рассылка отправлена";
var statusMailingNotRequired = "Рассылка не требуется";
  for (var i = 0; i < data.length; ++i) {
    var row = data[i];
    var currentDate = new Date();
    var scheduledDate = new Date(row[3]); // Здесь указывается столбец с назначенной датой для отправки электропочтовой рассылки.
    var formattedCurrentDate = Utilities.formatDate(currentDate, "GMT+0300", "dd.MM.yyyy");
    Logger.log(currentDate);
    var formattedScheduledDate = Utilities.formatDate(scheduledDate, "GMT+0300", "dd.MM.yyyy");
    Logger.log("Getting new Date(): " + scheduledDate);
    Logger.log("Getting plain row[3]: " + row[3]);
    Logger.log("Using toString() and after replace(): " + row[3].toString().replace(/\./g, "/"));
    Logger.log("Using valueOf() of row[3]: " + scheduledDate.valueOf());
    var bookingNumber = [i+2];
    var contactFullName = row[0];
    var contactGivenName = contactFullName.split(" ").slice(0, -1).join(" ");
    var contactEmail = row[1];
    var mailingTopic = row[2];
    var cheapTicketsFinderLink = "http://go.ruslanchik.ru/";
    var mailingStatus = row[4]; // Столбец, в котором выставляется статус отправки/неотправки писем электропочтовой рассылки.
    var senderName = "Наталья Селецкая (Мини-гостиница Бердянская 56)";
    var replyTo = "natalya@berdyanskaya56.ru";
    Logger.log(formattedCurrentDate + " ==? " + formattedScheduledDate);


        /* Выставление статуса «Рассылка не требуется» для бронирований, по которым ввиду своего скудоумия не смог настроить отправку рассылки. */
        if (currentDate.valueOf() > scheduledDate.valueOf() && mailingStatus == statusMailingAwaiting) {
        sheet.getRange(startRow + i, 5).setValue(statusMailingNotRequired);
        Logger.log("Выставлен статус " + "«" + statusMailingNotRequired + "»" + " электропочтовой рассылки для бронирования № " + bookingNumber + " (" + contactFullName + ")" + " по причине несвоевременности её отправки гостю.");
        }

        /* Тематическая рассылка для гостей с детьми («забота о здоровье детей») с общей рассылкой о дешёвых билетах в Ейск и обратно. */
        if (formattedCurrentDate == formattedScheduledDate && mailingTopic == mailingChildrenHealth && mailingStatus == statusMailingAwaiting) {
        /* Здесь рассылка о здоровье детей перед отдыхом. */
        var subject_children_health = "Email subject";
        var message_children_health = "Email body.";
        MailApp.sendEmail(contactEmail, subject_children_health, message_children_health, {name: senderName, replyTo: replyTo});
        /* Здесь рассылка о дешёвых билетах в Ейск и обратно для гостей с детьми. */
        var subject_cheap_tickets = "Email subject";
        var message_cheap_tickets = "Email body";
        MailApp.sendEmail(contactEmail, subject_cheap_tickets, message_cheap_tickets, {name: senderName, replyTo: replyTo});
        /* А здесь уже проставление статуса отправки. */
        sheet.getRange(startRow + i, 5).setValue(statusMailingSent);
        Logger.log("Отправлены тематическая и общая электропочтовые рассылки для бронирования № " + bookingNumber + " (" + contactFullName + ")" + " гостю на " + contactEmail + ".");
        }
        SpreadsheetApp.flush(); // Здесь завершается обновление ячеек 5-го столбца, в котором проставляется статус отправки электропочтовых рассылок гостям.
  }
}

入力を確認するには、上記のリンクに移動して、2016(このシート=SubtractDatesFromDate()から減算する日付を取得する)という名前のシートとРассылка(このシートで私のスクリプトが実行される) を確認できます。

続行するには、私のスクリプトは、=SubtractDaysFromDate()呼び出されたシートのこれらの減算された日付 ( ) を持つすべてのセルを調べ、現在の日付が(つまり、このシートのすべてのセルを含む列)Рассылкаと等しいかどうかを確認してから、電子メールを送信します。scheduledDate=SubtractDaysFromDate()

scheduledDate問題は、 (結果の )の日付のごく一部=SubtractDaysFromDate()は正しく認識されますが、別の部分は認識されずに として取得されること01.01.1970です。

問題を説明するために、スクリプト ログを次に示します。

[16-07-08 14:31:40:171 EAT] 08.07.2016 ==? 01.01.1970
[16-07-08 14:31:40:172 EAT] Fri Jul 08 14:31:40 GMT+03:00 2016
[16-07-08 14:31:40:173 EAT] Getting new Date(): Fri Jun 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:173 EAT] Getting plain row[3]: Fri Jun 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:174 EAT] Using toString() and after replace(): Fri Jun 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:175 EAT] Using valueOf() of row[3]: 1464901200000
[16-07-08 14:31:40:175 EAT] 08.07.2016 ==? 03.06.2016
[16-07-08 14:31:40:176 EAT] Fri Jul 08 14:31:40 GMT+03:00 2016
[16-07-08 14:31:40:177 EAT] Getting new Date(): Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:178 EAT] Getting plain row[3]: Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:178 EAT] Using toString() and after replace(): Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:179 EAT] Using valueOf() of row[3]: 1470171600000
[16-07-08 14:31:40:179 EAT] 08.07.2016 ==? 03.08.2016
[16-07-08 14:31:40:181 EAT] Fri Jul 08 14:31:40 GMT+03:00 2016
[16-07-08 14:31:40:181 EAT] Getting new Date(): Invalid Date
[16-07-08 14:31:40:182 EAT] Getting plain row[3]: #ERROR!
[16-07-08 14:31:40:182 EAT] Using toString() and after replace(): #ERROR!
[16-07-08 14:31:40:183 EAT] Using valueOf() of row[3]: NaN
[16-07-08 14:31:40:183 EAT] 08.07.2016 ==? 01.01.1970

上記のコードを見ると、一部の日付では比較が完全に機能することが明らかです。

[16-07-08 14:31:40:176 EAT] Fri Jul 08 14:31:40 GMT+03:00 2016 (note: this a current date)
[16-07-08 14:31:40:177 EAT] Getting new Date(): Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:178 EAT] Getting plain row[3]: Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:178 EAT] Using toString() and after replace(): Wed Aug 03 2016 00:00:00 GMT+0300 (MSK)
[16-07-08 14:31:40:179 EAT] Using valueOf() of row[3]: 1470171600000
[16-07-08 14:31:40:179 EAT] 08.07.2016 ==? 03.08.2016

それらの他の部分については、そうではありません:

[16-07-08 14:31:40:181 EAT] Fri Jul 08 14:31:40 GMT+03:00 2016 (note: this a current date)
[16-07-08 14:31:40:181 EAT] Getting new Date(): Invalid Date
[16-07-08 14:31:40:182 EAT] Getting plain row[3]: #ERROR!
[16-07-08 14:31:40:182 EAT] Using toString() and after replace(): #ERROR!
[16-07-08 14:31:40:183 EAT] Using valueOf() of row[3]: NaN
[16-07-08 14:31:40:183 EAT] 08.07.2016 ==? 01.01.1970

理解を深めるために、GS シートへのリンクを次に示します: https://docs.google.com/spreadsheets/d/1azDfWZWDSVTVVKLaJZlsRDbW21-Cps8Hx8M2kbclw-g/edit?pli=1#gid=2147296153

正しく機能させるために重要なことを見逃していないか確認していただけますか? によって計算された日付の一部が=SubtractDatesFromDate()スクリプトで正しく認識され、別の部分が正しく認識されないのはなぜですか?

どうすれば修正できますか?この問題を解決するための結果志向の試みはすべて高く評価されます。


編集しました。これは、既に編集したコードの一部です。

function SendElectronicMailing1() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var bookingsSheetName = "2016"; // Здесь указывается название листа откуда брать даты заезда по бронированиям.
var mailingSheetName = "M1"; // Здесь указывается название листа где обновлять статусы отправки электропочтовой рассылки.
var bookingsTableSheet = ss.getSheetByName(bookingsSheetName);
var mailingListSheet = ss.getSheetByName(mailingSheetName);
var startRow = 2; // Здесь первая строка для отправки рассылки.
var bookingsNumRows = bookingsTableSheet.getLastRow()-1; // Здесь количество строк, которые необходимо обработать для отправки рассылки.
var mailingNumRows = mailingListSheet.getLastRow()-1;
var bookingsDataRange = bookingsTableSheet.getRange(startRow, 1, bookingsNumRows, 2);
var mailingDataRange = mailingListSheet.getRange(startRow, 1, mailingNumRows, 6);
var bookingsData = bookingsDataRange.getValues();
var mailingData = mailingDataRange.getValues();
/* Здесь переменные с тематическими рассылками. */
var mailingChildrenHealth = "Здоровье детей перед отдыхом";
var mailingNonPersonalized = "–";
/* Здесь переменные со статусами электропочтовых рассылок. */
var statusMailingAwaiting = "Рассылка ожидает отправки";
var statusMailingSent = "Рассылка отправлена";
var statusMailingNotRequired = "Рассылка не требуется";
  for (var i = 0; i < bookingsData.length; ++i) {
    var row = bookingsData[i];
    var currentDate = new Date();
    var scheduledDate = new Date(row[0]).setHours(0,0,0,0)+(-10*24*3600*1000); // Здесь указывается столбец с назначенной датой для отправки электропочтовой рассылки. Её еще можно считать через new Date().setHours(0,0,0,0)+(-10*24*3600*1000).
    var formattedCurrentDate = Utilities.formatDate(currentDate, "GMT+0300", "dd.MM.yyyy");
    Logger.log("It is a current date: " + currentDate);
    var formattedScheduledDate = Utilities.formatDate(scheduledDate, "GMT+0300", "dd.MM.yyyy");
    Logger.log("Getting new Date() of row " + [i+2] + ": " + scheduledDate);
    Logger.log("Getting clear value of row " + [i+2] + ": " + row[3]);
    Logger.log("Using toString() and after replace(): " + [i+2] + ": " + row[3].toString().replace(/\./g, "/"));
    Logger.log("Using valueOf() of row " + [i+2] + ": " + scheduledDate.valueOf());
    var bookingNumber = [i+2];
    var contactFullName = row[0];
    var contactGivenName = contactFullName.split(" ").slice(0, -1).join(" ");
    var contactEmail = row[1];
    var mailingTopic = mailingData.row[2];
    var cheapTicketsFinderLink = "http://go.ruslanchik.ru/";
    var mailingStatus = row[4]; // Столбец, в котором выставляется статус отправки/неотправки писем электропочтовой рассылки.
    var senderName = "Наталья Селецкая (Мини-гостиница Бердянская 56)";
    var replyTo = "natalya@berdyanskaya56.ru";
    Logger.log(formattedCurrentDate + " ==? " + formattedScheduledDate);
}
}

Mu の質問はrow[2]、シート内の値を取得し、それをリンク先Рассылкаの変数に書き込む方法です。適切な使用方法になりますか?for (var i = 0; i < bookingsData.length; ++i) {2016var mailingTopic = mailingData.row[2];

4

1 に答える 1

0

これはかなりばかげており、問題を解決できない可能性がありますが、GAS は非常に扱いにくい場合があることがわかりました。普段なら使い捨てなのですが、自分ではスクリプトを編集できないので…

左の '(' を移動してみてください。つまり、d の周りにあります。操作の順序でこれを処理する必要があることはわかっていますが、そうでない可能性がある場合、深刻な問題が発生する可能性があります。そして、他に何が問題を引き起こすのか考えられません。スプレッドシートの日付が間違ってフォーマットされていない限り、一部の日付のみで、そうではないようです.したがって、これの代わりに:

var output = new Date(date.getTime()-d*(24*3600*1000));

試す:

var output = new Date(date.getTime()-(d*24*3600*1000));

編集:

これは明らかに正確な結果ではありませんが、同じ SS 上の 2 つのシート間でセルを移動することを示しています。

function example() {

  var ss = SpreadsheetApp.getActiveSpreadsheet();
  var sheet1 = ss.getActiveSheet();
  var sheet2 = ss.getSheetByName('sheet2');

  var startRow = 1;
  var numRows = sheet1.getLastRow()-1; 
  var dataRange = sheet1.getRange(startRow, 1, numRows, 6);
  var data = dataRange.getValues();

  var row = []

  for (var i in data) {
    row.push(data[i] + 10);
  }

  sheet2.appendRow(row);
}
于 2016-07-08T15:09:00.580 に答える