I want to trigger a google script every 15 min in specific time slots, which are 9:30am - 4:30pm and 9:30pm - 5:30am, Monday to Friday only.
I want the script to send email when it is triggered. I write some codes (searching from google) and it successfully sends email to me, but I don’t know how to trigger the script in specific time slots I mentioned above. It is my first time try to write codes and I don’t know how to do it. I google for it, but my code doesn’t work. So, I come here looking for help. Thank you very much.
It is not feasible (or very convoluted) to set up an Apps Script trigger for specific time slots
Much easier is to set it up as everyMinutes(n) and then implement a conditional statement within the function run on trigger to decide either the email shall be sent or not
Sample (compare it carefully against the conditions you want to implement):
//run only once:
function createTrigger(){
ScriptApp.newTrigger("runOnTrigger")
.timeBased()
.everyMinutes(15)
.create();
}
function runOnTrigger(){
var date = new Date();
var weekDay = date.getDay();
var hours = date.getHours();
var minutes = date.getMinutes();
//stop execution if it's a weekday
if(weekDay == 0 || weekDay == 6){
return;
}
// stop execution if it's between 4:30pm and 9:30pm
if(((hours == 16 && minutes >= 30) || hours >= 17) && ((hours == 21 && minutes <= 30) || hours < 21)){
return;
}
// stop execution if it's between 9:30pm and 5:30 am
if(((hours == 5 && minutes >= 30) || hours > 6) && ((hours == 16 && minutes <= 30) || hours < 16)){
return;
}
// implement here your request (e.g. sending emails) that shall be run if you are in the correc time slot
}
Related
I have a script working fine however, I cannot work out how to add a popup reminder for after an all day event as started.... example:
All day even is midnight to midnight, I want a notification at 9am that day.
my script uses event.addPopupReminder but this seems like it only works for before the event.
When I manually look, you have the option to add a reminder "On the day at 9am" but how do I code this? using -900 within the code does not work
On the day at 9am
(I've snipped the code)
By default, if the cell is blank, it'll add a notification at 9am the day before, which I want to change to 9am on the day
if (C2 == "") {C2 = '900'}
event.addPopupReminder(C2)
On the spreadsheet, I've entered -900 (for 9am on the day) and -1000 (for 10am on the day), neither have worked.
The function you are asking doesn't exist, but there is a way to do it anyway. You just have to calcul the minutes between 9am and the start of your event manually.
You haven't put a lot of code, so I wrote something that probably need to be adapted to your case.
What the following function do :
Get all events of the day and for each :
Define the minutes between 9am and the start time
Add a reminder
function setReminderTodayEvents() {
var today = new Date();
today.setHours(9);
today.setMinutes(0);
today.setSeconds(0);
var events = CalendarApp.getDefaultCalendar()
.getEventsForDay(today);
for (var x = 0; x < events.length; x++) {
var date_start = events[x].getStartTime();
var delta = (date_start - today) / 1000 / 60; //ms to minute
events[x].addPopupReminder(delta);
}
}
Suggestion : add a trigger to execute that function every day between 7 and 8
References:
getEventsForDay()
getStartTime()
addPopupReminder()
I'm trying to create a schedule heat map so that we can adjust staffing times and days. In order to make this as easy as possible, I've come up with a tabular structure that allows the user to input the employee's name, then select their shift start time and their shift end time from a drop-down, and then use checkboxes to indicate which days they will work, as shown:
The end result would be a heat map that counts the number of instances that a value exists in the range between the start time and end time, broken down by hour and by day. My original thought was to use COUNTIFS thusly: =COUNTIFS(Calculations!D:D, ">=9:00:00", Sheet9!D:D, "=TRUE") Where Calculations!D:D is the column of the selected Start Time, where ">=9:00:00" checks to see if the start time is greater than or equal to 9AM, and where Sheet9!D:D, "=TRUE" checks to see if the checkbox for that day is checked. So this example would check to see if someone is working at 9AM on Monday.
However, this didn't pan out since we're checking for any value greater than 9AM, and most employees won't be working more than 10 hours, so I'm getting false positives.
My next thought was to use a named range that would start at the Start Time value and then, if necessary, loop back through to the End Time (for example, if an employee started at 10PM and their shift ended at 7AM). Since this range would be dynamic (not all employees will work strictly 8 hours per day), I would need to check to see if a value exists within the range, however, I'm not sure how to A: Loop through or B: check to see if a value is in the dynamic range. I assume this will require Google Apps Script to pull off, but I'm not well-versed in it, and I've been beating my head against a wall trying to figure this out. Any help would be appreciated!
Oh, and here's a screenshot of the desired output, with a couple of values filled in:
I don't know if this is exactly what you wanted but I wanted to practice using times is Google Scripts so I had a crack at solving your problem. However, there are a couple of little bits I didn't have time to write that I've commented in the script.
If I've understood your problem correctly, you want to take a series of shifts life this:
And convert it into a heatmap like this:
(I've not done all your formatting, but I'm guessing you want to see how many people are available for every hour of the day)
In my code, I called the first sheet above "Roster" and the second sheet "RosterOverview" - note the capitalisation and lack of spaces.
Approach
I wrote two scripts. One (rostering) to check which days a staff member worked and a second to see the hours they worked and update the values in RosterOverview (updateCalendar).
For each member of staff, I took their working hours and working days. I checked to see if they worked on a certain day and, if they did, sent their working hours to a script called updateCalendar. This second script then looked to see if they worked a night shift i.e. over midnight or not. The script then adds a 1 to each hour window a staff member is rostered for.
The scripts
function rostering() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var rosterSheet = ss.getSheetByName("Roster");
var overviewSheet = ss.getSheetByName("RosterOverview");
var range = rosterSheet.getRange(2,2,4,9); //You will need to work out how to accurately get this range for your data - 2 and 2 are the starting row and column respectively i.e. B2; 4 indicates the number of rows; 9 indicates the number of columns
var times = range.getValues();
var calendarRange = overviewSheet.getRange("B2:I25"); //On RosterOverview, this is the range of cells for Monday-Monday, midnight-2300. You need 8 columns of data because the extra one is for Sunday night overflowing into Monday morning.
//You will need to write a bit of code yourself to make sure all the values in calendarRange are set to 0. I have set this manually and pull them below when I was testing.
var calendarValues = calendarRange.getValues();
//This section loops through each staff member's hours and days.
//It looks to see if the check box is ticked for each day of the week in turn.
//If a checkbox is ticked, it calls a second script: updateCalendar
//It's probably possible to write a separate loop to go through the days of the week
for (var i = 0; i < range.getHeight(); i++){
//if Monday shift
if(times[i][2] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 0)
}
//if Tuesday shift
if(times[i][3] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 1)
}
//if Wednesday shift
if(times[i][4] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 2)
}
//if Thursday shift
if(times[i][5] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 3)
}
//if Friday shift
if(times[i][6] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 4)
}
//if Saturday shift
if(times[i][7] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 5)
}
//if Sunday shift
if(times[i][8] == true){
calendarValues = updateCalendar(times[i][0], times[i][1], calendarValues, 6)
}
}
//Once ithas looped through al the staff member's shifts, it updates the RosterOverview
//sheet with how many people are on each hour
calendarRange.setValues(calendarValues);
}
function updateCalendar(startTime, endTime, roster, day){ //Let day be a digit: 0 for Monday, 1 for Tuesday.... Roster is the calendarValues array from the roster function
if(startTime > endTime){ //night shift over midnight
for (var j = startTime.getHours(); j < 24; j++){
roster[j][day] = roster[j][day] + 1;
}
for (var j = 0; j < endTime.getHours(); j++){
roster[j][day + 1] = roster[j][day + 1] + 1;
}
} else {
for (var j = startTime.getHours(); j < endTime.getHours(); j++){
roster[j][day] = roster[j][day] + 1;
}
}
return roster;
}
What you need to do
It's probably best to add a custom menu so you can run it from the sheet: see this Fundamentals codelab to learn how: https://developers.google.com/codelabs/apps-script-fundamentals-3#0
You need to make sure your Google Sheet and the Google script are set to the same timezone. To set the timezone for the script, from the script editor you want to click on [Use legacy editor] in the top right hand corner then go to File > Project properties. Once set, return to the new editor.
You will need to write a little bit of code that selects the range of data for your staff members' shifts and a bit of code that resets the RosterOverview spreadsheet values to 0s before you run the script again.
Let me know if you have any problems.
I currently have a script in place to auto reply to messages sent over the weekend, checking for unread messages in the last 5 minutes, and then replying and marking as read.
I was wondering if it were possible to only have this script look for unread messages INSIDE a specific folder/filter. The reason for this is we have lots of different emails going to this box, and we don't want auto replies to most emails, just some, which have been filtered.
If it is not possible to have a script look inside a filter, is it possible to only have to reply to messages that match a certain condition?
This is the current script I have in place, running every five minutes:
function autoReply() {
var interval = 5; // if the script runs every 5 minutes; change otherwise
var date = new Date();
var day = date.getDay();
var hour = date.getHours();
if ([6,0].indexOf(day) > -1 || (day == 1 && hour < 10) || (day == 5 && hour >= 15)) {
var timeFrom = Math.floor(date.valueOf()/1000) - 60 * interval;
var threads = GmailApp.search('is:inbox after:' + timeFrom);
for (var i = 0; i < threads.length; i++) {
if (threads[i].isUnread()){
threads[i].reply("Thank you for reaching CTGINC. Your email is very important to us, one of our customer service experts will respond within the next 24-48 hours. Thank you!");
threads[i].markRead();
threads[i].markImportant();
}
}
}
}
Requirement:
Find unread emails inside a label.
Solution:
Enhance your current GmailApp.search() query to include this criteria.
Example:
//replace "labelname" with the name of your label
var threads = GmailApp.search('in:labelname is:unread after:' + timeFrom);
Explanation:
As you can see in the query, it's only looking for threads inside that specific label that are unread and recieved before your var timeFrom.
GmailApp.search() uses the same search operators as the search bar at the top of your Gmail. It's worth reading up on them here, there are loads of different possibilities.
Cat, have you played with having your filters apply labels and then using getLabels()?
When I was searching for an answer to this, I only encountered developers looking for ways to update their apps' live tiles more and more frequently. I want the opposite, sort of.
See I'm developing a weather app, and I want it to update every hour but for a specific hourly range only. That is, I don't want the user to have the ability to update the tile once every hour because 1) people sleep and 2) the API I'm using is free only for the first 1,000 calls per day. In other words, users don't need to it update every hour and I can't afford to give them the option to anyway.
So is it possible to get, for example, the live tile to update every hour from 8am to 11pm, and to not make any calls from 12pm till 7am?
If you make the call to the API in your ScheduledAgent, simply wrap the call in an if block that checks the time. I had a similar need to update the tile once a day (it was counting down the days until Xmas).
This code is in my ScheduledAgent.cs file. It checks the date (should only trigger in December and before the 26th) and sets the countdown, then sends a toast notification only on Xmas morning. It should be a good example of how to restrict API calls to a set time of dat in your background task.
if (DateTime.Now.Month == 12 && DateTime.Now.Day < 26)
{
//number of days until the 25th
var countdown = ((new DateTime(DateTime.Now.Year, 12, 25).DayOfYear) - DateTime.Now.DayOfYear);
if (secondaryTile != null)
{
var imageString = "/Images/Tiles/" + countdown + ".png";
var newTileData = new StandardTileData
{
BackgroundImage = new Uri(imageString, UriKind.Relative)
};
secondaryTile.Update(newTileData);
}
var now = DateTime.Now;
if (now.Day == 25 && now.TimeOfDay.Hours == 9 && (now.TimeOfDay.Minutes > 14 && now.TimeOfDay.Minutes < 46))
{
var toast = new ShellToast { Title = "Xmas Countdown", Content = "Merry Xmas! Thank you for using 'Quick Xmas List' and have a safe holiday!" };
toast.Show();
}
}
I'm trying to create a trigger that activates on weekdays only and at a specific time, but I don't know what I'm doing wrong. Here is my code.
function createTriggers() {
var days = [ScriptApp.WeekDay.MONDAY, ScriptApp.WeekDay.TUESDAY,
ScriptApp.WeekDay.WEDNESDAY, ScriptApp.WeekDay.THURSDAY,
ScriptApp.WeekDay.FRIDAY];
var d = new Date();
var time = d.toLocaleTimeString();
if (time == '3:05:00 PM EDT') {
for (var i = 0; i < days.length; i++) {
ScriptApp.newTrigger(Lunch1)
.timeBased().onWeekDay(days[i])
.everyMinutes().create();
}
}
}
I'm guessing that you mean you want to make it so a function is triggered (runs) every weekday at a certain time of the day. Here's what I would do.
First, in the Script Editor:
Go to Resources - Current Project's Triggers
Click Add a new trigger
Select the function that you want to be triggered
Change the next box to "Time-driven"
Change the next box to "Hour timer"
Then change the last box to "every hour"
Then at the very beginning of your function, add this code:
var d = new Date();
if (d.getDay() == 6 || d.getDay() == 0) return;
// more info here: http://www.w3schools.com/jsref/jsref_getday.asp
That will stop the rest of the script from running if it's a Saturday or Sunday.
Then, say you want the script to only run at one specific time each weekday, you could set the time-driven trigger to "Every minute" and then add another if statement after the one above:
if (d.getHours() != 15 && d.getMinutes() != 5) return;
That will stop the rest of the script from running if it's not exactly 15:05 (3:05 PM).
This is definitely not the most efficient way to do this, but it works.
Also, this is worth mentioning: I'm not sure if there's anything that would prevent you from triggering the script every minute. Google does impose quotas for what your scripts can do, but I didn't see anything about how many times a script can be triggered in a day. Here's the chart: https://developers.google.com/apps-script/guides/services/quotas
You should do something like this:
function createTriggers(func, hour) {
var weekdays = [ScriptApp.WeekDay.MONDAY, ScriptApp.WeekDay.TUESDAY,
ScriptApp.WeekDay.WEDNESDAY, ScriptApp.WeekDay.THURSDAY,
ScriptApp.WeekDay.FRIDAY];
for (var dayIndex in weekdays) {
var day = weekdays[dayIndex];
ScriptApp.newTrigger(func).timeBased()
.onWeekDay(day).atHour(hour).create();
}
}
This code creates a trigger for weekdays in a specific time calling the func function.