기본 콘텐츠로 건너뛰기

[지메일 팁 1] 지메일에서 예약전송하는 방법

지메일이 좋은 이유는, 지메일에서 제공하는 기본적인 기능외에도 사용자들이 필요로하는 유용한 기능들을 직접 추가할 수 있는 Open API 들을 제공하고 있다는 것입니다. 3rd party 플러그인 모듈들도 다 이런 API 를 이용하여 개발한 것으로 보입니다.

지메일에서 예약 전송을 이용하는 방법에는 두가지가 있습니다.

목차




Google 스크립트와 Google 스프레드시트를 이용한 나만의 예약 전송 기능을 만들어 활용하는 방법


구글 문서도구 스프레드시트의 스크립트 기능을 활용하면 다양하고 유용한 유틸리티를 만들 수 있습니다. 이 문서에서 소개하는 내용은 간단하게 구글문서 도구에서 내 스프레드시트 (예, My Gmail Scheduler) 를 작성하여 공개된 구글 스크립트를 삽입하여 내 지메일을 예약 전송하는 방법에 대해서 소개하고자 합니다.

아래 스프레드시트(My Gmail Scheduler)에서 보는 것과 같이, 지메일에서 작성중인 이메일 (초안 상태)들에 대해서 원하는 시간에 예약 일정을 설정하여 ‘Gmail Scheduler’ 스크립트를 수행하면 예약전송 OK


예약전송 스크립트 스프레드시트 만드는 순서

1. 스프레드시트 ‘My Gmail Scheduler’ 생성
2. 스프데드시트 -> ‘도구’ -> 스크립트 편집기 수행


3. 아래 스크립트 코드 입력

/* This Google Script was written by Amit Agarwal */

/* Web: http://www.labnol.org/?p=24867            */

/* Apps Script Development: http://ctrlq.org      */

/* Please retain this message in your copy        */

function onOpen() {
 var ss = SpreadsheetApp.getActiveSpreadsheet();
 var menu = [
   {name: "Step 1: Authorize", functionName: "authorize_"},
   {name: "Step 2: Fetch Messages", functionName: "initialize_"},
   {name: "Step 3: Schedule Messages", functionName: "setSchedule_"},
   {name: "Cancel Pending Jobs", functionName: "cancelJobs_"}
 ];  
 ss.addMenu("Gmail Scheduler", menu);
}

function authorize_() {  
 deleteTriggers_();
 SpreadsheetApp.getActive().toast("Authorization successful. Please choose Step 2 to fetch the Gmail drafts into the spreadsheet.");
}

function fetchDraftMessages_() {
 var sheet = SpreadsheetApp.getActiveSpreadsheet();
 var drafts = GmailApp.getDraftMessages();
 if (drafts.length > 0) {
   var rows = [];    
   for (var i=0; i
     if (drafts[i].getTo() !== "") {
       rows.push([drafts[i].getId(), drafts[i].getTo(), drafts[i].getSubject(), "", ""]);
     }
   }
   sheet.getActiveSheet().getRange(2,1,rows.length,5).setValues(rows);
   sheet.toast("Excellent. Now please enter the date and time when you would like these messages to be delivered.");
 } else {
   sheet.toast("We could not find any messages in your Gmail drafts folder.");
 }
}

function cancelJobs_() {
 deleteTriggers_();
 var sheet = SpreadsheetApp.getActiveSheet();
 var data  = sheet.getDataRange().getValues();
 for (var row=1; row
   sheet.getRange("E"+(row+1)).setValue("Not Scheduled");
 }
 SpreadsheetApp.getActive().toast("There are no pending messages in queue.");
}

function initialize_() {
 deleteTriggers_();
 clearSheet_();  
 fetchDraftMessages_();
}

function clearSheet_() {
 var sheet = SpreadsheetApp.getActiveSheet();
 sheet.getRange(2, 1, sheet.getLastRow()+1, 5).clearContent();
}

function setSchedule_() {
 deleteTriggers_();
 try {
   var sheet = SpreadsheetApp.getActiveSheet();
   var data  = sheet.getDataRange().getValues();
   var time  = new Date().getTime();
   var code  = [];
   for (var row in data ) {     
     if (row != 0) {
       var schedule = data[row][3];
       if ( schedule !== "" ) {
         if ( schedule.getTime() > time ) {
           ScriptApp.newTrigger("sendMails")
           .timeBased()
           .at(schedule)
           .inTimezone(SpreadsheetApp.getActiveSpreadsheet().getSpreadsheetTimeZone())
           .create();
           code.push("Scheduled");
         } else {
           code.push("Date is in the past");  
         }
       } else {
         code.push("Not Scheduled");
       }
     }
   }
   for (var i=0; i
     sheet.getRange("E" + (i+2)).setValue(code[i]);
   }
   SpreadsheetApp.getActiveSpreadsheet().toast("The mails have been scheduled. You can close this sheet. For help, contact amit@labnol.org", "Success", -1);
 } catch (e) {
   SpreadsheetApp.getActive().toast(e.toString());
 }
}

function deleteTriggers_() {
 var triggers = ScriptApp.getProjectTriggers();
 for (var i=0; i
   if (triggers[i].getHandlerFunction() === "sendMails") {
     ScriptApp.deleteTrigger(triggers[i]);
   }
 }
}


function sendMails() {
 try {
   var sheet = SpreadsheetApp.getActiveSheet();
   var data  = sheet.getDataRange().getValues();
   var time  = new Date().getTime();
   for (var row=1; row
     if (data[row][4] == "Scheduled") {             
       var schedule = data[row][3];
       if ( schedule != "" ) {
         if ( schedule.getTime() <= time ) {
           var status = dispatchDraft_(data[row][0]);
           sheet.getRange("E"+(row+1)).setValue(status);
         }
       }
     }
   }
 } catch (e) {
   SpreadsheetApp.getActive().toast(e.toString());
 }
}

function dispatchDraft_(id) {
 
 try {
   
   var message = GmailApp.getMessageById(id);
   
   if (message) {
     
     var body = message.getBody();
     var raw  = message.getRawContent();
     
     /* Credit - YetAnotherMailMerge */
     
     var regMessageId = new RegExp(id, "g");
     if (body.match(regMessageId) != null) {
       var inlineImages = {};
       var nbrOfImg = body.match(regMessageId).length;
       var imgVars = body.match(/]+>/g);
       var imgToReplace = [];
       if(imgVars != null){
         for (var i = 0; i < imgVars.length; i++) {
           if (imgVars[i].search(regMessageId) != -1) {
             var id = imgVars[i].match(/realattid=([^&]+)&/);
             if (id != null) {
               id = id[1];
               var temp = raw.split(id)[1];
               temp = temp.substr(temp.lastIndexOf('Content-Type'));
               var imgTitle = temp.match(/name="([^"]+)"/);
               var contentType = temp.match(/Content-Type: ([^;]+);/);
               contentType = (contentType != null) ? contentType[1] : "image/jpeg";
               var b64c1 = raw.lastIndexOf(id) + id.length + 3; // first character in image base64
               var b64cn = raw.substr(b64c1).indexOf("--") - 3; // last character in image base64
               var imgb64 = raw.substring(b64c1, b64c1 + b64cn + 1); // is this fragile or safe enough?
               var imgblob = Utilities.newBlob(Utilities.base64Decode(imgb64), contentType, id); // decode and blob
               if (imgTitle != null) imgToReplace.push([imgTitle[1], imgVars[i], id, imgblob]);
             }
           }
         }
       }
       
       for (var i = 0; i < imgToReplace.length; i++) {
         inlineImages[imgToReplace[i][2]] = imgToReplace[i][3];
         var newImg = imgToReplace[i][1].replace(/src="[^\"]+\"/, "src=\"cid:" + imgToReplace[i][2] + "\"");
         body = body.replace(imgToReplace[i][1], newImg);
       }
     }
     
     var options = {
       cc          : message.getCc(),
       bcc         : message.getBcc(),
       htmlBody    : body,
       replyTo     : message.getReplyTo(),
       inlineImages: inlineImages,
       attachments : message.getAttachments()
     }
     
     GmailApp.sendEmail(message.getTo(), message.getSubject(), body, options);
     message.moveToTrash();
     return "Delivered";
   } else {
     return "Message not found in Drafts";
   }
 } catch (e) {
   return e.toString();
 }
}

4. 스크립트 저장 및 실행
  1). 반드시 위 스크립트 코드를 입력한 후 저장 버튼을 클릭
  2). 저장 완료후, 실행->onOpen 함수를 수행  


5. 스크립트 onOpen 이 정상 수행이 되면, ‘내 지메일 예약 전송' 스프레드시트 메뉴에 ‘Gmail Scheduler’ 가 새롭게 생성이 됩니다.


6. 지메일에서 예약 전송하기 위한 메일 메시지들을 작성 후 전송하지 않고 초안(Draft) 상태 (임시보관함에 저장)로 저장하기



7. 스프레드시트에서 예약 전송 메시지 가져오기
  - 메뉴 -> Gmail Scheduler-> Step2: Fetch Messages 수행

8. 지메일에 예약전송 대상이 되는 메시지의 header 정보가 표시가 되고, 시트의 ‘D’ 컬럼에서 예약 전송하고자 하는 메시지를 선택 원하는 예약 일자를 입력 (예, 2014-08-01 08:00:00) 반드시 날짜 (YYYY-MM-DD) 와 시간 (HH:MM:SS) 을 입력해야 함


8번 과정을 통해서 지메일 예약이 정상 설정이 되면 스프레드시트는 닫기를 한 후, 나중에 다시 오픈하여 ‘E’ 컬럼을 확인하여 전송 상태 유무를 확인 할 수 있습니다. 정상으로 전송이 되면 ‘E’ 컬럼에 Delivered 표시가 됩니다.

크롬스토어에서 3rd party 앱을 설치하여 사용하는 방법이 있습니다.


크롬스토어에서 무료로 설치할 수 있는 예약전송의 대표적인 앱  Boomerang 을 크롬브라우저에서 설치하시면 됩니다.

크롬 브라우저에서 지메일 로그인 후, Boomerang 앱이 정상 설치 되면 아래 이미지에서와 같이 지메일 상단 오른쪽에 Boomerang 아이콘이 보여야 합니다.


편지쓰기시 하단에 ‘Send Later’ 라는 버튼이 보입니다. 이 버튼을 클릭하여 예약을 원하는 날짜및 시간을 입력하여 전송 하면 됩니다.

자세한 사항은 구글문서 참조

댓글

이 블로그의 인기 게시물

[알아두면 쓸모 있는 구글 문서 팁] 문서 공유시- 사용자 이름 대신에 익명의 동물이 표시 되는 이유와 동물 종류

구글 드라이브에는 다른 유사 서비스에서는 제공하지 않는 구글 만의 유니크한 기능들이 있다 구글 문서를  불특정 다수에게 전체 공개로 공유할 수 있습니다. 불특정인이 구글 문서에 접속한 경우 익명의 동물로 표시됩니다.  ' 웹에 공개' 또는 '링크가 있는 사용자' 공유 설정을 선택하면 인식할 수 없는 이름이나 익명의 동물이 표시될 수 있습니다. 파일에서 인식할 수 없는 이름을 볼 수 있는 몇 가지 이유는 다음과 같습니다. 메일링 리스트와 파일을 공유합니다. Google 계정이 없는 사용자와 파일을 공유하며, 그 사용자가 다른 사용자에게 공유 초대를 전달했습니다. 내 파일을 수정할 수 있는 누군가가 파일을 다른 사용자와 공유했습니다. 다른 사용자가 자신의 Google 계정 이름을 변경했습니다. 공유 설정 페이지에서 해당 사용자 이름 위로 마우스를 이동하여 이메일 주소를 확인하세요. 익명의 동물 다른 사용자에게 개별적으로 보기 또는 수정 권한을 부여하거나 메일링 리스트에 속해 있는 경우에만 사용자 이름이 표시됩니다. 파일 권한을 '링크가 있는 사용자'로 설정하면 파일을 보고 있는 사용자의 이름이 표시되지 않습니다. 대신 다른 사용자가 익명으로 라벨이 지정되어 표시되고 각 익명 사용자는 다양한 익명의 동물로 나열됩니다. 파일 권한을 '링크가 있는 사용자'로 설정했지만 특정 사용자와 파일을 공유하는 경우 파일을 공유한 사용자의 이름이 표시됩니다. 그 외 다른 사용자가 파일을 볼 때는 익명으로 나타납니다. 비공개 파일의 익명 동물 파일 권한을 '링크가 있는 사용자'로 설정한 다음 이를 '특정 사용자'로 변경하면 다음과 같은 경우 여러 익명의 동물이 표시될 수 있습니다. 누군가 파일을 여러 번 여는 경우에는 익명의 동물 목록에서 오래되고 연결이 끊긴 세션을 강제 종료하는 데 조금 시간이 걸릴 수 있습니다. 누군가 온...

[팁] Google Slide 프리젠테이션시 모든 한글폰트가 '굴림체' 로 바뀌는 현상을 해결한 크롬 확장 프로그램 소개

구글 문서도구인 구글 슬라이드를 이용하여 프리젠테이션을 많이 하는 분들을 위한 희소식 현재 구글 슬라이드에서는 슬라이드 편집시 사용한 고유 한글 폰트들은 프리젠테이션 모드로 전환할 경우는 모두 '굴림체' 로 바뀌어 표시가 되는 불편함이 있었습니다. 예). 슬라이드 편집에서 사용한 '궁서체' 한글 폰트는, 프리젠테이션 모드에서는 '굴림체'로 바뀌어 디스플레이됨 예). 슬라이드 편집 모드 - '궁서체' 폰트 사용 프리젠테이션 모드에서 '굴림체' 로 변경됨    따라서, 이러한 현상을 해결하는 크롬 확장 프로그램이 개발 되었습니다.  크롬 확장 프로그램 명 - ShowAsis 입니다. 크 롬 웹스토어 링크 -  https://goo.gl/PVPkZz 이 확장 프로그램을 사용하여 슬라이드 프리젠테이션을 하면, 편집 모드의 폰트 그대로 프리젠테이션시에도 그대로 한글 폰트로 디스플레이 됩니다. 단, 단점은 슬라이드가 애니메이션 슬라이드가 있는 경우는 애니메이션이 동작하지 않습니다. ----------------------- G Suite/Google Apps 전문 블로그 -  charlychoi.blogspot.kr 도서 '기업과 학교를 위한 구글크롬북'

[ChatGPT 모델 옵션 완벽 분석: 나에게 맞는 최적의 모델은?]

  ChatGPT 모델 옵션 완벽 분석: 나에게 맞는 최적의 모델은? GPT-4, GPT-4o, o3, o4-mini... 헷갈리는 모델들, 속 시원히 정리해 드립니다! ChatGPT를 사용하다 보면 다양한 모델 옵션 때문에 어떤 것을 선택해야 할지 고민될 때가 많습니다. 특히 글쓰기, 창의적인 작업, 일상 업무 등 다양한 용도로 AI를 활용하는 경우, 각 모델의 특성을 이해하고 상황에 맞게 선택하는 것이 중요합니다. 이 글에서는 ChatGPT에서 제공하는 주요 모델들의 의미와 특징, 장단점을 자세히 살펴보고, 여러분의 필요에 맞는 최적의 모델을 선택할 수 있도록 가이드를 제공합니다. 모델별 개요와 특징 ChatGPT에 표시되는 여러 모델은 크게  GPT 계열 (대규모 언어 모델 기반)과  o-시리즈 (강화학습으로 추론 능력을 향상시킨 모델)로 나눌 수 있습니다. 각 모델의 상세 내용을 살펴보겠습니다. 1. GPT-4 (기존 GPT-4 모델) 의미 및 특징 2023년 초 출시된 GPT-4는 GPT-3.5 대비 월등한 성능을 보인 OpenAI의 주력 언어 모델입니다. 뛰어난 텍스트 생성 능력과 이미지 입력 처리(멀티모달) 기능을 처음 도입했으며, 출시 당시 여러 시험에서 상위 10% 성적을 기록할 만큼 높은 지능 수준을 입증했습니다. 복잡한 문제 해결과 자연스러운 창작에 강점이 있었습니다. 장점 높은 출력 품질 : 복잡한 질문에도 깊이 있는 답변 생성 풍부한 표현력 : 창의적인 아이디어 제시에 유리 멀티모달 : 이미지 이해 및 설명 가능 (초기 버전) 단점 느린 응답 속도 높은 비용 및 자원 소모 : 사용량 제한 존재 지원 종료 예정 : 최신 모델로 대체 진행 중 2. GPT-4o (GPT-4 Omni 모델) 의미 및 특징 GPT-4를 개선한  옴니(Omni) 모델 로, 텍스트, 이미지, 오디오 등 모든 유형의 입력/출력을 단일 모델로 처리합니다. 이름의 'o'는 "모든 것(omni)"을 의미하며, GPT-4 대비 성능 향상 및...

Claude Max 요금제, 과연 월 $100의 가치가 있을까? Pro 사용자들의 불만과 Research 기능의 실체

  Claude Max 요금제, 과연 월 $100의 가치가 있을까? Pro 사용자들의 불만과 Research 기능의 실체 클로드가 한국 사용자들에게도 Research 서비스를 오픈했군요, Max (월 100달러) 가입을 유도하는 문구가 클로드 사이트 접속시 유난하게 보입니다.  최근 Anthropic의 Claude가 새로운 'Max' 요금제(월 $100)를 출시하며 기존 Pro 사용자들 사이에서 논란이 일고 있습니다. 특히, Max 요금제의 핵심 기능으로 내세우는 'Research' 기능의 실효성과 더불어, 기존 Pro 요금제(월 $20)의 사용 환경이 이전보다 악화되었다는 불만이 터져 나오고 있습니다   ( https://shorturl.at/RAlpf  ).  이는 Anthropic이 수익 극대화를 위해 의도적으로 Pro 사용자들을 Max 요금제로 유도하려는 전략이 아니냐는 의구심마저 낳고 있습니다. 점점 더 팍팍해지는 Claude Pro: 의도된 불편함인가? 가장 큰 불만은 Claude Max 출시 이후 기존 Pro 요금제의 사용 경험이 눈에 띄게 저하되었다는 점입니다. 사용자들은 이전보다 ▲더 잦은 사용량 제한 ▲짧아진 세션 유지 시간 ▲엄격해진 토큰 크기 제한 등을 체감하고 있다고 입을 모읍니다. 과거 월 $20로 누렸던 비교적 자유로운 사용 환경과 성능을 기대하기 어려워졌다는 것입니다.  이는 마치 Max 요금제로의 업그레이드를 강요하는 듯한 인상을 줍니다. Pro 요금제는 '일상적인 생산성'을 위한 옵션으로 남겨두고, 조금 더 깊이 있는 작업이나 활용을 원하는 사용자는 울며 겨자 먹기로 월 $100짜리 Max 요금제를 선택하도록 유도하는 전략으로 비칠 수 있습니다. 월 $100의 가치? Claude Max 'Research' 기능 살펴보기 Claude Max의 핵심 기능인 'Research'는 방대한 정보를 빠르게 분석하고 요약해 주는, 소위 '딥 리서치' 기능입...

Gemini 2.5 Pro, '책 쓰기'의 새로운 지평을 열다: 『AI Deep Research 완전 정복』 탄생 비화

  Gemini 2.5 Pro, '책 쓰기'의 새로운 지평을 열다: 『AI Deep Research 완전 정복』 탄생 비화 AI 시대를 맞아 많은 작가들이 생성형 AI를 글쓰기 파트너로 활용하려 시도합니다. 하지만 솔직히 말해, 짧은 글이나 아이디어 구상을 넘어  '책 한 권' 을 쓰는 여정에서 AI와 성공적으로 협업하기란 쉽지 않은 과제였습니다. 가장 큰 어려움은 바로  '맥락 유지' 였죠. 이야기가 길어질수록 AI가 앞서 나눈 대화나 설정된 스토리를 잊어버리고 엉뚱한 방향으로 가기 일쑤였으니까요. 그런데 이번에 『AI Deep Research 완전 정복』 을 집필하면서, 저는  Gemini 2.5 Pro 와 함께 이 한계를 뛰어넘는 놀라운 경험을 했습니다. 단순히 'AI의 도움을 받았다'는 차원을 넘어,  진정한 '협업'을 통해 책 한 권을 완성 할 수 있었던 그 놀라움과 감탄을 여러분과 나누고 싶습니다. 긴 호흡의 집필, 흔들림 없는 맥락 유지 능력에 감탄하다! 책 쓰기는 마라톤과 같습니다. 수백 페이지에 달하는 긴 여정 동안 일관된 톤과 스토리를 유지하는 것이 핵심이죠. 기존의 생성형 AI들은 이 '긴 호흡'에 약점을 보이는 경우가 많았습니다. 세션이 끊기거나, 대화가 길어지면 이전 맥락을 놓쳐 작가가 끊임없이 방향을 다시 잡아줘야 했죠. 하지만  Gemini 2.5 Pro는 달랐습니다.  책 전체의 구조와 흐름, 등장인물의 설정과 같은 방대한 정보를 놀랍도록  안정적으로 기억하고 유지 했습니다. 마치 지치지 않는 조수처럼, 제가 설정한 큰 그림 안에서 일관성을 유지하며 다음 이야기를 함께 써 내려갔습니다. 덕분에 저는 전체 맥락을 바로잡는 데 에너지를 쏟는 대신, 내용의 깊이와 창의적인 표현에 더 집중할 수 있었습니다. 책 한 권을 쓰는 긴 과정 내내  세션을 유지하며 맥락을 이해하는 능력 , 이것이야말로 Gemini 2.5 Pro가 보여준 첫 번째 놀라움이었습니다....