3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

【営業向け】インサイドセールスでお客様とアポイントが取れたとき、社内で空いている会議室を自動で確保する

3
Posted at

やりたいこと&その理由

以前に以下の記事を書きました。

【営業向け】インサイドセールスでアポ取得できたときのお客様へのメールを自動生成する

WEB面談のアポイントが取れたときに社内の会議室を探すということをしておりましたが、お客様へのメールを生成するのと同時に、社内の会議室を確保できないかと思い、自動化いたしました。これで、会議室を探す時間を短縮し、さらに次のお客様へお電話することができます。

以前に書いたコードを少しリファクタリングしながら、新しくメソッドを追加しております。

実装方法

1: 自分のGoogleカレンダーを取得
2: 現在から2か月先の予定を配列で取得
3: 2の配列の中で、一番最近に作られた予定を取得
4: 社内の会議室の予定を取得
5: 2で取得した予定と、4で取得した予定を付け合わせる
6: もし、会議室が空いているようであれば、空いている会議室を確保する

完成系

コード.js
const CALENDAR_ID = 'カレンダーID';

//アポイントの予定をカレンダーから取得し、もし会議室がすでにセットされてあれば、
//メールの下書きを作成するのを止める
function judgeIfFunctionTrigger(){
  const calendar  = CalendarApp.getCalendarById(CALENDAR_ID);
  const startTime = new Date();
  const endTime   = new Date();
  endTime.setMonth(endTime.getMonth() + 2);
  var newestEvent = getNewestEvent(calendar, startTime, endTime);
  var guestArr = newestEvent.getGuestList();
  var count    = 0;
  for(let i = 0; i < guestArr.length; i++){
    if(guestArr[i].getEmail().match(/@resource.calendar.google.com/)){
      count += 1;
    }
  }
  if(count == 0){
    createEmailDraft(newestEvent);
  }
}

//お客様向けのメールを生成するメソッド。今回のメインになるメソッド。
function createEmailDraft(newestEvent) {
  if(newestEvent.getTitle().match(/WEBお打合せ/)){
    var checkIfMtgRoomIsEmpty = checkMeetingRoom(newestEvent);
    if(checkIfMtgRoomIsEmpty.boolean){
      bookMeetingRoom(newestEvent,checkIfMtgRoomIsEmpty.mtgEmail);
    }
  }
  
  var emailInformation = getInformationForEmail(newestEvent);
  var template_index   = HtmlService.createTemplateFromFile('body');
 template_index.arr = [emailInformation.companyName,
                       emailInformation.name,
                       emailInformation.mtgMonth + 1,
                       emailInformation.mtgDate,
                       emailInformation.mtgDay,
                       emailInformation.mtgHour,
                       emailInformation.mtgMinutes,
                       emailInformation.mtgUrl];

  var html             = template_index.evaluate().getContent();
  
  if(newestEvent.getTitle().match(/WEBお打合せ/) || newestEvent.getTitle().match(/往訪/) && newestEvent.isOwnedByMe() && newestEvent.getTitle().match(/確定/)){
    GmailApp.createDraft(emailInformation.toEmail, emailInformation.subject, html,{cc: emailInformation.ccEmail, htmlBody: html});
  }
}

//最新の予定(つまりアポイント)を取得するメソッド
function getNewestEvent(calendar, startTime, endTime){
  var events    = calendar.getEvents(startTime, endTime);
  var arrUpDate = [];
  for (var i = 0;i<events.length;i++){
    arrUpDate.push(events[i].getLastUpdated().getTime());
  }
  
  var maxDay = Math.max.apply(null,arrUpDate);//更新日時の最大値の取得
  var index  = arrUpDate.indexOf(maxDay);//インデックスの取得
  return events[index];
}


//社内で空いている会議室を探して、仮に会議室が空いていたらTrueを返している
function checkMeetingRoom(newestEvent){
  const newEventStartTime = newestEvent.getStartTime();
  const newEventEndTime   = newestEvent.getEndTime();
  var mtgRoom1 = "mtg1@resource.calendar.google.com";
  var mtgRoom2 = "mtg2@resource.calendar.google.com";
  var mtgRoom3 = "mtg3@resource.calendar.google.com";
  var mtgRoom4 = "mtg4@resource.calendar.google.com";
  var mtgRoom5 = "mtg5@resource.calendar.google.com";
  var mtgRoomArr = [mtgRoom1,mtgRoom2,mtgRoom3,mtgRoom4,mtgRoom5];
  for(let i = 0; i < mtgRoomArr.length; i++){
    var mtgCalendar = CalendarApp.getCalendarById(mtgRoomArr[i]);
    var checkEvent = mtgCalendar.getEvents(newEventStartTime, newEventEndTime);
    if(checkEvent[0] == null || checkEvent[1] == null){
      return {
        "boolean": true,
        "mtgEmail": mtgRoomArr[i]
      };
    }
  }
}

//会議室を予約するメソッド
function bookMeetingRoom(newestEvent,mtgRoomEmail){
  newestEvent.addGuest(mtgRoomEmail);
}

//メールに必要な情報を抜き出すメソッド
function getInformationForEmail(newestEvent){
  var dayOfWeekStr = ["日曜日", "月曜日", "火曜日", "水曜日", "木曜日", "金曜日", "土曜日"];
  
  var titleWithArr    = newestEvent.getTitle().split(" ");
  var companyName     = titleWithArr[1];
  var name            = titleWithArr[2];
  var mtgMonth        = newestEvent.getStartTime().getMonth();
  var mtgDate         = newestEvent.getStartTime().getDate();
  var mtgHour         = newestEvent.getStartTime().getHours();
  var mtgMinutes      = newestEvent.getStartTime().getMinutes();
  if(mtgMinutes == 0){
    mtgMinutes = "00";
  }
  var mtgDay          = dayOfWeekStr[newestEvent.getStartTime().getDay()];
  var toEmail         = newestEvent.getGuestList()[0].getEmail();
  var ccEmail         = 'saasbiz@wealth-park.com';
  var subject         = name + ':お打合せの日程に関しまして/WealthPark奥野';
  var mtgUrl          = newestEvent.getLocation();
  return {
    "companyName": companyName,
    "name": name,
    "mtgMonth": mtgMonth,
    "mtgDate": mtgDate,
    "mtgHour": mtgHour,
    "mtgMinutes": mtgMinutes,
    "mtgDay": mtgDay,
    "toEmail": toEmail,
    "ccEmail": ccEmail,
    "subject": subject,
    "mtgUrl": mtgUrl
  };
}

詰まったポイント

GoogleAppsScriptの仕様ですが、カレンダーが更新されるたびにメソッドがトリガーがされるようになっています。以下のようにカレンダーのゲストに会議室を入れるようにすると、それだけでカレンダーが更新されたと認識して再帰的に何度もメソッドが呼ばれてしまい、無駄なドラフトがたくさん作成されてしまいます。

function bookMeetingRoom(newestEvent,mtgRoomEmail){
  newestEvent.addGuest(mtgRoomEmail);
}

なので今回の実装ですが、createEmailDraftの前に、取得した予定のゲストに会議室が入っていないか判定をして、入っていないようであればcreateEmailDraftをトリガーするようにしています。

function judgeIfFunctionTrigger(){
  const calendar  = CalendarApp.getCalendarById(CALENDAR_ID);
  const startTime = new Date();
  const endTime   = new Date();
  endTime.setMonth(endTime.getMonth() + 2);
  var newestEvent = getNewestEvent(calendar, startTime, endTime);
  var guestArr = newestEvent.getGuestList();
  var count    = 0;
  for(let i = 0; i < guestArr.length; i++){
    if(guestArr[i].getEmail().match(/@resource.calendar.google.com/)){
      count += 1;
    }
  }
  if(count == 0){
    createEmailDraft(newestEvent);
  }
}

なので、今回は直接createEmailDraftを呼ぶのではなくて、judgeIfFunctionTriggerを先に呼ぶようにしています。

image.png

会議室のIDの調べ方

ちなみに、会議室のIDの調べ方は、自分のカレンダーIDを調べる方法とまったく同じです。

image.png

3
3
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?