I have written a script for a client in a Spreadsheet whose time is GMT-5 while my TimeZone is GMT+5. The column H on the spreadsheet's tab has a value as date without any time factor e.g. 01/25/2019.
On the start of script function I am setting the format of column like under the following .setNumberFormat("MM/dd/yyyy"). Also note that I am working with .getDisplayValues() which returns me test for the Date column.
There is nowhere in the script I am explicitly updating DateValue its value. After manipulating some other values on the other column I happen to update the row by calling the .setValues([row]).
Then the date changes to 01/24/2019 and when I change the format to include time factor then it is like 01/24/2020 19:00:00.
Interesting point is it does not occur on my end where my timeZone is GMT+5. Even though I changed the Spreadsheet's Timezone to GMT-5 and on my system as well but still it doesn't occur on my end but on the client machine.
The runtime version of the script is:
"runtimeVersion": "V8"
This can happen when the script time zone and spreadsheet time zone are different and you format the date object
I cannot reproduce your exact situation from the information you provided, but I invite you to do the following test:
Set your spreadsheet timezone to any timezone that is ahead of your script time zone
Set cell A1 of your active sheet to =today()
Run the following code:
function myFunction(){
var displayDate = SpreadsheetApp.getActiveSheet().getRange("A1").getDisplayValue();
Logger.log("displayDate: " + displayDate);
var date = SpreadsheetApp.getActiveSheet().getRange("A1").getValue();
Logger.log("actual date retrieved by Apps Script: " + date);
var timeZone = Session.getScriptTimeZone();
Logger.log("ScriptTimeZone: " + timeZone);
var timezone2 = SpreadsheetApp.getActive().getSpreadsheetTimeZone();
Logger.log("SpreadsheetTimeZone: " + timezone2);
var newDate = Utilities.formatDate(date, timeZone, "MM/dd/yyyy");
Logger.log("formatted date that will be set back to the spreadsheet: " + newDate);
SpreadsheetApp.getActiveSheet().getRange("A1").setValue(newDate);
}
function updatePatientTransactions(transactionSheetValues, patientNam, insuranceName) {
var tempRow = [];
for(var i= start; i<= end-2; i++) {
tempRow = transactionSheetValues[i];
tempRow[0] = patientName;
tempRow[3] = insuranceName;
transactionSheet.getRange(i+2,1,1,transactionSheet.getLastColumn()).setValues([tempRow]);
}
}
Observe the logs and the output
Sample logs:
Stackdriver logs
Jun 8, 2020, 1:03:00 PM Info displayDate: 6/8/2020
Jun 8, 2020, 1:03:01 PM Info actual date retrieved by Apps Script: Sun Jun 07 2020 23:00:00 GMT+0200 (Central European Summer Time)
Jun 8, 2020, 1:03:01 PM Info ScriptTimeZone: Europe/Paris
Jun 8, 2020, 1:03:01 PM Info SpreadsheetTimeZone: Europe/Athens
Jun 8, 2020, 1:03:01 PM Info formatted date that will be set back to the spreadsheet: 06/07/2020
so the date will be set back to yesterday.
Why?
When you retrieve a value that is formatted as Date and not Date Time, that is if the time is not set - Apps Script automatically sets it to midnight.
When it is the 8th of June midnight in Athens, it is the 7th of June 23:00 o'clock in Paris
This is why the conversion from a spreadsheet timezone to a script timezone lying behind will lead to a date change
Now, if you format the date value as a date string or process it otherwise, the informaiton about the original timezone might get lost and when you set the datestring back to the spreadsheet, you will set back the Apps Script date - different from the spreadsheet date - so be careful!
Related
I have this code:
var time = new Date();
time = Utilities.formatDate(time, "GMT+02", "dd/MM/yy hh:mm");
When I use the time parameter as a file name or log it into a spreadsheet cell - it logs the hours part as am only.
For example, both 03:20 am (03:20) and 03:20 pm (15:20) will be logged as 03:20.
I am trying to log it correctly in a military time format (15:20).
How can I do so?
You want to show 03:20 pm as 15:20.
If my understanding is correct, how about this modification?
From:
hh:mm
To:
HH:mm
Reference:
Date and Time Patterns
H: Hour in day (0-23)
h: Hour in am/pm (1-12)
I am trying to write a small Google Script which takes data from some cells of Google Sheet and Paste it in the GMAIL.
However, while pasting the 'Date Values' it always displays it in the following manner:-
Mon Apr 15 2019 00:00:00 GMT+0530 (IST)
Fri Apr 26 2019 00:00:00 GMT+0530 (IST)
But I need the dates in an appropriate way i.e.
"Mon Apr 15 2019" or
"Fri 04/26/2019"
I gone through these possible options i.e. Utilities.formatDate & .Split but somehow I am not able to write these codes appropriately. Can you please help me with this matter. Below I have mentioned the entire issue in detail.
My Code
function temp2() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var dataSheet = ss.getSheetByName('Sheet1');
var templateSheet = ss.getSheetByName('Sheet2');
var emailTemplate2 = templateSheet.getRange("G1").getValue();
var rec = templateSheet.getRange("C1").getValue();
var sub = templateSheet.getRange("D1").getValue();
var date = templateSheet.getRange("E1").getValue();
// Logger.log(rec);
MailApp.sendEmail(rec, sub, emailTemplate2);
}
here var date = templateSheet.getRange("E1").getValue(); is the part of code which picks value of date.
Do let me know if you need more details in this regard
Regards,
Alok
Requirement:
Format date value from cell in Google Apps Script.
Solution:
Pass value to date constructor new Date() then format using Utilities.formatDate().
// pass date to date constructor
var date = new Date(templateSheet.getRange("E1").getValue());
// "Mon Apr 15 2019" example
var formattedDate = Utilities.formatDate(date, "GMT+0", 'E MMM dd yyyy');
// "Fri 04/26/2019" example
var formattedDate = Utilities.formatDate(date, "GMT+0", 'E MM/dd/yyyy');
Note: I've set this to timezone "GMT+0" by default, you can change this to whichever time zone you need.
References:
new Date() for date constructor.
Utilities.formatDate() for formatting dates in Google Apps Script.
SimpleDateFormat for date format strings.
I'm about to use google sheet as my database for my android app small project. I'm using Google Script to handle the request from my app.
In my google sheet, I store;
A2:A = date as dd/mm/yyyy e.g 21/12/2019 but
the display format is dd-MMM e.g 21-Dec
C2:D = time as hh:mm:ss e.g 21:00:00 but
the display format is hh:mm e.g 21:00
Yes, I need a different format for the display and input.
My google sheet:
When I use google script to get a value of the cell, it seems that it is reformatted
the date looks like this: Sat Jan 01 2019 00:00:00 GMT+0700 (ICT)
the time the other hand, change in value a bit. 20:00:00 to 19:52:48
Is there any function to get cell real values as text without being reformatted?
The only thing that I can think of is instead of using getValues(), I can use getDisplayValues(). The values will not be reformatted, but it is not a solution for me, as it will take the display format.
Snippet of my code:
function updateData(e, sheet) {
var tgl = e.parameter.tgl;
var dtg = e.parameter.dtg;
var plg = e.parameter.plg;
var lbr = e.parameter.lbr;
var rangeHead = sheet.getRange("A2:A");
var valuesHead = rangeHead.getValues();
var rangeFirst = sheet.getRange("C2:D")
var valuesFirst = rangeFirst.getValues();
var rangeSecond = sheet.getRange("G2:G")
var valuesSecond = rangeSecond.getValues();
for (var i = 0; i < valuesHead.length; i++) {
if (valuesHead[i][0] === tgl) {
if(dtg!="null") { valuesFirst[i][0] = dtg; }
if(plg!="null") { valuesFirst[i][1] = plg; }
if(lbr!="null") { valuesSecond[i][0] = lbr; }
break;
}
}
rangeFirst.setValues(valuesFirst);
rangeSecond.setValues(valuesSecond);
}
The code won't work as I will comparing 21/12/2019 with Sat Jan 01 2019 00:00:00 GMT+0700 (GMT).
[UPDATE 1]:
Thank you P-Burke for the enlightenment. Now, I have an idea to solve the date problem. I know that the script pulls the date as date object, but I am unaware that it also saves as a date object. (hehe my bad) I don't realize it as there is no autocomplete when I call values[0][0]. of course, as it recognizes the object type at the run time.
So, my workaround will be; I will call getDate, getMonth+1, and getYear. After that, I will compare with my parameter freely.
Though, the time cell still a bit confusing for me. the time offset is 18 minutes 12 seconds. I don't think it's because of timezone different and my computer clock. the timezone different is too big and I 've made sure that the script, spreadsheet, and local timezone all the same. My computer clock is also only a minute less behind.
[UPDATE 2]:
Alright, enough with the confusion. It seemed that the script converts the time to Date object respect to my local timezone. I got this answer from another thread. So, actually, my local timezone changes many times and some of them have offset smaller than hours unit (one of the timezones used in my area is UTC +7:07:12h). The only source documenting those changes I could find is from https://www.timeanddate.com/time/zone/indonesia/jakarta. Finally, I gave up. For my goodness sake, I will just use getDisplayValue and ignore the seconds. Unless you guys have any other workaround, I will be so grateful.
Thank you once again to the community.
Firstly, and I don't know if this is related to your issue, but the spreadsheet and the script each have their own timezone setting:
Spreadsheet: File >> Spreadsheet Settings >> Time Zone.
Script: File >> Project Properties >> Time Zone.
And if these are different that can lead to confusion. One answer, if all your users are in the same timezone, is to set them to the same. Alternatively these can be determined from within your script as described here, and logic included to handle any differences. I don't understand the few minutes time difference, perhaps your PC clock is inaccurate?
The other point, which I think is more relevant to your question is that you effectively have multiple date/time formats in play. The picture below shows that in the spreadsheet times are edited in one format (02/01/2019 09:00:00), but displayed in whatever format is defined for the cell using the format menu. Yet when the cell values are pulled into a script using getValues() and displayed they appear as follows: Values: [[Thu Jan 31 09:00:00 GMT+00:00 2019, Wed Jan 02 09:00:00 GMT+00:00 2019]].
Yet in the code below, values[0][0] and values[0][1] are actually JavaScript Date() objects and can be compared in the usual way, alternatively they can be reformatted into whatever string format you require as illustrated in the code below:
function myFunction() {
var ss = SpreadsheetApp.getActive();
var ws = ss.getActiveSheet();
var input_range = ws.getRange("A1:B1");
var values = [];
values = input_range.getValues(); // Returns a multi-dimensional array, hence [0][0] to access.
Logger.log("Values: %s", values);
// As Date() objects the usual methods are available.
Logger.log("Date().getMonth(): %s",values[0][0].getMonth());
Logger.log("Date().getYear(): %s",values[0][1].getYear());
// This formats the date as Greenwich Mean Time in the format
// year-month-dateThour-minute-second.
var formattedDate = Utilities.formatDate(values[0][0], "GMT", "yyyy-MM-dd'T'HH:mm:ss'Z'");
Logger.log(formattedDate);
formattedDate = Utilities.formatDate(values[0][1], "GMT", "yyyy-MM-dd'T'HH:mm:ss'Z'");
Logger.log(formattedDate);
}
Logger.log output:
[19-01-31 11:43:17:635 GMT] Values: [[Thu Jan 31 09:00:00 GMT+00:00 2019, Wed Jan 02 09:00:00 GMT+00:00 2019]]
[19-01-31 11:43:17:636 GMT] Date().getMonth(): 0.0
[19-01-31 11:43:17:637 GMT] Date().getYear(): 2019.0
[19-01-31 11:43:17:638 GMT] 2019-01-31T09:00:00Z
[19-01-31 11:43:17:638 GMT] 2019-01-02T09:00:00Z
I wish to format the current date as "yyyy/MM/dd HH:mm". formatDate allows me to format the date but I need to specify a timezone, per the following:
Utilities.formatDate(new Date(), "GMT", "yyyy/MM/dd HH:mm")
This is almost what I want, but I want the result to be BST or GMT, as appropriate to the calendar. What timezone should I specify to get the answer without further manipulation of the text date? (I do have a clumsy workaround). The problem caught me out with a timed batched process that runs at midnight upon our recent BST clock change where timestamps of 23:00 the previous day were applied.
In your script properties, make sure you've selected the correct time zone. Then, in your script, instead of using "GMT", use Session.getScriptTimeZone().
function test() {
var date1 = new Date("March 10, 2018 10:00");
var date2 = new Date("March 11, 2018 10:00");
Logger.log(date1); // Sat Mar 10 10:00:00 GMT-05:00 2018
Logger.log(Utilities.formatDate(date1, Session.getScriptTimeZone(), "yyyy/MM/dd HH:mm")); // 2018/03/10 10:00
Logger.log(date2); // Sun Mar 11 10:00:00 GMT-04:00 2018
Logger.log(Utilities.formatDate(date2, Session.getScriptTimeZone(), "yyyy/MM/dd HH:mm")); // 2018/03/11 10:00
}
In the script editor, go to File > Project properties and set your time zone. (I used Eastern time for mine.)
I have a date stored as follows:
existingDate = Wed Apr 30 2014 00:00:00 GMT+0100 (BST)
When I use Utilities.formatDate to format the date the date is changed to the day before.
var formattedDate = Utilities.formatDate(new Date(existingDate), "GMT", "dd/MM/yyyy");
the formatted date is then set to 29/04/2014 and not 30/04/2014.
Has anyone else seen this behavior.
Utilities.formatDate seems to be working fine.
You have Apr 30 midnight in GMT-1 but then you tell it to format this date in a different timezone GMT, or more explicity GMT-0. The expected result is indeed Apr 29 23h.
The second parameter in Utilities.formatDate must be the timezone you desire.
There are some other issues with this google script.
If I want the timezone AEST but only have the date in the form of 1/27/2017, then Utilities.formatDate("1/27/2017","AEST","yyyy-MM-dd") will return a value of 2017-01-26, a day before it should be! However if I have a time with it such as "1/27/2017 12:00:00" then it returns the correct day. However if I use "GMT+11" as the timezone even without a time on the end if returns the correct day.
Example -
Logger.log('Example 1='+Utilities.formatDate(new Date("1/27/2017"),"AEST","yyyy-MM-dd"));
Logger.log('Example 2='+Utilities.formatDate(new Date("1/27/2017 12:00:00"),"AEST","yyyy-MM-dd"));
Logger.log('Example 3='+Utilities.formatDate(new Date("1/27/2017"),"GMT+11","yyyy-MM-dd"));
Output -
Example 1=2017-01-26
Example 2=2017-01-27
Example 3=2017-01-27
A workaround when reading a date from a spreadsheet cell that does not have a time is to add a time and when using a timezone like "AEST" or similar, something like this -
var displayDate = sheet.getRange(a1Notation).getDisplayValue(); // Not .getValue();
Logger.log(Utilities.formatDate(new Date(displayDate + " 12:00:00"), "AEST", "yyyy-MM-dd"));
However "GMT+11" would be easier because you don't need to add a time.