이호영 이호영 2025-08-18
Merge branch '5102_친구톡_발송속도_개선'
@4656be6275964d88e55670ebd8118241a7e87b9b
 
src/main/java/itn/com/cmm/MjonFTSendVO.java (added)
+++ src/main/java/itn/com/cmm/MjonFTSendVO.java
@@ -0,0 +1,130 @@
+package itn.com.cmm;
+
+import lombok.Getter;
+import lombok.Setter;
+import lombok.ToString;
+
+@Getter
+@Setter
+@ToString
+public class MjonFTSendVO{
+
+	
+	/**
+	 * @description : 수신자번호
+	 */
+	private String  phone;
+
+	/**
+	 * @description : [*이름*] - 치환문자
+	 */
+	private String 	name;
+	
+	/**
+	 * @description : [*1*] - 치환문자
+	 */
+	private String  rep1;
+	
+	/**
+	 * @description : [*2*] - 치환문자
+	 */
+	private String  rep2;
+	
+	/**
+	 * @description : [*3*] - 치환문자
+	 */
+	private String  rep3;
+	
+	/**
+	 * @description : [*4*] - 치환문자
+	 */
+	private String  rep4;
+	
+	/**
+	 * @description : 문자ID
+	 */
+	private String 	msgId;
+
+	/**
+	 * @description : 전송그룹ID (대량문자의 경우 하나의 그룹으로 세팅)
+	 */
+	private String msgGroupId; 
+	
+	/**
+	 * @description : 문자온 일반회원ID
+	 */
+	private String  userId;
+
+	/**
+	 * @description : 발신번호 (하이픈 등의 문자를 제외한 12byte이하의 숫자로 입력한다.)
+	 */
+	private String  callFrom;
+	/**
+	 * @description : 수신번호 (하이픈 등의 문자를 제외한 12byte이하의 숫자로 입력한다.)
+	 */
+	private String  callTo; 
+	/**
+	 * @description : 예약 발송일시
+	 */
+	private String  reqDate;
+	/**
+	 * @description :전송사(04:다우, 05:JJ, 07:IVT, 01:아이하트 , 02:현대퓨처넷, 03:아이엠오) 
+	 */
+	private String  agentCode;
+
+	/**
+	 * @description : MMS용 메시지제목
+	 */
+	private String  subject;
+	
+	/**
+	 * @description : SMS용 메시지본문
+	 */
+	private String  smsTxt;
+
+	/**
+	 * @description : 메세지타입(4: SMS 전송, 5: URL 전송, 6: MMS전송, 7: BARCODE전송, 8: 카카오 알림톡 전송)
+	 */
+	private String  msgType;
+	
+
+	/**
+	 * @description : 첨부파일 갯수
+	 */
+	private String 	fileCnt;
+	
+	/**
+	 * @description : 파일이름1
+	 */
+	private String 	filePath1;
+	
+	/**
+	 * @description : 파일이름2
+	 */
+	private String 	filePath2;
+	
+	/**
+	 * @description : 파일이름3
+	 */
+	private String 	filePath3;
+	
+
+
+	/**
+	 * @description : event 여부 / group tb에 넣는 용도 / 기본값 N 
+	 */
+	private String 	eventYn="N";
+	
+	
+	
+	/**
+	 * @description : 개별단가
+	 */
+	private String 	eachPrice;
+	
+
+	
+	
+	
+	
+}
src/main/java/itn/com/cmm/util/MsgSendUtils.java
--- src/main/java/itn/com/cmm/util/MsgSendUtils.java
+++ src/main/java/itn/com/cmm/util/MsgSendUtils.java
@@ -463,7 +463,7 @@
 		}
 	}
 
-	private static Boolean getReplaceYN(String smsTxtTemplate) {// 여러 치환 구문이 포함된 정규식 패턴
+	public static Boolean getReplaceYN(String smsTxtTemplate) {// 여러 치환 구문이 포함된 정규식 패턴
 
 	    if (smsTxtTemplate == null) {
 	        return false; // null일 경우 false 반환
src/main/java/itn/let/kakao/admin/kakaoAt/service/MjonKakaoATVO.java
--- src/main/java/itn/let/kakao/admin/kakaoAt/service/MjonKakaoATVO.java
+++ src/main/java/itn/let/kakao/admin/kakaoAt/service/MjonKakaoATVO.java
@@ -3,7 +3,11 @@
 import java.util.List;
 
 import itn.com.cmm.ComDefaultVO;
+import lombok.Getter;
+import lombok.Setter;
 
+@Getter
+@Setter
 public class MjonKakaoATVO extends ComDefaultVO{
 	
 	private static final long serialVersionUID = 1L;
@@ -30,6 +34,7 @@
 	private String  msgKind; // '문자 종류 일반:N, 광고:A, 선거:C',
 	private String  msgPayCode; // '재전송 기능에 의한 최종전송콘텐트 종류 저장',
 	private String  contSeq; // COMMENT 'MMS의 콘텐츠 Key(MMS_CONTENTS_INFO의 CONT_SEQ)',
+	private String  contents; // BIZ_ATTACHMENTS -> COMMENT 
 	private String  msgTypeResend; // '재전송할 문자 타입. 값이 있으면 재전송. 없으면 단 건 전송',
 	private String  centerSeqResend; // '재전송할 센터. NPro 내부적으로 사용함.',
 	private String  msgNoticetalkSenderKey; // '카카오 알림톡에 등록된 사용자 고유키',
@@ -154,88 +159,8 @@
 	private String yellowId;
 	private String bizKakaoResendTypeCnt;
 	
-	public String getMsgDiv() {
-		return msgDiv;
-	}
-	public void setMsgDiv(String msgDiv) {
-		this.msgDiv = msgDiv;
-	}
-	public Float getAgentPrice() {
-		return agentPrice;
-	}
-	public void setAgentPrice(Float agentPrice) {
-		this.agentPrice = agentPrice;
-	}
 	private String registPnttm;	//통계등록일시
 	
-	public String getNowDate() {
-		return nowDate;
-	}
-	public void setNowDate(String nowDate) {
-		this.nowDate = nowDate;
-	}
-	public String getTodayYn() {
-		return todayYn;
-	}
-	public void setTodayYn(String todayYn) {
-		this.todayYn = todayYn;
-	}
-	public String getReserveType() {
-		return reserveType;
-	}
-	public void setReserveType(String reserveType) {
-		this.reserveType = reserveType;
-	}
-	public String getAtchFiles() {
-		return atchFiles;
-	}
-	public void setAtchFiles(String atchFiles) {
-		this.atchFiles = atchFiles;
-	}
-	public String getApprovalPnttm() {
-		return approvalPnttm;
-	}
-	public void setApprovalPnttm(String approvalPnttm) {
-		this.approvalPnttm = approvalPnttm;
-	}
-	public String getPayCnt() {
-		return payCnt;
-	}
-	public void setPayCnt(String payCnt) {
-		this.payCnt = payCnt;
-	}
-	public String getPayPct() {
-		return payPct;
-	}
-	public void setPayPct(String payPct) {
-		this.payPct = payPct;
-	}
-	public String getAdminSmsNoticeYn() {
-		return adminSmsNoticeYn;
-	}
-	public void setAdminSmsNoticeYn(String adminSmsNoticeYn) {
-		this.adminSmsNoticeYn = adminSmsNoticeYn;
-	}
-	
-	public String getSearchAdminSmsNoticeYn() {
-		return searchAdminSmsNoticeYn;
-	}
-	public void setSearchAdminSmsNoticeYn(String searchAdminSmsNoticeYn) {
-		this.searchAdminSmsNoticeYn = searchAdminSmsNoticeYn;
-	}
-	
-	public String getMaxRegDate() {
-		return maxRegDate;
-	}
-	public void setMaxRegDate(String maxRegDate) {
-		this.maxRegDate = maxRegDate;
-	}
-	public String getMinRegDate() {
-		return minRegDate;
-	}
-	public void setMinRegDate(String minRegDate) {
-		this.minRegDate = minRegDate;
-	}
 
 
 	private String resultCode;	//문자 결과 코드 정보
@@ -288,1005 +213,12 @@
 	
 	private int totalCallCnt;	//수신자 전체 갯
 
-	public String getUserCallbackYn() {
-		return userCallbackYn;
-	}
-	public void setUserCallbackYn(String userCallbackYn) {
-		this.userCallbackYn = userCallbackYn;
-	}
-                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         
-	public String getCallbackYn() {
-		return callbackYn;
-	}
-	public void setCallbackYn(String callbackYn) {
-		this.callbackYn = callbackYn;
-	}
-
-	public int getCallRejectionCount() {
-		return callRejectionCount;
-	}
-	public void setCallRejectionCount(int callRejectionCount) {
-		this.callRejectionCount = callRejectionCount;
-	}
-	public int getSuccessCount() {
-		return successCount;
-	}
-	public void setSuccessCount(int successCount) {
-		this.successCount = successCount;
-	}
-
-
 	private int regCount;
-	public int getRegCount() {
-		return regCount;
-	}
-	public void setRegCount(int regCount) {
-		this.regCount = regCount;
-	}
-	public int getRegComCount() {
-		return regComCount;
-	}
-	public void setRegComCount(int regComCount) {
-		this.regComCount = regComCount;
-	}
-
-
 	private int regComCount;
-	
-	public String getAddrGrpId() {
-		return addrGrpId;
-	}
-	public void setAddrGrpId(String addrGrpId) {
-		this.addrGrpId = addrGrpId;
-	}
-	public String getAddrGrpNm() {
-		return addrGrpNm;
-	}
-	public void setAddrGrpNm(String addrGrpNm) {
-		this.addrGrpNm = addrGrpNm;
-	}
-
-	
 	private int orderByCode;		// 문자타입 정렬번호
 	private String detailType;
+	private String adFlag;
 	
-	public int getOrderByCode() {
-		return orderByCode;
-	}
-	public void setOrderByCode(int orderByCode) {
-		this.orderByCode = orderByCode;
-	}
-	public String getRegDate() {
-		return regDate;
-	}
-	public void setRegDate(String regDate) {
-		this.regDate = regDate;
-	}
-	public String getMsgId() {
-		return msgId;
-	}
-	public void setMsgId(String msgId) {
-		this.msgId = msgId;
-	}
-	public String getUserId() {
-		return userId;
-	}
-	public void setUserId(String userId) {
-		this.userId = userId;
-	}
-	public String getAgentFlag() {
-		return agentFlag;
-	}
-	public void setAgentFlag(String agentFlag) {
-		this.agentFlag = agentFlag;
-	}
-	public String getUserData() {
-		return userData;
-	}
-	public void setUserData(String userData) {
-		this.userData = userData;
-	}
-	public String getMsgSeq() {
-		return msgSeq;
-	}
-	public void setMsgSeq(String msgSeq) {
-		this.msgSeq = msgSeq;
-	}
-	public String getCurState() {
-		return curState;
-	}
-	public void setCurState(String curState) {
-		this.curState = curState;
-	}
-	public String getSentDate() {
-		return sentDate;
-	}
-	public void setSentDate(String sentDate) {
-		this.sentDate = sentDate;
-	}
-	public String getRsltDate() {
-		return rsltDate;
-	}
-	public void setRsltDate(String rsltDate) {
-		this.rsltDate = rsltDate;
-	}
-	public String getReportDate() {
-		return reportDate;
-	}
-	public void setReportDate(String reportDate) {
-		this.reportDate = reportDate;
-	}
-	public String getReqDate() {
-		return reqDate;
-	}
-	public void setReqDate(String reqDate) {
-		this.reqDate = reqDate;
-	}
-	public String getRsltCode() {
-		return rsltCode;
-	}
-	public void setRsltCode(String rsltCode) {
-		this.rsltCode = rsltCode;
-	}
-	public String getRsltCode2() {
-		return rsltCode2;
-	}
-	public void setRsltCode2(String rsltCode2) {
-		this.rsltCode2 = rsltCode2;
-	}
-	public String getRsltNet() {
-		return rsltNet;
-	}
-	public void setRsltNet(String rsltNet) {
-		this.rsltNet = rsltNet;
-	}
-	public String getCallTo() {
-		return callTo;
-	}
-	public void setCallTo(String callTo) {
-		this.callTo = callTo;
-	}
-	public String getCallFrom() {
-		return callFrom;
-	}
-	public void setCallFrom(String callFrom) {
-		this.callFrom = callFrom;
-	}
-	public String getSmsTxt() {
-		return smsTxt;
-	}
-	public void setSmsTxt(String smsTxt) {
-		this.smsTxt = smsTxt;
-	}
-	public String getSmsTxtArea() {
-		return smsTxtArea;
-	}
-	public void setSmsTxtArea(String smsTxtArea) {
-		this.smsTxtArea = smsTxtArea;
-	}
-	public String getMsgType() {
-		return msgType;
-	}
-	public void setMsgType(String msgType) {
-		this.msgType = msgType;
-	}
-	public String getMsgKind() {
-		return msgKind;
-	}
-	public void setMsgKind(String msgKind) {
-		this.msgKind = msgKind;
-	}
-	public String getMsgPayCode() {
-		return msgPayCode;
-	}
-	public void setMsgPayCode(String msgPayCode) {
-		this.msgPayCode = msgPayCode;
-	}
-	public String getContSeq() {
-		return contSeq;
-	}
-	public void setContSeq(String contSeq) {
-		this.contSeq = contSeq;
-	}
-	public String getMsgTypeResend() {
-		return msgTypeResend;
-	}
-	public void setMsgTypeResend(String msgTypeResend) {
-		this.msgTypeResend = msgTypeResend;
-	}
-	public String getCenterSeqResend() {
-		return centerSeqResend;
-	}
-	public void setCenterSeqResend(String centerSeqResend) {
-		this.centerSeqResend = centerSeqResend;
-	}
-	public String getMsgNoticetalkSenderKey() {
-		return msgNoticetalkSenderKey;
-	}
-	public void setMsgNoticetalkSenderKey(String msgNoticetalkSenderKey) {
-		this.msgNoticetalkSenderKey = msgNoticetalkSenderKey;
-	}
-	public String getMsgNoticetalkTmpKey() {
-		return msgNoticetalkTmpKey;
-	}
-	public void setMsgNoticetalkTmpKey(String msgNoticetalkTmpKey) {
-		this.msgNoticetalkTmpKey = msgNoticetalkTmpKey;
-	}
-	public String getMsgResendCount() {
-		return msgResendCount;
-	}
-	public void setMsgResendCount(String msgResendCount) {
-		this.msgResendCount = msgResendCount;
-	}
-	public String getMsgResenddate() {
-		return msgResenddate;
-	}
-	public void setMsgResenddate(String msgResenddate) {
-		this.msgResenddate = msgResenddate;
-	}
-	public String getSentDatePre() {
-		return sentDatePre;
-	}
-	public void setSentDatePre(String sentDatePre) {
-		this.sentDatePre = sentDatePre;
-	}
-	public String getRsltDatePre() {
-		return rsltDatePre;
-	}
-	public void setRsltDatePre(String rsltDatePre) {
-		this.rsltDatePre = rsltDatePre;
-	}
-	public String getReportDatePre() {
-		return reportDatePre;
-	}
-	public void setReportDatePre(String reportDatePre) {
-		this.reportDatePre = reportDatePre;
-	}
-	public String getRsltCodePre() {
-		return rsltCodePre;
-	}
-	public void setRsltCodePre(String rsltCodePre) {
-		this.rsltCodePre = rsltCodePre;
-	}
-	public String getRsltCode2Pre() {
-		return rsltCode2Pre;
-	}
-	public void setRsltCode2Pre(String rsltCode2Pre) {
-		this.rsltCode2Pre = rsltCode2Pre;
-	}
-	public String getRsltNetPre() {
-		return rsltNetPre;
-	}
-	public void setRsltNetPre(String rsltNetPre) {
-		this.rsltNetPre = rsltNetPre;
-	}
-	public String getConectMthd() {
-		return conectMthd;
-	}
-	public void setConectMthd(String conectMthd) {
-		this.conectMthd = conectMthd;
-	}
-
-	public String getAgentCode() {
-		return agentCode;
-	}
-	public void setAgentCode(String agentCode) {
-		this.agentCode = agentCode;
-	}
-	public String getConectMthdTxt() {
-		return conectMthdTxt;
-	}
-	public void setConectMthdTxt(String conectMthdTxt) {
-		this.conectMthdTxt = conectMthdTxt;
-	}
-	public String getAgentCodeTxt() {
-		return agentCodeTxt;
-	}
-	public void setAgentCodeTxt(String agentCodeTxt) {
-		this.agentCodeTxt = agentCodeTxt;
-	}
-	public String getCurStateTxt() {
-		return curStateTxt;
-	}
-	public void setCurStateTxt(String curStateTxt) {
-		this.curStateTxt = curStateTxt;
-	}
-	public String getMsgTypeTxt() {
-		return msgTypeTxt;
-	}
-	public void setMsgTypeTxt(String msgTypeTxt) {
-		this.msgTypeTxt = msgTypeTxt;
-	}
-	public String getSentDateTxt() {
-		return sentDateTxt;
-	}
-	public void setSentDateTxt(String sentDateTxt) {
-		this.sentDateTxt = sentDateTxt;
-	}
-	public String getSearchCondition2() {
-		return searchCondition2;
-	}
-	public void setSearchCondition2(String searchCondition2) {
-		this.searchCondition2 = searchCondition2;
-	}
-	public String getDelFlag() {
-		return delFlag;
-	}
-	public void setDelFlag(String delFlag) {
-		this.delFlag = delFlag;
-	}
-	public String getDelFlagTxt() {
-		return delFlagTxt;
-	}
-	public void setDelFlagTxt(String delFlagTxt) {
-		this.delFlagTxt = delFlagTxt;
-	}
-	public String getSearchCondition3() {
-		return searchCondition3;
-	}
-	public void setSearchCondition3(String searchCondition3) {
-		this.searchCondition3 = searchCondition3;
-	}
-	public String getSearchCondition4() {
-		return searchCondition4;
-	}
-	public void setSearchCondition4(String searchCondition4) {
-		this.searchCondition4 = searchCondition4;
-	}
-	public String getMmsSubject() {
-		return mmsSubject;
-	}
-	public void setMmsSubject(String mmsSubject) {
-		this.mmsSubject = mmsSubject;
-	}
-	public String getFileCnt() {
-		return fileCnt;
-	}
-	public void setFileCnt(String fileCnt) {
-		this.fileCnt = fileCnt;
-	}
-	public String getFileType1() {
-		return fileType1;
-	}
-	public void setFileType1(String fileType1) {
-		this.fileType1 = fileType1;
-	}
-	public String getFileName1() {
-		return fileName1;
-	}
-	public void setFileName1(String fileName1) {
-		this.fileName1 = fileName1;
-	}
-	public String getFileType2() {
-		return fileType2;
-	}
-	public void setFileType2(String fileType2) {
-		this.fileType2 = fileType2;
-	}
-	public String getFileName2() {
-		return fileName2;
-	}
-	public void setFileName2(String fileName2) {
-		this.fileName2 = fileName2;
-	}
-	public String getFileType3() {
-		return fileType3;
-	}
-	public void setFileType3(String fileType3) {
-		this.fileType3 = fileType3;
-	}
-	public String getFileName3() {
-		return fileName3;
-	}
-	public void setFileName3(String fileName3) {
-		this.fileName3 = fileName3;
-	}
-	public String getSubject() {
-		return subject;
-	}
-	public void setSubject(String subject) {
-		this.subject = subject;
-	}
-	public String getMsgGroupId() {
-		return msgGroupId;
-	}
-	public void setMsgGroupId(String msgGroupId) {
-		this.msgGroupId = msgGroupId;
-	}
-	public String getMsgGroupCnt() {
-		return msgGroupCnt;
-	}
-	public void setMsgGroupCnt(String msgGroupCnt) {
-		this.msgGroupCnt = msgGroupCnt;
-	}
-	public String[] getCallToList() {
-		return callToList;
-	}
-	public void setCallToList(String[] strings) {
-		this.callToList = strings;
-	}
-	public String[] getImgFilePath() {
-		return imgFilePath;
-	}
-	public void setImgFilePath(String[] imgFilePath) {
-		this.imgFilePath = imgFilePath;
-	}
-	public String getNeoType() {
-		return neoType;
-	}
-	public void setNeoType(String neoType) {
-		this.neoType = neoType;
-	}
-	public int getMsgCnt() {
-		return msgCnt;
-	}
-	public void setMsgCnt(int msgCnt) {
-		this.msgCnt = msgCnt;
-	}
-	public String getSearchCondition5() {
-		return searchCondition5;
-	}
-	public void setSearchCondition5(String searchCondition5) {
-		this.searchCondition5 = searchCondition5;
-	}
-	public String getNtceBgnde() {
-		return ntceBgnde;
-	}
-	public void setNtceBgnde(String ntceBgnde) {
-		this.ntceBgnde = ntceBgnde;
-	}
-	public String getEachPrice() {
-		return eachPrice;
-	}
-	public void setEachPrice(String eachPrice) {
-		this.eachPrice = eachPrice;
-	}
-	public String getTotPrice() {
-		return totPrice;
-	}
-	public void setTotPrice(String totPrice) {
-		this.totPrice = totPrice;
-	}
-	public String getBeforeUrl() {
-		return beforeUrl;
-	}
-	public void setBeforeUrl(String beforeUrl) {
-		this.beforeUrl = beforeUrl;
-	}
-	public String getReserveYn() {
-		return reserveYn;
-	}
-	public void setReserveYn(String reserveYn) {
-		this.reserveYn = reserveYn;
-	}
-	public String getReserveCYn() {
-		return reserveCYn;
-	}
-	public void setReserveCYn(String reserveCYn) {
-		this.reserveCYn = reserveCYn;
-	}
-	public String getCancelDate() {
-		return cancelDate;
-	}
-	public void setCancelDate(String cancelDate) {
-		this.cancelDate = cancelDate;
-	}
-	public String[] getImgFileId() {
-		return imgFileId;
-	}
-	public void setImgFileId(String[] imgFileId) {
-		this.imgFileId = imgFileId;
-	}
-	public String[] getTemplateYn() {
-		return templateYn;
-	}
-	public void setTemplateYn(String[] templateYn) {
-		this.templateYn = templateYn;
-	}
-	public String getNtceEndde() {
-		return ntceEndde;
-	}
-	public void setNtceEndde(String ntceEndde) {
-		this.ntceEndde = ntceEndde;
-	}
-	public static long getSerialversionuid() {
-		return serialVersionUID;
-	}
-	public String getSendRate() {
-		return sendRate;
-	}
-	public void setSendRate(String sendRate) {
-		this.sendRate = sendRate;
-	}
-	public float getSendRateInfo() {
-		return sendRateInfo;
-	}
-	public void setSendRateInfo(float sendRateInfo) {
-		this.sendRateInfo = sendRateInfo;
-	}
-	public String getDivideChk() {
-		return divideChk;
-	}
-	public void setDivideChk(String divideChk) {
-		this.divideChk = divideChk;
-	}
-	public String getDivideCnt() {
-		return divideCnt;
-	}
-	public void setDivideCnt(String divideCnt) {
-		this.divideCnt = divideCnt;
-	}
-	public String getDivideTime() {
-		return divideTime;
-	}
-	public void setDivideTime(String divideTime) {
-		this.divideTime = divideTime;
-	}
-	public String getBefCash() {
-		return befCash;
-	}
-	public void setBefCash(String befCash) {
-		this.befCash = befCash;
-	}
-	public String getRecommId() {
-		return recommId;
-	}
-	public void setRecommId(String recommId) {
-		this.recommId = recommId;
-	}
-	public String getBefPoint() {
-		return befPoint;
-	}
-	public void setBefPoint(String befPoint) {
-		this.befPoint = befPoint;
-	}
-	public String[] getNameList() {
-		return nameList;
-	}
-	public void setNameList(String[] nameList) {
-		this.nameList = nameList;
-	}
-	public String[] getRep1List() {
-		return rep1List;
-	}
-	public void setRep1List(String[] rep1List) {
-		this.rep1List = rep1List;
-	}
-	public String[] getRep2List() {
-		return rep2List;
-	}
-	public void setRep2List(String[] rep2List) {
-		this.rep2List = rep2List;
-	}
-	public String[] getRep3List() {
-		return rep3List;
-	}
-	public void setRep3List(String[] rep3List) {
-		this.rep3List = rep3List;
-	}
-	public String[] getRep4List() {
-		return rep4List;
-	}
-	public void setRep4List(String[] rep4List) {
-		this.rep4List = rep4List;
-	}
-	public String getThisPoint() {
-		return thisPoint;
-	}
-	public void setThisPoint(String thisPoint) {
-		this.thisPoint = thisPoint;
-	}
-	public String getEndDate() {
-		return endDate;
-	}
-	public void setEndDate(String endDate) {
-		this.endDate = endDate;
-	}
-	public String getStartDate() {
-		return startDate;
-	}
-	public void setStartDate(String startDate) {
-		this.startDate = startDate;
-	}
-	public int getPhoneNumberCnt() {
-		return phoneNumberCnt;
-	}
-	public void setPhoneNumberCnt(int phoneNumberCnt) {
-		this.phoneNumberCnt = phoneNumberCnt;
-	}
-	public String getPhmAuthType() {
-		return phmAuthType;
-	}
-	public void setPhmAuthType(String phmAuthType) {
-		this.phmAuthType = phmAuthType;
-	}
-	public String getRefundYn() {
-		return refundYn;
-	}
-	public void setRefundYn(String refundYn) {
-		this.refundYn = refundYn;
-	}
-	public String getFilePath1() {
-		return filePath1;
-	}
-	public void setFilePath1(String filePath1) {
-		this.filePath1 = filePath1;
-	}
-	public String getFilePath2() {
-		return filePath2;
-	}
-	public void setFilePath2(String filePath2) {
-		this.filePath2 = filePath2;
-	}
-	public String getFilePath3() {
-		return filePath3;
-	}
-	public void setFilePath3(String filePath3) {
-		this.filePath3 = filePath3;
-	}
-	public String getResultCodeTxt() {
-		return resultCodeTxt;
-	}
-	public void setResultCodeTxt(String resultCodeTxt) {
-		this.resultCodeTxt = resultCodeTxt;
-	}
-	public String getTxtReplYn() {
-		return txtReplYn;
-	}
-	public void setTxtReplYn(String txtReplYn) {
-		this.txtReplYn = txtReplYn;
-	}
-	public String getsPrice() {
-		return sPrice;
-	}
-	public void setsPrice(String sPrice) {
-		this.sPrice = sPrice;
-	}
-	public String getmPrice() {
-		return mPrice;
-	}
-	public void setmPrice(String mPrice) {
-		this.mPrice = mPrice;
-	}
-	public String getpPrice() {
-		return pPrice;
-	}
-	public void setpPrice(String pPrice) {
-		this.pPrice = pPrice;
-	}
-	public String getP2Price() {
-		return p2Price;
-	}
-	public void setP2Price(String p2Price) {
-		this.p2Price = p2Price;
-	}
-	public String getP3Price() {
-		return p3Price;
-	}
-	public void setP3Price(String p3Price) {
-		this.p3Price = p3Price;
-	}
-	public String getShortMsgCnt() {
-		return shortMsgCnt;
-	}
-	public void setShortMsgCnt(String shortMsgCnt) {
-		this.shortMsgCnt = shortMsgCnt;
-	}
-	public String getLongMsgCnt() {
-		return longMsgCnt;
-	}
-	public void setLongMsgCnt(String longMsgCnt) {
-		this.longMsgCnt = longMsgCnt;
-	}
-	public String getAtchFileId1() {
-		return atchFileId1;
-	}
-	public void setAtchFileId1(String atchFileId1) {
-		this.atchFileId1 = atchFileId1;
-	}
-	public String getAtchFileId2() {
-		return atchFileId2;
-	}
-	public void setAtchFileId2(String atchFileId2) {
-		this.atchFileId2 = atchFileId2;
-	}
-	public String getAtchFileId3() {
-		return atchFileId3;
-	}
-	public void setAtchFileId3(String atchFileId3) {
-		this.atchFileId3 = atchFileId3;
-	}
-	public String getAtchFileSn1() {
-		return atchFileSn1;
-	}
-	public void setAtchFileSn1(String atchFileSn1) {
-		this.atchFileSn1 = atchFileSn1;
-	}
-	public String getAtchFileSn2() {
-		return atchFileSn2;
-	}
-	public void setAtchFileSn2(String atchFileSn2) {
-		this.atchFileSn2 = atchFileSn2;
-	}
-	public String getAtchFileSn3() {
-		return atchFileSn3;
-	}
-	public void setAtchFileSn3(String atchFileSn3) {
-		this.atchFileSn3 = atchFileSn3;
-	}
-	
-	public String getMsgTypeName() {
-		return msgTypeName;
-	}
-	public void setMsgTypeName(String msgTypeName) {
-		this.msgTypeName = msgTypeName;
-	}
-	public int getSendCount() {
-		return sendCount;
-	}
-	public void setSendCount(int sendCount) {
-		this.sendCount = sendCount;
-	}
-	public double getSupplyPrice() {
-		return supplyPrice;
-	}
-	public void setSupplyPrice(double supplyPrice) {
-		this.supplyPrice = supplyPrice;
-	}
-	public double getVatPrice() {
-		return vatPrice;
-	}
-	public void setVatPrice(double vatPrice) {
-		this.vatPrice = vatPrice;
-	}
-	public double getTotalPrice() {
-		return totalPrice;
-	}
-	public void setTotalPrice(double totalPrice) {
-		this.totalPrice = totalPrice;
-	}
-	public String getDetailType() {
-		return detailType;
-	}
-	public void setDetailType(String detailType) {
-		this.detailType = detailType;
-	}
-	public String getMsgGroupSCnt() {
-		return msgGroupSCnt;
-	}
-	public void setMsgGroupSCnt(String msgGroupSCnt) {
-		this.msgGroupSCnt = msgGroupSCnt;
-	}
-	public String getMsgGroupFWCnt() {
-		return MsgGroupFWCnt;
-	}
-	public void setMsgGroupFWCnt(String msgGroupFWCnt) {
-		MsgGroupFWCnt = msgGroupFWCnt;
-	}
-	public String getTotSPrice() {
-		return totSPrice;
-	}
-	public void setTotSPrice(String totSPrice) {
-		this.totSPrice = totSPrice;
-	}
-	public String getTotFWPrice() {
-		return totFWPrice;
-	}
-	public void setTotFWPrice(String totFWPrice) {
-		this.totFWPrice = totFWPrice;
-	}
-	public String getMsgGroupSCntSum() {
-		return msgGroupSCntSum;
-	}
-	public void setMsgGroupSCntSum(String msgGroupSCntSum) {
-		this.msgGroupSCntSum = msgGroupSCntSum;
-	}
-	public String getMsgGroupFWCntSum() {
-		return MsgGroupFWCntSum;
-	}
-	public void setMsgGroupFWCntSum(String msgGroupFWCntSum) {
-		MsgGroupFWCntSum = msgGroupFWCntSum;
-	}
-	public String getTotSPriceSum() {
-		return totSPriceSum;
-	}
-	public void setTotSPriceSum(String totSPriceSum) {
-		this.totSPriceSum = totSPriceSum;
-	}
-	public String getTotFWPriceSum() {
-		return totFWPriceSum;
-	}
-	public void setTotFWPriceSum(String totFWPriceSum) {
-		this.totFWPriceSum = totFWPriceSum;
-	}
-	public String getSmiId() {
-		return smiId;
-	}
-	public void setSmiId(String smiId) {
-		this.smiId = smiId;
-	}
-	public List<String> getDividDay() {
-		return dividDay;
-	}
-	public void setDividDay(List<String> dividDay) {
-		this.dividDay = dividDay;
-	}
-	public String getUserNm() {
-		return userNm;
-	}
-	public void setUserNm(String userNm) {
-		this.userNm = userNm;
-	}
-	public String getMbtlnum() {
-		return mbtlnum;
-	}
-	public void setMbtlnum(String mbtlnum) {
-		this.mbtlnum = mbtlnum;
-	}
-	public String getEmailAdres() {
-		return emailAdres;
-	}
-	public void setEmailAdres(String emailAdres) {
-		this.emailAdres = emailAdres;
-	}
-	public String getAuthorCode() {
-		return authorCode;
-	}
-	public void setAuthorCode(String authorCode) {
-		this.authorCode = authorCode;
-	}
-	public String getResultLogUpdtPnttm() {
-		return resultLogUpdtPnttm;
-	}
-	public void setResultLogUpdtPnttm(String resultLogUpdtPnttm) {
-		this.resultLogUpdtPnttm = resultLogUpdtPnttm;
-	}
-	public String getRepAgent() {
-		return repAgent;
-	}
-	public void setRepAgent(String repAgent) {
-		this.repAgent = repAgent;
-	}
-	public String getEventYn() {
-		return eventYn;
-	}
-	public void setEventYn(String eventYn) {
-		this.eventYn = eventYn;
-	}
-
-	public String getSpamKeyword() {
-		return spamKeyword;
-	}
-	public void setSpamKeyword(String spamKeyword) {
-		this.spamKeyword = spamKeyword;
-	}
-	public String getSpamMsgGroupId() {
-		return spamMsgGroupId;
-	}
-	public void setSpamMsgGroupId(String spamMsgGroupId) {
-		this.spamMsgGroupId = spamMsgGroupId;
-	}
-	public String getSpamStatus() {
-		return spamStatus;
-	}
-	public void setSpamStatus(String spamStatus) {
-		this.spamStatus = spamStatus;
-	}
-
-	public String getSmishingYn() {
-		return smishingYn;
-	}
-	public void setSmishingYn(String smishingYn) {
-		this.smishingYn = smishingYn;
-	}
-	public int getTotalCallCnt() {
-		return totalCallCnt;
-	}
-	public void setTotalCallCnt(int totalCallCnt) {
-		this.totalCallCnt = totalCallCnt;
-	}
-	public String getDelayYn() {
-		return delayYn;
-	}
-	public void setDelayYn(String delayYn) {
-		this.delayYn = delayYn;
-	}
-	public String getDelayCompleteYn() {
-		return delayCompleteYn;
-	}
-	public void setDelayCompleteYn(String delayCompleteYn) {
-		this.delayCompleteYn = delayCompleteYn;
-	}
-	public String getVipYn() {
-		return vipYn;
-	}
-	public void setVipYn(String vipYn) {
-		this.vipYn = vipYn;
-	}
-	public String getResultCode() {
-		return resultCode;
-	}
-	public void setResultCode(String resultCode) {
-		this.resultCode = resultCode;
-	}
-	public String getResultCode2() {
-		return resultCode2;
-	}
-	public void setResultCode2(String resultCode2) {
-		this.resultCode2 = resultCode2;
-	}
-	public String getRegistPnttm() {
-		return registPnttm;
-	}
-	public void setRegistPnttm(String registPnttm) {
-		this.registPnttm = registPnttm;
-	}
-	public String getBizKakaoResendData() {
-		return bizKakaoResendData;
-	}
-	public void setBizKakaoResendData(String bizKakaoResendData) {
-		this.bizKakaoResendData = bizKakaoResendData;
-	}
-	public String getBizKakaoTitle() {
-		return bizKakaoTitle;
-	}
-	public void setBizKakaoTitle(String bizKakaoTitle) {
-		this.bizKakaoTitle = bizKakaoTitle;
-	}
-	public String getSmsTxtTrans() {
-		return smsTxtTrans;
-	}
-	public void setSmsTxtTrans(String smsTxtTrans) {
-		this.smsTxtTrans = smsTxtTrans;
-	}
-	public String getBizKakaoTitle01() {
-		return bizKakaoTitle01;
-	}
-	public void setBizKakaoTitle01(String bizKakaoTitle01) {
-		this.bizKakaoTitle01 = bizKakaoTitle01;
-	}
-	public String getBizKakaoTitle02() {
-		return bizKakaoTitle02;
-	}
-	public void setBizKakaoTitle02(String bizKakaoTitle02) {
-		this.bizKakaoTitle02 = bizKakaoTitle02;
-	}
-	public String getBizKakaoResendOrgnlTxt() {
-		return bizKakaoResendOrgnlTxt;
-	}
-	public void setBizKakaoResendOrgnlTxt(String bizKakaoResendOrgnlTxt) {
-		this.bizKakaoResendOrgnlTxt = bizKakaoResendOrgnlTxt;
-	}
-	public String getBizKakaoResendYn() {
-		return bizKakaoResendYn;
-	}
-	public void setBizKakaoResendYn(String bizKakaoResendYn) {
-		this.bizKakaoResendYn = bizKakaoResendYn;
-	}
-	public String getBizKakaoResendType() {
-		return bizKakaoResendType;
-	}
-	public void setBizKakaoResendType(String bizKakaoResendType) {
-		this.bizKakaoResendType = bizKakaoResendType;
-	}
-	public String getBizKakaoJsonFile() {
-		return bizKakaoJsonFile;
-	}
-	public void setBizKakaoJsonFile(String bizKakaoJsonFile) {
-		this.bizKakaoJsonFile = bizKakaoJsonFile;
-	}
-	public String getYellowId() {
-		return yellowId;
-	}
-	public void setYellowId(String yellowId) {
-		this.yellowId = yellowId;
-	}
-	public String getBizKakaoResendTypeCnt() {
-		return bizKakaoResendTypeCnt;
-	}
-	public void setBizKakaoResendTypeCnt(String bizKakaoResendTypeCnt) {
-		this.bizKakaoResendTypeCnt = bizKakaoResendTypeCnt;
-	}
 	
 	
 	
src/main/java/itn/let/kakao/admin/kakaoAt/web/MjonKakaoATController.java
--- src/main/java/itn/let/kakao/admin/kakaoAt/web/MjonKakaoATController.java
+++ src/main/java/itn/let/kakao/admin/kakaoAt/web/MjonKakaoATController.java
@@ -796,9 +796,16 @@
 				String lastUpdtPnttm = resultChannelList.get(i).getLastUpdtPnttm();
 				kakaoProfileVO.setSenderKey(senderKey); 
 				kakaoProfileVO.setProfileId(profileId);
-				
-				KakaoReturnVO tmpProfileVO = kakaoApiProfile.kakaoApiProfileList(kakaoProfileVO);
-				
+				KakaoReturnVO tmpProfileVO = null;
+//				try {
+//
+//					tmpProfileVO = kakaoApiProfile.kakaoApiProfileList(kakaoProfileVO);					
+//				} catch (Exception e) {
+//					e.printStackTrace();
+//					// TODO: handle exception
+//				}
+
+				tmpProfileVO = kakaoApiProfile.kakaoApiProfileList(kakaoProfileVO);			
 				ChannelIDVO returnChannelVO = new ChannelIDVO();
 				
 				returnChannelVO.setSenderKey(tmpProfileVO.getSenderKey());
 
src/main/java/itn/let/kakao/kakaoComm/BizKakaoPriceVO.java (added)
+++ src/main/java/itn/let/kakao/kakaoComm/BizKakaoPriceVO.java
@@ -0,0 +1,34 @@
+package itn.let.kakao.kakaoComm;
+
+import java.math.BigDecimal;
+
+import lombok.AllArgsConstructor;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import lombok.Setter;
+import lombok.ToString;
+
+/**
+ * @  BIZ_KAKAO_PRICE 테이블 관련  
+ */
+@Getter
+@Setter
+@NoArgsConstructor
+@AllArgsConstructor
+@ToString
+public class BizKakaoPriceVO {
+
+	private Long bizKakaoPriceId;
+
+	private String bizKakaoAtPrice;         // 알림톡 단가
+	private String bizKakaoFtPrice;         // 친구톡 단가
+	private String bizKakaoFtImgPrice;      // 친구톡 이미지 단가
+	private String bizKakaoFtWideImgPrice;  // 친구톡 와이드 이미지 단가
+	private String bizSmsPrice;             // SMS 단가
+	private String bizMmsPrice;             // MMS 단가
+
+	private String msgGroupId;  // 메시지 그룹 ID
+
+
+	// getters and setters 생략
+}(No newline at end of file)
src/main/java/itn/let/kakao/kakaoComm/KakaoSendAdvcVO.java
--- src/main/java/itn/let/kakao/kakaoComm/KakaoSendAdvcVO.java
+++ src/main/java/itn/let/kakao/kakaoComm/KakaoSendAdvcVO.java
@@ -45,7 +45,13 @@
 	private String subMsgType;			// 대체문자 타입
 	private String reqDate;				// 예약일시
 	
+	private String msgKind;				// 예약일시
+	
 	private String jsonStr;				// jsonStr
+	
+	
+
+	private String adFlag;				//친구톡 광고성 정보 사용 유무(Y:사용 , N:미사용)
 
 	// =====
 	// =====
@@ -53,6 +59,7 @@
 	private String eachPrice;		// sms 단가
 	private String smsPrice;		// sms 단가
 	private String mmsPrice;		// mms 단가
+	private String picturePrice;		// mms 단가
 	private String totPrice;		// mms 단가
 	private String befCash;		// mms 단가
 	private String befPoint;		// mms 단가
@@ -62,6 +69,9 @@
 	private String atDelayYn;	// 카카오 알림톡 단가
 	private String bizKakaoResendOrgnlTxt;	// 카카오 알림톡 단가
 	private String bizKakaoResendType;	// 카카오 알림톡 단가
+	private String filePath1;	// 대체문자 이미지 
+	private String fileCnt;	// 파일 카운트
+	private String bizKakaoImageType;	// 파일 카운트
 	
 	
 	
@@ -83,7 +93,7 @@
 			"\n , msgType=[" + msgType + "]" +
 			"\n , templateContent=[" + templateContent + "]" +
 			"\n , templateTitle=[" + templateTitle + "]" +
-			"\n , buttonList=[" + buttonList.toString() + "]" +
+			"\n , buttonList=[" + (buttonList != null ? buttonList.toString() : "") + "]" +
 			"\n , subMsgSendYn=[" + subMsgSendYn + "]" +
 			"\n , subMsgTxt=[" + subMsgTxt + "]" +
 			"\n , subMsgType=[" + subMsgType + "]" +
@@ -103,6 +113,8 @@
 			"\n , atDelayYn=[" + atDelayYn + "]" +
 			"\n , bizKakaoResendOrgnlTxt=[" + bizKakaoResendOrgnlTxt + "]" +
 			"\n , bizKakaoResendType=[" + bizKakaoResendType + "]" +
+			"\n , filePath1=[" + filePath1 + "]" +
+			"\n , bizKakaoImageType=[" + bizKakaoImageType + "]" +
 			"\n ]";
 	}
 
src/main/java/itn/let/kakao/kakaoComm/KakaoSendUtil.java
--- src/main/java/itn/let/kakao/kakaoComm/KakaoSendUtil.java
+++ src/main/java/itn/let/kakao/kakaoComm/KakaoSendUtil.java
@@ -1,12 +1,17 @@
 package itn.let.kakao.kakaoComm;
 
+import java.io.UnsupportedEncodingException;
+import java.math.BigDecimal;
+import java.math.RoundingMode;
 import java.text.ParseException;
 import java.text.SimpleDateFormat;
 import java.util.ArrayList;
 import java.util.Calendar;
 import java.util.Date;
+import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
+import java.util.function.Function;
 import java.util.regex.Matcher;
 import java.util.regex.Pattern;
 
@@ -19,12 +24,17 @@
 import org.springframework.stereotype.Component;
 
 import egovframework.rte.fdl.idgnr.EgovIdGnrService;
+import itn.com.cmm.MjonFTSendVO;
+import itn.com.cmm.MjonMsgSendVO;
+import itn.com.cmm.util.MsgSendUtils;
 import itn.com.cmm.util.StringUtil;
 import itn.let.kakao.kakaoComm.kakaoApi.KakaoApiJsonSave;
 import itn.let.kakao.kakaoComm.kakaoApi.KakaoApiTemplate;
+import itn.let.kakao.user.kakaoAt.service.impl.KakaoAlimTalkDAO;
 import itn.let.mail.service.StatusResponse;
 import itn.let.mjo.mjocommon.MjonCommon;
 import itn.let.mjo.msg.service.MjonMsgVO;
+import itn.let.mjo.msg.service.impl.MjonMsgDAO;
 import itn.let.mjo.msgdata.service.MjonMsgDataService;
 import itn.let.mjo.spammsg.web.ComGetSpamStringParser;
 import itn.let.module.base.PriceAndPoint;
@@ -38,9 +48,15 @@
 
 	@Autowired
 	KakaoApiJsonSave kakaoApiJsonSave;
+
+	@Resource(name="kakaoAlimTalkDAO")
+	private KakaoAlimTalkDAO kakaoAlimTalkDAO;
 	
 	@Resource(name = "MjonMsgDataService")
 	private MjonMsgDataService mjonMsgDataService;
+
+	@Resource(name = "mjonMsgDAO")
+	private MjonMsgDAO mjonMsgDAO;
 	
 	@Autowired
 	KakaoApiTemplate kakaoApiTemplate;
@@ -50,6 +66,7 @@
 
 	@Autowired
 	private MjonCommon mjonCommon;
+	
 	
 	// 클래스 수준에서 정적 Pattern 정의 (성능 최적화)
 	private static final Pattern REPLACEMENT_PATTERN = Pattern.compile("#\\{[^}]+\\}");
@@ -138,7 +155,7 @@
 			log.info("");
 			
 /** @공통 기본값 */		
-			KakaoSendAdvcVO sendVO = createSendVO(kakaoVO);
+			KakaoSendAdvcVO sendVO = createATSendVO(kakaoVO);
 			String msgId = idList.get(i);
 			sendVO.setMsgId(msgId);
 
@@ -273,18 +290,265 @@
 		return kakaoSendAdvcListVO;
 	}
 	
-	private Calendar setupBaseDate(KakaoVO kakaoVO, boolean isNotified) throws ParseException {
-		// 예약 시간 기본값 설정
-		Date now = new Date();
-		// ReqDate가 비어 있으면 현재 시간으로 설정, 그렇지 않으면 ReqDate로 설정
-		// 화면에서 예약문자면 예약시간을 regDate로 설정한다.
-		Date baseDate;
-		if (StringUtils.isEmpty(kakaoVO.getReqDate())) {
-			kakaoVO.setReqDate(DATE_FORMATTER.format(now));  // ReqDate에 현재 시간 설정
-		    baseDate = now;
-		} else {
-		    baseDate = DATE_FORMATTER.parse(kakaoVO.getReqDate()); // ReqDate를 baseDate로 설정
+	/**
+	 * @methodName	: populateSendListsFT 
+	 * @author		: 이호영
+	 * @date		: 2025. 4. 18.
+	 * @description	: 
+	 * @return : List<KakaoSendAdvcVO>
+	 * @param kakaoVO
+	 * @param isNotified
+	 * @param statusResponse
+	 * @return
+	 * @throws Exception
+	 * 
+	 */
+	public List<KakaoSendAdvcVO> populateSendListsFT(KakaoVO kakaoVO, boolean isNotified, StatusResponse statusResponse) throws Exception {
+		
+		//사용자 현재 보유 금액 불러오기(문자 발송 금액 차감 이전 금액)
+//		String befCash = kakaoVO.getBefCash();
+		
+		log.info("kakaoVO.ftToString() ::  [{}]", kakaoVO.ftToString());		
+		
+		List<KakaoSendAdvcVO> kakaoSendAdvcListVO = new ArrayList<>();
+		Calendar calendar = setupBaseDate(kakaoVO, isNotified);
+		
+		
+		 // 친구톡 내용
+		String templateContent = kakaoVO.getTemplateContent();
+		 // 실패 대체 문자
+		String subMsgTxt = kakaoVO.getSubMsgTxt();
+		log.info(" + StringUtils.isNotEmpty(subMsgTxt) :: [{}]", StringUtils.isNotEmpty(subMsgTxt));
+		if(StringUtils.isNotEmpty(subMsgTxt)) {
+			kakaoVO.setSubMsgSendYn("Y");
+			
+			// 광고문자면 처리 - 광고 Y
+			if ("Y".equals(kakaoVO.getAdFlag())) {
+				subMsgTxt = "(광고)" + subMsgTxt + "\n" + "무료거부 0808800858";
+			}
+			
 		}
+		
+		
+
+		// 사용자 개인 단가 정보 불러오기
+		MberManageVO mberManageVO = mjonMsgDataService.selectMberManageInfo(kakaoVO.getUserId());
+		// 시스템 기본 단가 정보 불러오기
+		JoinSettingVO sysJoinSetVO = mjonMsgDataService.selectJoinSettingInfo();
+		
+
+		// 치환 문구가 있는지 확인
+		Boolean replaceYN = MsgSendUtils.getReplaceYN(templateContent);
+		Boolean replaceSubYN = MsgSendUtils.getReplaceYN(subMsgTxt);
+
+		
+		/** @MSGID KEY값 */
+		List<String> idList = mjonCommon.getNextCustomMsgCId(kakaoVO.getMjonFTSendVOList().size());
+
+
+		Map<String, Function<MjonFTSendVO, String>> placeholders = new HashMap<>();
+		placeholders.put("[*이름*]", MjonFTSendVO::getName);
+		placeholders.put("[*1*]", MjonFTSendVO::getRep1);
+		placeholders.put("[*2*]", MjonFTSendVO::getRep2);
+		placeholders.put("[*3*]", MjonFTSendVO::getRep3);
+		placeholders.put("[*4*]", MjonFTSendVO::getRep4);
+
+		String imageType = kakaoVO.getImageType();
+		// 개인단가
+		Float kakaoMemberFtPrice = 
+				imageType  == null ? mberManageVO.getKakaoFtPrice() :
+				"I".equals(imageType) ? mberManageVO.getKakaoFtImgPrice() :
+				"W".equals(imageType) ? mberManageVO.getKakaoFtWideImgPrice() :
+				mberManageVO.getKakaoFtPrice();
+		
+		// 시스템단가
+		Float kakaoSysJoinFtPrice = 
+				imageType  == null ? sysJoinSetVO.getKakaoFtPrice() :
+				"I".equals(imageType) ? sysJoinSetVO.getKakaoFtImgPrice() :
+				"W".equals(imageType) ? sysJoinSetVO.getKakaoFtWideImgPrice() :
+				sysJoinSetVO.getKakaoFtPrice();
+
+		Float kakaoFtPrice = 
+				getValidPrice(kakaoMemberFtPrice, kakaoSysJoinFtPrice);
+		
+		// 대체문자가 있을경우 사용 
+		float shortPrice = getValidPrice(mberManageVO.getShortPrice(), sysJoinSetVO.getShortPrice());
+		float longPrice = getValidPrice(mberManageVO.getLongPrice(), sysJoinSetVO.getLongPrice());
+		float picturePrice = getValidPrice(mberManageVO.getPicturePrice(), sysJoinSetVO.getPicturePrice());
+//		String shortPStr = Float.toString(shortPrice);
+//		String mmsPStr = Float.toString(longPrice);
+//		String imgPrice = Float.toString(picturePrice);
+
+		
+		String imgFilePath = "";
+		if(StringUtils.isNotEmpty(kakaoVO.getAtchFileId()) &&  
+				("I".equals(imageType) ||	"W".equals(imageType))) {
+			
+
+			imgFilePath = mjonMsgDAO.selectPhotoImgFileRealPath(kakaoVO.getAtchFileId());
+			
+		}
+		
+		
+
+		/** @jsonStr 필요유무 */
+		boolean hasButtons =  CollectionUtils.isNotEmpty(kakaoVO.getButtonVOList());
+		String sharedJsonStr = null;
+		
+		
+
+		List<MjonFTSendVO> mjonFTSendVOList = kakaoVO.getMjonFTSendVOList();
+
+		for (int i = 0; i < mjonFTSendVOList.size(); i++) {
+			MjonFTSendVO mjonFTSendVO = mjonFTSendVOList.get(i);
+
+			KakaoSendAdvcVO sendVO = createFTSendVO(kakaoVO, calendar);
+			// 공통 가격 설정
+			sendVO.setSmsPrice(Float.toString(shortPrice));
+			sendVO.setMmsPrice(Float.toString(longPrice));
+			sendVO.setPicturePrice(Float.toString(picturePrice));
+
+			sendVO.setCallTo(mjonFTSendVO.getPhone());
+			sendVO.setMsgId(idList.get(i));
+			
+			String smsTxt = templateContent;
+			// 치환 문자면
+			if(replaceYN) {
+				
+				// 각 치환 구문을 확인하고 치환할 값이 없으면 오류 반환
+				for (Map.Entry<String, Function<MjonFTSendVO, String>> entry : placeholders.entrySet()) {
+					String placeholder = entry.getKey();
+					String value = entry.getValue().apply(mjonFTSendVO);
+					if (smsTxt.contains(placeholder)) {
+						if (StringUtils.isEmpty(value)) {
+							statusResponseSet(statusResponse, HttpStatus.BAD_REQUEST, "치환 문구중 " + placeholder + " 데이터가 없습니다.");
+							return null;
+						}
+						smsTxt = smsTxt.replace(placeholder, value);
+					}
+				}
+			}
+			sendVO.setTemplateContent(smsTxt);
+			
+			
+			
+			String subMsgTxtTemp = null;
+			
+			if(StringUtils.isNotEmpty(subMsgTxt)) {
+				subMsgTxtTemp = subMsgTxt;
+				
+				if(replaceSubYN) {
+					// 각 치환 구문을 확인하고 치환할 값이 없으면 오류 반환
+					for (Map.Entry<String, Function<MjonFTSendVO, String>> entry : placeholders.entrySet()) {
+						String placeholder = entry.getKey();
+						String value = entry.getValue().apply(mjonFTSendVO);
+						if (subMsgTxtTemp.contains(placeholder)) {
+							if (StringUtils.isEmpty(value)) {
+								statusResponseSet(statusResponse, HttpStatus.BAD_REQUEST, "치환 문구중 " + placeholder + " 데이터가 없습니다.");
+								return null;
+							}
+							subMsgTxtTemp = subMsgTxtTemp.replace(placeholder, value);
+						}
+					}
+				}
+			}
+			sendVO.setSubMsgTxt(subMsgTxtTemp);
+			
+			
+			//대체문자가 있으면
+			// Step 1-4: 실패 대체 문자 치환데이터 설정
+			if(StringUtils.isNotEmpty(subMsgTxtTemp)) { // 대체문자가 있나?
+				
+				String sendType =  "MMS";
+				if(StringUtils.isEmpty(imgFilePath)) {
+					int smsTxtByte = mjonCommon.getSmsTxtBytes(subMsgTxtTemp);	// 문자 byte 수 계산
+					sendType = getMsgType(smsTxtByte);					// 문자 타입(SHORT / MMS) 판별
+				}
+				sendVO.setSubMsgType(sendType);								// 실패 대체 문자 타입 설정
+
+				if ("INVALID".equals(sendType)) {
+					// INVALID 타입이면 길이 초과 에러 응답 후 리턴
+					statusResponseSet(statusResponse, HttpStatus.BAD_REQUEST, "전송 문자 길이를 초과하였습니다.");
+					return kakaoSendAdvcListVO;
+				}
+
+				float chosenPrice = 0f;
+
+				if(StringUtils.isNotEmpty(imgFilePath)) {
+					chosenPrice = Math.max(picturePrice, kakaoFtPrice);
+					sendVO.setFilePath1(imgFilePath);
+					sendVO.setFileCnt("1");			
+					
+				}else if ("MMS".equals(sendType)) {
+					// MMS 타입일 경우: longPrice(장문 가격)와 카카오톡 단가 중 큰 값을 선택
+					chosenPrice = Math.max(longPrice, kakaoFtPrice);
+				} else {
+					// SHORT 타입일 경우: shortPrice(단문 가격)와 카카오톡 단가 중 큰 값을 선택
+					chosenPrice = Math.max(shortPrice, kakaoFtPrice);
+				}
+				sendVO.setEachPrice(Float.toString(chosenPrice));	// 선택된 단가 설정
+				
+				
+			}else {
+				// 대체문자가 없으면 카카오톡 단가 그대로 사용
+				sendVO.setEachPrice( Float.toString(kakaoFtPrice) );
+			}
+
+
+			// 타이틀과 버튼이 있고
+			if(hasButtons || StringUtils.isNotEmpty(kakaoVO.getTemplateImageUrl())) {
+				// 
+				if (StringUtils.isEmpty(sharedJsonStr)) {
+					// 치환 데이터가 없고 아직 생성되지 않았으면 한 번만 생성
+					sharedJsonStr = kakaoApiJsonSave.kakaoApiFTJsonSave_advc(kakaoVO);
+					sendVO.setJsonStr(sharedJsonStr);
+				}
+				sendVO.setBizJsonName(idList.get(0));
+				
+			}
+			
+			kakaoSendAdvcListVO.add(sendVO);
+			log.info(" sendVO.toString() :: [{}]",sendVO.toString());
+		}
+		
+		
+		return kakaoSendAdvcListVO;
+	}
+	
+	public static String getMsgTypeWithByteValidation(MjonFTSendVO  sendVO, String p_smsTxt) throws UnsupportedEncodingException {
+		
+
+		//	// 내문자저장함에 저장 후 문자를 발송하는 경우 문자 타입이 숫자가 아닌 문자로 넘어와서 변경 처리함
+		//	if ("P".equals(msgType) || "L".equals(msgType)) {
+		//	    msgType = "6";
+		//	} else if ("S".equals(msgType)) {
+		//	    msgType = "4";
+		//	}
+			
+		int smsTxtByte = MjonCommon.getSmsTxtBytes(p_smsTxt);
+		String msgType = SHORT_MSG_TYPE;
+	
+		// 1. 2000 Byte 초과는 에러 처리
+		if (smsTxtByte > 2000) {
+			return "INVALID";
+		}
+
+		// 2. 첨부파일 여부 확인 (첨부파일이 있으면 장문으로 설정)
+		if (StringUtils.isNotEmpty(sendVO.getFilePath1())) {
+			msgType = LONG_MSG_TYPE;
+		}
+		// 3. 문자 길이에 따라 메시지 타입 설정 (90 Byte 초과는 장문)
+		else if (smsTxtByte > 90) {
+			msgType = LONG_MSG_TYPE;
+		}
+		return msgType;
+	}
+	
+
+	private Calendar setupBaseDate(KakaoVO kakaoVO, boolean isNotified) throws ParseException {
+
+		// baseDate 추출
+		Date baseDate = resolveBaseDate(kakaoVO);
 		
 		// 시간 성정
 		Calendar calendar = Calendar.getInstance();
@@ -299,6 +563,31 @@
 		return calendar;
 	}
 
+	public Date resolveBaseDate(KakaoVO kakaoVO) throws ParseException {
+		Date now = new Date();
+
+		if (StringUtils.isEmpty(kakaoVO.getReqDate())) {
+			kakaoVO.setReqDate(DATE_FORMATTER.format(now));
+			return now;
+		}
+		return DATE_FORMATTER.parse(kakaoVO.getReqDate());
+	}
+
+	// 2. 친구톡 발송 제한 시간인지 확인
+	public boolean isRestrictedFriendTalkTime(Date baseDate) {
+		Calendar cal = Calendar.getInstance();
+		cal.setTime(baseDate);
+
+		int hour = cal.get(Calendar.HOUR_OF_DAY);
+		int minute = cal.get(Calendar.MINUTE);
+
+		// 20:50 이후 ~ 익일 08:00 이전은 제한
+		if ((hour == 20 && minute >= 50) || hour > 20 || hour < 8) {
+			return true;
+		}
+		return false;
+	}
+	
 	/**
 	 * @methodName	: createSendVO 
 	 * @author		: 이호영
@@ -309,14 +598,45 @@
 	 * @return
 	 * 
 	 */
-	private KakaoSendAdvcVO createSendVO(KakaoVO kakaoVO) {
+	private KakaoSendAdvcVO createATSendVO(KakaoVO kakaoVO) {
 		KakaoSendAdvcVO sendVO = new KakaoSendAdvcVO();
 		sendVO.setMsgType("8");
+		sendVO.setAgentCode("04");
 		sendVO.setSenderKey(kakaoVO.getSenderKey());
 		sendVO.setTemplateCode(kakaoVO.getTemplateCode());
 		sendVO.setUserId(kakaoVO.getUserId());
 		sendVO.setCallFrom(kakaoVO.getCallFrom());
+		return sendVO;
+	}
+	
+	
+	/**
+	 * @methodName	: createFTSendVO 
+	 * @author		: 이호영
+	 * @date		: 2025. 4. 23.
+	 * @description	: 
+	 * @return : KakaoSendAdvcVO
+	 * @param kakaoVO
+	 * @return
+	 * 
+	 */
+	private KakaoSendAdvcVO createFTSendVO(KakaoVO kakaoVO, Calendar calendar) {
+		KakaoSendAdvcVO sendVO = new KakaoSendAdvcVO();
+
+		sendVO.setMsgType("9"); // 알림톡 8 친구톡 9
 		sendVO.setAgentCode("04");
+		// 발송시간 : 친구톡은 분할 발송이 없어 처음 vo 생성 시 입력
+		sendVO.setReqDate(DATE_FORMATTER.format(calendar.getTime())); 
+		
+		sendVO.setSenderKey(kakaoVO.getSenderKey());
+		sendVO.setTemplateCode(kakaoVO.getTemplateCode());
+		sendVO.setUserId(kakaoVO.getUserId());
+		sendVO.setCallFrom(kakaoVO.getCallFrom());
+		sendVO.setSubMsgSendYn(kakaoVO.getSubMsgSendYn());
+		
+		sendVO.setAdFlag(kakaoVO.getAdFlag());
+		
+		
 		return sendVO;
 	}
 	
@@ -378,7 +698,7 @@
 	}
 
 
-	public Float getValidPrice(Float personalPrice, Float defaultPrice) {
+	public static Float getValidPrice(Float personalPrice, Float defaultPrice) {
 		return (personalPrice != null && personalPrice > 0) ? personalPrice : defaultPrice;
 	}
 
@@ -692,6 +1012,10 @@
 		//카카오 친구톡 개인 단가가 없는 경우 시스템 단가로
 		if(mberManageVO.getKakaoFtPrice() == 0.0f)
 			mberManageVO.setKakaoFtPrice(sysJoinSetVO.getKakaoFtPrice());
+		if(mberManageVO.getKakaoFtImgPrice() == 0.0f)
+			mberManageVO.setKakaoFtImgPrice(sysJoinSetVO.getKakaoFtImgPrice());
+		if(mberManageVO.getKakaoFtWideImgPrice() == 0.0f)
+			mberManageVO.setKakaoFtWideImgPrice(sysJoinSetVO.getKakaoFtWideImgPrice());
 		
 		
 		// SMS 인경우
@@ -703,7 +1027,9 @@
 		// 사용자 개인 단가가 없으면 시스템 단가로
 		if(mberManageVO.getLongPrice() == 0.0f)
 			mberManageVO.setLongPrice(sysJoinSetVO.getLongPrice());
-		
+
+		if(mberManageVO.getPicturePrice() == 0.0f)
+			mberManageVO.setPicturePrice(sysJoinSetVO.getPicturePrice());
 
 		return mberManageVO;
 	}
@@ -1202,4 +1528,175 @@
 		statusResponse.setMessage(msg);
 		
 	}
+	
+	
+	// 보유 금액이 충분한지 확인하는 메서드
+	public boolean isCashSufficient(String userId, List<KakaoSendAdvcVO> kakaoSendAdvcListVO) throws Exception {
+		
+		
+		String userMoney = priceAndPoint.getBefCash(userId);
+		// 쉼표 제거
+		userMoney = userMoney.replace(",", "");
+
+		
+		// 사용자 보유 금액 BigDecimal 변환 (HALF_EVEN 적용)
+		BigDecimal befCash = new BigDecimal(userMoney).setScale(2, RoundingMode.HALF_EVEN);
+		log.info(" + userMoney :: [{}]", userMoney);
+		log.info(" + befCash :: [{}]", befCash);
+		
+		
+		// 총 메시지 금액 계산 (HALF_EVEN 적용)
+		BigDecimal totalEachPrice = kakaoSendAdvcListVO.stream()
+			.map(msg -> new BigDecimal(String.valueOf(msg.getEachPrice()))) // 변환 오류 방지
+			.reduce(BigDecimal.ZERO, BigDecimal::add)
+			.setScale(2, RoundingMode.HALF_EVEN); // 일관성 유지
+
+		log.info(" + totalEachPrice :: [{}]", totalEachPrice);
+		// 비교 수행
+		return befCash.compareTo(totalEachPrice) >= 0;
+	}
+	
+	
+
+	/**
+	 * @methodName	: insertKakaoAtDataJsonInfo_advc 
+	 * @author		: 이호영
+	 * @date		: 2025. 4. 24.
+	 * @description	: INSERT INTO BIZ_ATTACHMENTS
+	 * @return : void
+	 * @param kakaoSendAdvcListVO
+	 * 
+	 */
+	public void insertKakaoAtDataJsonInfo_advc(List<KakaoSendAdvcVO> kakaoSendAdvcListVO) {
+
+		List<KakaoSendAdvcVO> jsonInfoData = new ArrayList<>(kakaoSendAdvcListVO);
+		jsonInfoData.removeIf(t -> StringUtils.isBlank(t.getJsonStr()));
+		log.info(" + jsonInfoData Insert :: [{}]", jsonInfoData.size());
+		if(jsonInfoData.size() > 0) {
+			kakaoAlimTalkDAO.insertKakaoAtDataJsonInfo_advc(jsonInfoData);
+		}
+
+	}
+	
+	
+
+	/**
+	 * @methodName	: insertKakaoData_advc 
+	 * @author		: 이호영
+	 * @date		: 2025. 3. 20.
+	 * @description	: 카카오 batch 발송 => mj_msg_data
+	 * @return : int
+	 * @param kakaoSendAdvcVOList
+	 * @param parentLoopCount
+	 * @param isJsonNotEmpty
+	 * @param isJsonNameAllSame
+	 * @return
+	 * 
+	 */
+	public int insertKakaoData_advc(List<KakaoSendAdvcVO> kakaoSendAdvcVOList) {
+
+
+		// 시작 시간 측정
+		long totalStartTime = System.currentTimeMillis();
+
+		int totalSize = kakaoSendAdvcVOList.size(); // 총 데이터 개수
+		// Batch 크기 설정 (고정값)
+//		int batchSize = 10000; 465
+		int batchSize = 50000; // 9분 18초
+
+		log.info("총 데이터 개수 :: [{}] ", totalSize);
+		log.info("설정된 Batch 크기 :: [{}] ", batchSize);
+
+		// 총 insert 카운트
+		int instCnt = 0;
+		int batchCount = 0;
+
+		// 각 배치별 실행 시간 기록
+		List<Double> batchExecutionTimes = new ArrayList<>();
+
+	
+		// 첫 번째 배치에서만 삽입했는지 추적하는 플래그
+		for (int i = 0; i < totalSize; i += batchSize) {
+			// Batch 시작 시간 측정
+			long batchStartTime = System.currentTimeMillis();
+
+			// Batch 리스트 생성
+			List<KakaoSendAdvcVO> batchList = kakaoSendAdvcVOList.subList(i, Math.min(i + batchSize, totalSize));
+			System.out.println("Batch 시작 인덱스: " + i);
+
+			// mj_msg_data 테이블 insert
+			int insertedCount = kakaoAlimTalkDAO.insertKakaoAtDataInfo_advc(batchList);
+			
+			/** @kakaoSendUtil.populateSendLists  
+			 * 하단에서 
+			 * getJsonStr 데이터 처리 후 활용
+			 *  */
+			instCnt += insertedCount;
+
+			// Batch 종료 시간 측정 및 실행 시간 계산
+			long batchEndTime = System.currentTimeMillis();
+			double batchExecutionTimeInSeconds = (batchEndTime - batchStartTime) / 1000.0;
+
+			// 실행 시간 기록
+			batchExecutionTimes.add(batchExecutionTimeInSeconds);
+			batchCount++;
+		}
+
+		// 종료 시간 측정
+		long totalEndTime = System.currentTimeMillis();
+
+		// 총 실행 시간 계산 (밀리초 -> 초로 변환)
+		double totalExecutionTimeInSeconds = (totalEndTime - totalStartTime) / 1000.0;
+
+		// 실행 시간 출력
+		log.info("총 배치 실행 횟수 :: [{}] ", batchCount);
+		log.info("batchSize :: [{}] ", batchSize);
+		log.info("총 실행 시간 :: [{}] ", totalExecutionTimeInSeconds + "초");
+		log.info("총 삽입 건수 :: [{}] ", instCnt);
+
+		// 각 배치별 실행 시간 출력
+		for (int k = 0; k < batchExecutionTimes.size(); k++) {
+			System.out.println("배치 " + (k + 1) + " 실행 시간 :: " + batchExecutionTimes.get(k) + "초");
+		}
+
+		return instCnt;
+		
+	}
+	
+	
+	
+	
+
+
+	public void insertKakaoGroupDataTb_advc(int instCnt, KakaoVO kakaoVO, KakaoSendAdvcVO sendVO) throws Exception {
+		// TODO Auto-generated method stub
+
+//		log.info(" + insertKakaoGroupDataTb_advc kakaoVO :: \n[{}]", kakaoVO.toString());;
+//		log.info(" + insertKakaoGroupDataTb_advc kakaoSendAdvcVOList :: \n[{}]", sendVO.toString());
+		
+		
+
+		sendVO.setTemplateContent(kakaoVO.getTemplateContent());
+		sendVO.setMsgGroupCnt(Integer.toString(instCnt));
+		sendVO.setReserveYn(kakaoVO.getReserveYn());
+		sendVO.setBefCash(priceAndPoint.getBefCash(sendVO.getUserId()));
+		sendVO.setBefPoint(priceAndPoint.getBefPoint(sendVO.getUserId()));
+		sendVO.setAdFlag(kakaoVO.getAdFlag());
+		
+		Float eachPrice = Float.parseFloat(sendVO.getEachPrice());
+
+		Float totPrice = eachPrice * instCnt;
+		sendVO.setTotPrice(String.format("%.1f", totPrice));
+		
+		sendVO.setAtDelayYn(kakaoVO.getAtSmishingYn()); 
+		sendVO.setBizKakaoResendOrgnlTxt(kakaoVO.getSubMsgTxt());
+		sendVO.setBizKakaoResendType(sendVO.getSubMsgType());
+		sendVO.setBizKakaoImageType(kakaoVO.getImageType());
+		
+		kakaoAlimTalkDAO.insertKakaoGroupDataTb_advc(sendVO);
+		
+	}
+	
+	
+	
 }
src/main/java/itn/let/kakao/kakaoComm/KakaoVO.java
--- src/main/java/itn/let/kakao/kakaoComm/KakaoVO.java
+++ src/main/java/itn/let/kakao/kakaoComm/KakaoVO.java
@@ -4,6 +4,7 @@
 import java.util.List;
 import java.util.Map;
 
+import itn.com.cmm.MjonFTSendVO;
 import itn.let.mjo.msg.service.MjonMsgVO;
 import lombok.Getter;
 import lombok.Setter;
@@ -259,9 +260,14 @@
 	private String msgResendAllGroupId;
 	private String msgResendAllTmpKey;
 	private String msgResendAllYellowId;
+	
+	private String bizKakaoResendType;
+	private String fileCnt;
 
 	private List<Map<String, String>> varListMap;
 
+	private List<MjonFTSendVO> mjonFTSendVOList = new ArrayList<>();
+	
 	@Override
 	public String toString() {
 		String varListMapString = "[";
@@ -285,6 +291,18 @@
 			varListMapString += sb.toString();
 		}
 		varListMapString += "]";
+		
+		
+		StringBuilder mjonFTListSb = new StringBuilder("[");
+		if (mjonFTSendVOList != null && !mjonFTSendVOList.isEmpty()) {
+			String prefix = "";
+			for (MjonFTSendVO vo : mjonFTSendVOList) {
+				mjonFTListSb.append(prefix).append(vo == null ? "null" : vo.toString());
+				prefix = ", ";
+			}
+		}
+		mjonFTListSb.append("]");
+		
 		
 		return "KakaoSendAdvcVO[" +
 			"\n senderKey=[" + senderKey + "]" +
@@ -312,8 +330,61 @@
 			"\n , varListMap=[" + varListMapString + "]" +
 			"\n , befCash=[" + getBefCash() + "]" +
 			"\n , befPoint=[" + getBefPoint() + "]" +
+			"\n , mjonFTSendVOList=" + mjonFTListSb.toString() +
 			"\n ]";
 	}
 	
+	public String ftToString() {
+		StringBuilder sb = new StringBuilder("KakaoFTSendVO[");
+		sb.append("\n senderKey=[").append(senderKey).append("]");
+		sb.append("\n , imageFileName=[").append(imageFileName).append("]");
+		sb.append("\n , imageType=[").append(imageType).append("]");
+		sb.append("\n , imgTitle=[").append(imgTitle).append("]");
+		sb.append("\n , imgLink=[").append(imgLink).append("]");
+		sb.append("\n , templateContent=[").append(templateContent).append("]");
+		sb.append("\n , templateImageUrl=[").append(templateImageUrl).append("]");
+		sb.append("\n , smsTxtArea=[").append(getSubMsgTxt()).append("]");
+		sb.append("\n , subMsgSendYn=[").append(subMsgSendYn).append("]");
+		sb.append("\n , subMsgTxtReplYn=[").append(subMsgTxtReplYn).append("]");
+		sb.append("\n , subMsgType=[").append(subMsgType).append("]");
+		sb.append("\n , subMsgTxt=[").append(subMsgTxt).append("]");
+		sb.append("\n , reserveYn=[").append(getReserveYn()).append("]");
+		sb.append("\n , menuTopTab=[").append(menuTopTab).append("]");
+		sb.append("\n , bizJsonYn=[").append(bizJsonYn).append("]");
+		sb.append("\n , senderKey=[").append(senderKey).append("]");
+		sb.append("\n , callFrom=[").append(getCallFrom()).append("]");
+		sb.append("\n , kakaoFtPrice=[").append(getEachPrice()).append("]");
+		sb.append("\n , reqDate=[").append(getReqDate()).append("]");
+		sb.append("\n , spamStatus=[").append(getSpamStatus()).append("]");
+		sb.append("\n , txtReplYn=[").append(getTxtReplYn()).append("]");
+		sb.append("\n , atSmishingYn=[").append(getAtSmishingYn()).append("]");
+//		sb.append("\n , tmpBtnSelect=[").append(getTmpBtnSelect()).append("]");
+		StringBuilder btnListSb = new StringBuilder("[");
+		if (buttonVOList != null && !buttonVOList.isEmpty()) {
+			String prefix = "";
+			for (KakaoButtonVO btn : buttonVOList) {
+				btnListSb.append(prefix).append(btn == null ? "null" : btn.toString());
+				prefix = ", ";
+			}
+		}
+		btnListSb.append("]");
+		sb.append("\n , buttonVOList=").append(btnListSb);
+
+
+		// mjonFTSendVOList 내용
+		StringBuilder ftList = new StringBuilder("[");
+		if (mjonFTSendVOList != null && !mjonFTSendVOList.isEmpty()) {
+			String prefix = "";
+			for (MjonFTSendVO vo : mjonFTSendVOList) {
+				ftList.append(prefix).append(vo == null ? "null" : vo.toString());
+				prefix = ", ";
+			}
+		}
+		ftList.append("]");
+		sb.append("\n , mjonFTSendVOList=").append(ftList);
+
+		sb.append("\n]");
+		return sb.toString();
+	}
 	
 }
src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiImageUpload.java
--- src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiImageUpload.java
+++ src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiImageUpload.java
@@ -5,7 +5,10 @@
 import java.io.File;
 import java.io.FileInputStream;
 import java.io.InputStream;
+import java.text.SimpleDateFormat;
+import java.time.LocalDateTime;
 import java.util.ArrayList;
+import java.util.Date;
 import java.util.HashMap;
 import java.util.List;
 import java.util.ListIterator;
@@ -15,6 +18,8 @@
 import javax.imageio.ImageIO;
 import javax.swing.ImageIcon;
 
+import org.apache.commons.io.FilenameUtils;
+import org.apache.commons.lang3.StringUtils;
 import org.apache.http.HttpEntity;
 import org.apache.http.HttpResponse;
 import org.apache.http.client.HttpClient;
@@ -30,16 +35,21 @@
 import org.json.simple.JSONObject;
 import org.json.simple.parser.JSONParser;
 import org.springframework.beans.factory.annotation.Value;
+import org.springframework.http.HttpStatus;
 import org.springframework.stereotype.Component;
 import org.springframework.web.multipart.MultipartFile;
 
 import egovframework.rte.fdl.property.EgovPropertyService;
+import itn.com.cmm.service.EgovFileMngService;
 import itn.com.cmm.service.EgovFileMngUtil;
 import itn.com.cmm.service.FileVO;
 import itn.let.kakao.kakaoComm.KakaoReturnVO;
 import itn.let.kakao.kakaoComm.KakaoVO;
+import itn.let.mail.service.StatusResponse;
 import itn.let.utl.fcc.service.EgovStringUtil;
+import lombok.extern.slf4j.Slf4j;
 
+@Slf4j
 @Component("kakaoApiImageUpload")
 public class KakaoApiImageUpload {
 
@@ -57,7 +67,17 @@
 	
     @Resource(name = "propertiesService")
     protected EgovPropertyService propertyService;
-	
+
+    /** 첨부파일 저장경로 */
+	@Value("#{globalSettings['Globals.file.saveDir']}")
+	private String fileSaveDir;
+
+    @Resource(name="EgovFileMngUtil")
+	private EgovFileMngUtil fileUtil;
+
+    
+    @Resource(name="EgovFileMngService")
+    private EgovFileMngService fileMngService;
 	
 	/**
 	* @Method Name : kakaoApiImageUpload
@@ -219,13 +239,13 @@
 						.addTextBody("bizId", mjonBizId)
 						.addTextBody("apiKey", mjonBizKakaoApiKey)
 						.addTextBody("imageType", kakaoVO.getImageType())
-						.addTextBody("title", kakaoVO.getImgTitle())
-						.addTextBody("link", kakaoVO.getImgLink())
+						.addTextBody("title", "test")
+						.addTextBody("link", "https://maaa.com")
 						.addTextBody("senderKey", kakaoVO.getSenderKey())
 						.addBinaryBody("image", new File(filePath),ContentType.MULTIPART_FORM_DATA,fullFileName)
 						.build();
 				
-				httpPost.setEntity(httpEntity);
+ 				httpPost.setEntity(httpEntity);
 
 		        CloseableHttpResponse response = httpClient.execute(httpPost);
 
@@ -245,6 +265,7 @@
 					String msg = object.get("message").toString();
 					String imgUrl = "";
 					
+					log.info(" : code :: [{}]", code);
 					if(code.equals("200")) {
 						imgUrl = object.get("image").toString();
 					}
@@ -273,6 +294,174 @@
 	}
 	
 	
+	
+
+	/**
+	 * @methodName	: kakaoApiImageUpload_advc 
+	 * @author		: 이호영
+	 * @date		: 2025. 6. 4.
+	 * @description	: kakaoApiImageUpload 수정
+	 * @return : StatusResponse
+	 * @param kakaoVO
+	 * @param files
+	 * @param i
+	 * @return
+	 * @throws Exception 
+	 * 
+	 */
+	public StatusResponse kakaoApiImageUpload_advc(KakaoVO kakaoVO, Map<String, MultipartFile> files, int fileKeyParam) throws Exception {
+//		try {
+			String storePathString = propertyService.getString("Globals.fileStorePath");
+			File saveFolder = new File(storePathString);
+			if (!saveFolder.exists()) saveFolder.mkdirs();
+
+//			for (MultipartFile file : files.values()) {
+			MultipartFile file = files.values().stream().findFirst().orElse(null);
+			if (file == null || file.isEmpty()) {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "유효한 이미지 파일이 없습니다.", LocalDateTime.now());
+			}
+			
+
+			String originalName = file.getOriginalFilename();
+			if (originalName == null || originalName.isEmpty()) {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "파일명이 비어 있습니다.", LocalDateTime.now());
+			}
+			
+			String ext = FilenameUtils.getExtension(originalName).toLowerCase();
+			if (!ext.matches("jpg|jpeg|png")) {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "지원하지 않는 이미지 형식입니다.");
+			}
+
+			long size = file.getSize();
+			if (size > 5 * 1024 * 1024) {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "이미지 용량은 5MB 이내여야 합니다.");
+			}
+
+			BufferedImage image = ImageIO.read(file.getInputStream());
+			if (image == null) {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "이미지를 읽을 수 없습니다.");
+			}
+
+			int width = image.getWidth();
+			int height = image.getHeight();
+			String type = kakaoVO.getImageType();
+
+			if ("W".equals(type)) {
+				if (width != 800 || height != 600) {
+					return new StatusResponse(HttpStatus.BAD_REQUEST, "와이드 이미지는 800x600 사이즈만 허용됩니다.");
+				}
+			} else {
+				float ratio = width / (float) height;
+//					log.info("width : [{}], ",width);
+//					log.info("height : [{}], ",height);
+//					log.info("ratio : [{}], ",ratio);
+				if (width < 500 || ratio < 0.75 || ratio > 2.0) {
+					return new StatusResponse(HttpStatus.BAD_REQUEST, "일반 이미지는 가로 500px 이상, 비율 2:1 이상 또는 3:4 이하만 허용됩니다.");
+				}
+			}
+			
+			String atchFileId = this.saveImgFile(files);
+			
+			
+			
+			
+
+			String newName = EgovStringUtil.getTimeStamp() + fileKeyParam;
+			String filePath = storePathString + File.separator + newName + "." + ext;
+			file.transferTo(new File(filePath));
+
+			// 카카오 API 호출
+			CloseableHttpClient httpClient = HttpClients.createDefault();
+			String apiUrl = mjonBizUrl + "/v3/kakao/image/upload";
+
+			HttpPost httpPost = new HttpPost(apiUrl);
+			/*HttpEntity httpEntity = MultipartEntityBuilder.create()
+					.addTextBody("bizId", mjonBizId)
+					.addTextBody("apiKey", mjonBizKakaoApiKey)
+					.addTextBody("imageType", kakaoVO.getImageType())
+					.addTextBody("title", kakaoVO.getImgTitle())
+					.addTextBody("link", kakaoVO.getImgLink())
+					.addTextBody("senderKey", kakaoVO.getSenderKey())
+					.addBinaryBody("image", new File(filePath), ContentType.MULTIPART_FORM_DATA, newName + "." + ext)
+					.build();
+			*/
+
+			HttpEntity httpEntity = MultipartEntityBuilder.create()
+					.addTextBody("bizId", mjonBizId)
+					.addTextBody("apiKey", mjonBizKakaoApiKey)
+					.addTextBody("imageType", kakaoVO.getImageType())
+					.addTextBody("title", originalName)
+					.addTextBody("link", StringUtils.isEmpty(kakaoVO.getImgLink()) ? "https://" : kakaoVO.getImgLink())
+					.addTextBody("senderKey", kakaoVO.getSenderKey())
+					.addBinaryBody("image", new File(filePath), ContentType.MULTIPART_FORM_DATA, newName + "." + ext)
+					.build();
+			
+			httpPost.setEntity(httpEntity);
+			
+			
+
+			CloseableHttpResponse response = httpClient.execute(httpPost);
+			int statusCode = response.getStatusLine().getStatusCode();
+
+			if (statusCode == 200) {
+				String result = EntityUtils.toString(response.getEntity(), "UTF-8");
+				JSONParser parser = new JSONParser();
+				JSONObject object = (JSONObject) parser.parse(result);
+
+				String code = object.get("code").toString();
+				if ("200".equals(code)) {
+					Map<String, Object> returnMap = new HashMap<>();
+					returnMap.put("imgUrl", object.get("image").toString());
+					returnMap.put("fileName", originalName);
+					returnMap.put("atchFileId", atchFileId);
+					
+					return new StatusResponse(HttpStatus.OK, "이미지 등록이 완료 되었습니다.", returnMap);
+				} else {
+					return new StatusResponse(HttpStatus.BAD_REQUEST, object.get("message").toString(), LocalDateTime.now());
+				}
+			} else {
+				return new StatusResponse(HttpStatus.BAD_REQUEST, "카카오 API 요청 실패", LocalDateTime.now());
+			}
+//			}
+//		} catch (Exception e) {
+//			log.error("kakaoApiImageUpload_advc API Error", e);
+//			return new StatusResponse(HttpStatus.BAD_REQUEST, "친구톡 이미지 등록에 실패했습니다.", LocalDateTime.now());
+//		}
+	}
+
+	
+	
+	private String saveImgFile(Map<String, MultipartFile> files) throws Exception {
+
+
+		
+		String atchFileId = "";
+		String isThumbFile = "";
+		String imagePath = "";
+		String KeyStr = "CANVASIMG_";
+		
+		
+		Date now = new Date();
+		SimpleDateFormat formatDate = new SimpleDateFormat("yyyyMMdd");
+		String fdlDate = formatDate.format(now);
+
+		
+		imagePath = fileSaveDir+"/file/MMS/" + fdlDate;
+			
+		
+		
+		if (!files.isEmpty()) {
+			List<FileVO> result = fileUtil.parseImageFileInf(files, KeyStr, 0, atchFileId, imagePath, isThumbFile);
+			atchFileId = fileMngService.insertFileInfs(result);
+		}
+		
+		
+		return atchFileId;
+	}
+
+
+
+
 	/**
 	* @Method Name : kakaoApiTemplateImageUpload
 	* @작성일 : 2023. 2. 16.
@@ -488,17 +677,20 @@
 			jsonObject.put("apiKey", mjonBizKakaoApiKey);
 			jsonObject.put("imageUrl", kakaoVO.getTemplateImageUrl());
 			
+			log.info("kakaoVO.getTemplateImageUrl() :: [{}]", kakaoVO.getTemplateImageUrl());
+			
 			HttpClient httpClient = HttpClientBuilder.create().build(); 
 			HttpPost httpPost = new HttpPost(sendUrl); 
 			httpPost.setEntity(new StringEntity(jsonObject.toString(), "UTF-8")); 
 			httpPost.addHeader("Content-type", "application/json"); 
-			httpPost.addHeader("Accept", "application/json");
+			httpPost.addHeader("Accept", "application/json"); 
 			
 			HttpResponse response = httpClient.execute(httpPost);
 
 			String result = "";
 			String statusCode = Integer.toString(response.getStatusLine().getStatusCode());
 			
+			log.info(" + statusCode :: [{}]", statusCode);			
 			if(statusCode.equals("200")) {
 				
 				result = EntityUtils.toString(response.getEntity()); 
@@ -528,5 +720,7 @@
 		
 		return kakaoReturnVO;
 	}
+
+
 	
 }
src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiJsonSave.java
--- src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiJsonSave.java
+++ src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiJsonSave.java
@@ -10,11 +10,17 @@
 import java.util.List;
 import java.util.Map;
 
+import org.apache.commons.lang3.StringUtils;
 import org.json.simple.JSONArray;
 import org.json.simple.JSONObject;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.beans.factory.annotation.Value;
 import org.springframework.stereotype.Component;
+
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.node.ArrayNode;
+import com.fasterxml.jackson.databind.node.ObjectNode;
 
 import itn.com.cmm.util.StringUtil;
 import itn.let.kakao.kakaoComm.KakaoButtonVO;
@@ -210,6 +216,139 @@
 	
 	/*
 	 * 친구톡 발송시 이미지, 버튼 추가에 따른 Json 파일 생성 
+	 * 2025.04.18
+	 * 우영두
+	 * 파일은 하나만 생성해서 동일하게 사용함.
+	 * 
+	 * */
+	public String kakaoApiFTJsonSave_advc(KakaoVO kakaoVO) throws JsonProcessingException {
+		// json파일 저장
+		
+
+		ObjectMapper mapper = new ObjectMapper();
+		ObjectNode jo = mapper.createObjectNode();
+		
+		
+		
+		// 버튼
+		if (kakaoVO.getButtonVOList() != null && !kakaoVO.getButtonVOList().isEmpty()) {
+			ArrayNode buttonList = mapper.createArrayNode();
+
+			for (KakaoButtonVO buttonInfoVO : kakaoVO.getButtonVOList()) {
+				ObjectNode button = mapper.createObjectNode();
+				button.put("name", buttonInfoVO.getName());
+				button.put("type", buttonInfoVO.getLinkType());
+
+				switch (buttonInfoVO.getLinkType()) {
+					case "WL":
+						button.put("url_mobile", buttonInfoVO.getLinkMo());
+						button.put("url_pc", buttonInfoVO.getLinkPc());
+						break;
+					case "AL":
+						button.put("scheme_ios", buttonInfoVO.getLinkIos());
+						button.put("scheme_android", buttonInfoVO.getLinkAnd());
+						break;
+					case "BC":
+						// 상담톡
+						break;
+					case "BT":
+						// 봇 전환
+						break;
+				}
+				buttonList.add(button);
+			}
+			jo.set("button", buttonList);
+		}
+		
+		
+		// 이미지
+		String imageType = kakaoVO.getImageType();
+		if (StringUtils.isNotEmpty(imageType)) {
+			ObjectNode image = mapper.createObjectNode();
+			image.put("img_url", kakaoVO.getTemplateImageUrl());
+			image.put("img_link", StringUtils.isNotEmpty(kakaoVO.getImgLink()) ? kakaoVO.getImgLink() : kakaoVO.getTemplateImageUrl());
+			jo.set("image", image);
+
+			// wide 여부
+			if ("W".equals(imageType)) {
+				ObjectNode extra = mapper.createObjectNode();
+				extra.put("wide", "Y");
+				jo.set("extra", extra);
+			}
+			
+		}
+		
+
+		// 문자열로 변환 (이스케이프 없음)
+		return mapper.writeValueAsString(jo);
+		
+		
+		
+		/*
+		
+		
+		// 버튼리스트 JSON 생성
+		JSONArray buttonList = new JSONArray();
+		for(KakaoButtonVO buttonInfoVO : kakaoVO.getButtonVOList()) {
+			JSONObject buttonInfo = new JSONObject();
+			
+			buttonInfo.put("name", buttonInfoVO.getName());
+			buttonInfo.put("type", buttonInfoVO.getLinkType());
+			
+			if(buttonInfoVO.getLinkType().equals("WL")) {
+				buttonInfo.put("url_mobile", buttonInfoVO.getLinkMo());
+				buttonInfo.put("url_pc", buttonInfoVO.getLinkPc());
+			}else if(buttonInfoVO.getLinkType().equals("AL")) {
+				buttonInfo.put("scheme_ios", buttonInfoVO.getLinkIos());
+				buttonInfo.put("scheme_android", buttonInfoVO.getLinkAnd());
+			}else if(buttonInfoVO.getLinkType().equals("BC")) {
+				// 상담톡 진행시 등록해야함
+			}else if(buttonInfoVO.getLinkType().equals("BT")) {
+				// 봇 전환 시 전달
+			}
+			buttonList.add(buttonInfo);
+		}
+		
+		// 강조유형 JSON 생성
+		JSONObject templateImageInfo = new JSONObject();
+		JSONObject templateImageExtInfo = new JSONObject();
+		String imageType = kakaoVO.getImageType();
+		
+		if(StringUtils.isNotEmpty(imageType)) {
+			templateImageInfo.put("img_url", kakaoVO.getTemplateImageUrl());
+			templateImageInfo.put("img_link", StringUtils.isNotEmpty(kakaoVO.getImgLink())  ? kakaoVO.getImgLink() : kakaoVO.getTemplateImageUrl() );
+		}
+		
+		// wide 여부
+		if ("W".equals(imageType)) {
+			ObjectNode extra = mapper.createObjectNode();
+			extra.put("wide", "Y");
+			jo.set("extra", extra);
+		}
+		
+		
+		JSONObject jo = new JSONObject();
+		
+		if(buttonList.size() != 0) {
+			jo.put("button", buttonList);
+		}
+		
+		if(templateImageInfo.size() != 0) {
+			jo.put("image", templateImageInfo);
+		}
+		
+		if(templateImageExtInfo.size() != 0) {
+			jo.put("extra", templateImageExtInfo);
+		}
+		
+		// 입력 json 데이터를 파일로 변경
+		String jsonStr = jo.toString();
+		
+		return jsonStr;*/
+	}
+
+	/*
+	 * 친구톡 발송시 이미지, 버튼 추가에 따른 Json 파일 생성 
 	 * 2024.01.17
 	 * 우영두
 	 * 파일은 하나만 생성해서 동일하게 사용함.
@@ -298,7 +437,7 @@
 			// 입력 json 데이터를 파일로 변경
 			String jsonStr = jo.toString();
 			System.out.println("jsonFileName  : "+jsonFileName);
-		
+			
 			File outPut = new File(jsonFileName);
 			outPut.createNewFile();
 			
src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiProfile.java
--- src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiProfile.java
+++ src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoApiProfile.java
@@ -230,7 +230,8 @@
 				if(code.equals("200")) {
 					
 					JSONObject templateProfile = (JSONObject) object.get("data");
-					
+					System.out.println("templateProfile = " + templateProfile.toJSONString());
+
 					String senderKey = getStringValue(templateProfile, "senderKey");						//발신프로필키
 					String uuid = getStringValue(templateProfile, "uuid");									//카카오톡 채널
 					String name = getStringValue(templateProfile, "name");									//카카오톡 채널 발신프로필 명
src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoFTJsonSave.java
--- src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoFTJsonSave.java
+++ src/main/java/itn/let/kakao/kakaoComm/kakaoApi/KakaoFTJsonSave.java
@@ -30,115 +30,6 @@
 	
 	static String json;
 	
-	@SuppressWarnings("unchecked")
-	public String kakaoApiJsonSave(KakaoVO kakaoVO, String[] varValInfo) {
-		// json파일 저장
-		
-		
-		Date nowDate = new Date();
-		SimpleDateFormat todayFrom = new SimpleDateFormat("yyyyMMdd");
-		SimpleDateFormat timeFrom = new SimpleDateFormat("HHmmss");
-		String jsonFileName = mjonBizJsonDir+"/"+kakaoVO.getUserId()+"/"+todayFrom.format(nowDate)+"/"+kakaoVO.getSendType(); // 아이디/날짜/타입
-		
-		String fileName = timeFrom.format(nowDate)+"_"+kakaoVO.getDestPhone()+".json";
-		
-		try {
-			
-			File userIdFile = new File(jsonFileName);
-			if(!userIdFile.exists()) {
-				userIdFile.mkdirs(); // 없으면 하위 디렉토리 까지 생성
-				jsonFileName = jsonFileName +"/"+fileName;
-			}else {
-				
-				jsonFileName = jsonFileName +"/"+fileName;
-				System.out.println("jsonFileName  : "+jsonFileName);
-				File file1 = new File(jsonFileName);
-				if (file1.isFile()) {
-					return jsonFileName;
-				}
-			}
-			
-			
-			
-			
-			KakaoReturnVO templateDetail = kakaoApiTemplate.selectKakaoApiTemplateDetail(kakaoVO);
-			
-			// 버튼리스트 JSON 생성
-			JSONArray buttonList = new JSONArray();
-			
-			// 버튼 수량 체크 후 진행
-			for(KakaoButtonVO buttonInfoVO : templateDetail.getButtonList()) {
-				JSONObject buttonInfo = new JSONObject();
-				
-				buttonInfo.put("name", buttonInfoVO.getName());
-				buttonInfo.put("type", buttonInfoVO.getLinkType());
-				
-				if(buttonInfoVO.getLinkType().equals("WL")) {
-					buttonInfo.put("url_mobile", buttonInfoVO.getLinkMo());
-					buttonInfo.put("url_pc", buttonInfoVO.getLinkPc());
-				}else if(buttonInfoVO.getLinkType().equals("AL")) {
-					buttonInfo.put("scheme_ios", buttonInfoVO.getLinkIos());
-					buttonInfo.put("scheme_android", buttonInfoVO.getLinkAnd());
-				}else if(buttonInfoVO.getLinkType().equals("BC")) {
-					// 상담톡 진행시 등록해야함
-				}else if(buttonInfoVO.getLinkType().equals("BT")) {
-					// 봇 전환 시 전달
-				}
-				buttonList.add(buttonInfo);
-			}
-			
-			
-			
-			
-			// Image JSON 생성
-			JSONObject imageJson = new JSONObject();
-			// img형과 wide 형인경우만 등록 (if문으로 제어)
-			imageJson.put("img_url", "등록된 이미지 URL");
-			imageJson.put("img_link", "이동 페이지 URL");
-			
-			
-			// Wide JSON 생성
-			JSONObject wideJson = new JSONObject();
-			// wide 형인경우만 등록(if문으로 제어)
-			wideJson.put("wide", "Y");
-						
-						
-						
-			
-			JSONObject jo = new JSONObject();
-			
-			if(imageJson.size() != 0) {
-				jo.put("image", imageJson);
-			}
-			if(wideJson.size() != 0) {
-				jo.put("extra", wideJson);
-			}
-			
-			if(buttonList.size() != 0) {
-				jo.put("button", buttonList);
-			}
-			
-			
-			
-			
-			// 입력 json 데이터를 파일로 변경
-			String jsonStr = jo.toString();
-			System.out.println("jsonFileName  : "+jsonFileName);
-		
-			File outPut = new File(jsonFileName);
-			outPut.createNewFile();
-			
-			BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outPut), "euc-kr"));
-			bw.write(jsonStr);
-			bw.close();
-			
-		} catch (IOException e) {
-			System.out.println("json 생성 실패");
-			e.printStackTrace();
-		}
-		return jsonFileName;
-	}
-	
 	
 	@SuppressWarnings("unchecked")
 	public String kakaoApiJsonSave(KakaoVO kakaoVO) {
src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkDAO.java
--- src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkDAO.java
+++ src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkDAO.java
@@ -108,4 +108,9 @@
 	public void insertKakaoGroupDataTb_advc(KakaoSendAdvcVO sendVO) {
 		insert("kakaoAlimTalkDAO.insertKakaoGroupDataTb_advc", sendVO);
 	}
+
+	public KakaoVO selectBizLog(String bizUmid) {
+		return (KakaoVO) select("kakaoAlimTalkDAO.selectBizLog", bizUmid); 
+//		return (KakaoVO) select("kakaoAlimTalkDAO.selectKakaoAtUmid", bizUmid); 
+	}
 }
src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkServiceImpl.java
--- src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkServiceImpl.java
+++ src/main/java/itn/let/kakao/user/kakaoAt/service/impl/KakaoAlimTalkServiceImpl.java
@@ -29,6 +29,7 @@
 import itn.com.cmm.LoginVO;
 import itn.com.cmm.MjonMsgSendVO;
 import itn.com.utl.fcc.service.EgovStringUtil;
+import itn.let.kakao.kakaoComm.BizKakaoPriceVO;
 import itn.let.kakao.kakaoComm.KakaoSendAdvcVO;
 import itn.let.kakao.kakaoComm.KakaoSendUtil;
 import itn.let.kakao.kakaoComm.KakaoVO;
@@ -45,6 +46,7 @@
 import itn.let.mjo.msgholiday.service.impl.MsgHolidayDAO;
 import itn.let.mjo.pay.service.MjonPayService;
 import itn.let.mjo.pay.service.MjonPayVO;
+import itn.let.mjo.pay.service.impl.MjonPayDAO;
 import itn.let.module.base.PriceAndPoint;
 import itn.let.sym.site.service.JoinSettingVO;
 import itn.let.sym.site.service.impl.SiteManagerDAO;
@@ -86,6 +88,9 @@
 	/** userManageService */
 	@Resource(name = "userManageService")
 	private EgovUserManageService userManageService;
+
+	@Autowired
+	private MjonPayDAO mjonPayDAO;
 	
 	@Autowired
 	KakaoSendUtil kakaoSendUtil;
@@ -95,6 +100,11 @@
 	
 	@Autowired
 	private PriceAndPoint priceAndPoint;
+
+	final String KAKAO_SUCCESS_CODE = "7000";
+	final String RESEND_YN_YES = "Y";
+	final String SMS_SUCCESS_CODE = "4100"; // SMS 성공 코드
+	final String MMS_SUCCESS_CODE = "6600"; // MMS 성공 코드
 	
 	//발신프로필 상태값 변경(삭제/복구 기능)
 	@Override
@@ -819,44 +829,108 @@
 	}
 	
 	
-	/*
-	 * 카카오 친구톡 발송 실패에 따른 금액 환불 처리
-	 * 카카오 친구톡 대체문자 선택에 대해 성공시 친구톡과 문자 간 금액 차액의 환불도 처리 됨. 
-	 * */
+	/**
+	 * @ 카카오 친구톡 환불 
+	 */
 	@Override
 	public void selectKakaoFtSentRefundList() throws Exception {
 		
-		/**
-		 * 1. 카카오 FT 전송성공 확인
-		 * 2. 카카오 FT 전송실패, 대채문자 전송확인
-		 * 3. 카카오 FT 전송 실패 확인
-		 */
 		List<KakaoVO> kakaoFtSentRefundList = kakaoAlimTalkDAO.selectKakaoFtSentRefundList();
 		
-		for(KakaoVO vo : kakaoFtSentRefundList) {
-			System.out.println(vo.getMsgGroupId() +"________결과 : " +vo.getRsltCode() +"     대체문자 전송 : "+vo.getSubMsgSendYn());
+		for(KakaoVO kakaoVO : kakaoFtSentRefundList) {
+			System.out.println(kakaoVO.getMsgGroupId() +"________결과 : " +kakaoVO.getRsltCode() +"     대체문자 전송 : "+kakaoVO.getSubMsgSendYn());
 			
-			if(vo.getRsltCode().equals("7000")) {//친구톡 발송 성공시
+			if(KAKAO_SUCCESS_CODE.equals(kakaoVO.getRsltCode())) {//친구톡 발송 성공시
 			
-				kakaoAlimTalkDAO.updateKakaoFtSend(vo);
+				processFtSendCharge(kakaoVO); 
 				
-			}else if(vo.getSubMsgSendYn().equals("Y")) {//친구톡 발송 실패 했을 경우
-				
-				//대체문자 발송 UMID 번호 조회 - 알림톡 쿼리 동일하게 사용
-				KakaoVO info = kakaoAlimTalkDAO.selectKakaoAtUmid(vo);
-				
-				System.out.println("대체문자 전송 : " + info.getBizUmid());
-				
-				if (info.getBizUmid() != null) {//대체문자 발송 완료인 경우
-					kakaoAlimTalkDAO.updateKakaoAtSubMsgSend(vo);
-				}else {
-					kakaoAlimTalkDAO.updateKakaoFtNotSend(vo);	 
-				}
+			}else if(RESEND_YN_YES.equals(kakaoVO.getSubMsgSendYn())) {
+				//친구톡 발송 실패 했고, 대체문자 발송 했을 경우
+				handleAlternativeMessageScenario(kakaoVO);
 				
 			}else {
-				kakaoAlimTalkDAO.updateKakaoFtNotSend(vo);
+				handleRefund(kakaoVO, "카카오 친구톡 전송 실패로 인한 결재 금액 환불");
 			}
 		}
+	}
+	
+
+	private void processFtSendCharge(KakaoVO kakaoVO) throws Exception {
+		
+
+		if(RESEND_YN_YES.equals(kakaoVO.getSubMsgSendYn())) {
+
+			BizKakaoPriceVO bizKakaoPriceVO = mjonPayDAO.selectBizKakaoPrice(kakaoVO.getMsgGroupId());
+			
+			BigDecimal sendPrice = null;
+			if(StringUtils.isEmpty(kakaoVO.getBizKakaoImageType())){
+				sendPrice = new BigDecimal(bizKakaoPriceVO.getBizKakaoFtPrice());
+			}else if("I".equals(kakaoVO.getBizKakaoImageType())){
+				sendPrice = new BigDecimal(bizKakaoPriceVO.getBizKakaoFtImgPrice());
+			}else if("W".equals(kakaoVO.getBizKakaoImageType())){
+				sendPrice = new BigDecimal(bizKakaoPriceVO.getBizKakaoFtWideImgPrice());
+			}
+
+			// 예시: 각 건당 가격이 이미 String 형태라면 변환
+			BigDecimal eachPrice = new BigDecimal(kakaoVO.getEachPrice());
+
+			// 차이 계산
+			BigDecimal diffPrice = eachPrice.subtract(sendPrice);
+			
+			
+			if (diffPrice.compareTo(BigDecimal.ZERO) > 0) {
+				String result = diffPrice.toString();
+				kakaoVO.setEachPrice(result);
+
+				handleRefund(kakaoVO, "카카오 친구톡 전송으로 인한 결재 차액 환불");
+			}
+			
+		}else {
+			mjonMsgDAO.updateRefundY(kakaoVO);
+			
+		}
+		// TODO Auto-generated method stub
+		
+	}
+
+	private void handleAlternativeMessageScenario(KakaoVO kakaoVO) throws Exception {
+		KakaoVO bizLogVO = kakaoAlimTalkDAO.selectBizLog(kakaoVO.getBizUmid());
+		log.info("대체문자 전송 UMID: {}", kakaoVO.getBizUmid());
+	
+		// 대체문자가 성공적으로 발송되었는지 확인 (SMS 또는 MMS 성공)
+		boolean isAlternativeMessageSuccessful = false;
+		if (bizLogVO != null && StringUtils.isNotEmpty(bizLogVO.getBizLogCallStatus())) {
+			if (SMS_SUCCESS_CODE.equals(bizLogVO.getBizLogCallStatus()) 
+					|| MMS_SUCCESS_CODE.equals(bizLogVO.getBizLogCallStatus())) {
+				isAlternativeMessageSuccessful = true;
+			}
+		}
+
+		log.info("bizLogVO.getBizLogCallStatus() :: [{}]", bizLogVO.getBizLogCallStatus());
+		log.info("isAlternativeMessageSuccessful :: [{}]", isAlternativeMessageSuccessful);
+		
+		// 대체문자 성공이면 환불 완료처리면 한다.
+		if (isAlternativeMessageSuccessful) {
+			mjonMsgDAO.updateRefundY(kakaoVO);
+		} else {
+			// 대체문자 발송 실패 된 경우 (친구톡 비용 환불 필요)
+			handleRefund(kakaoVO, "카카오 친구톡 전송 실패로 인한 결재 금액 환불");
+		}
+	}
+	
+	private void handleRefund(KakaoVO vo, String msg) throws Exception {
+		// mj_cash 테이블에 환불 내역 추가 및 회원 금액 업데이트
+		// eachPrice는 환불될 금액이므로 양수여야 합니다.
+		priceAndPoint.insertCashAndPoint(
+			vo.getUserId(),
+			Float.parseFloat(vo.getEachPrice()), // 환불 금액은 양수
+			msg,
+			vo.getMsgGroupId(),
+			vo.getUserData()
+		);
+		
+		// 해당 row 환불 처리 (mj_msg_data.REFUND_YN = 'Y')
+		mjonMsgDAO.updateRefundY(vo);
 	}
 
 	@Override
@@ -913,7 +987,7 @@
 		
 		
 /** @전송금액 확인 --------------------------------------------------*/
-		if (!isCashSufficient(userId, kakaoSendAdvcListVO)) {
+		if (!kakaoSendUtil.isCashSufficient(userId, kakaoSendAdvcListVO)) {
 			log.error("Insufficient balance for message sending.");
 			return new StatusResponse(HttpStatus.BAD_REQUEST, "문자 발송에 필요한 보유 잔액이 부족 합니다.");
 		}
@@ -921,7 +995,8 @@
 		
 
 /** @json파일이 있을 떄 biz_attachments insert */
-		this.insertKakaoAtDataJsonInfo_advc(kakaoSendAdvcListVO);
+		kakaoSendUtil.insertKakaoAtDataJsonInfo_advc(kakaoSendAdvcListVO);
+//		this.insertKakaoAtDataJsonInfo_advc(kakaoSendAdvcListVO);
 		
 		
 		Map<String, List<KakaoSendAdvcVO>> priceGroupedMessages = kakaoSendAdvcListVO.stream()
@@ -941,7 +1016,7 @@
 			
 
 			// 발송 데이터 삽입
-			int instCnt = this.insertKakaoData_advc(groupedMsgList);
+			int instCnt = kakaoSendUtil.insertKakaoData_advc(groupedMsgList);
 //			int instCnt = 6;
 			
 			if(instCnt > 0) {
@@ -951,7 +1026,7 @@
 				KakaoSendAdvcVO sendVO = groupedMsgList.get(0);
 
 /** @groupData 테이블 insert */
-				this.insertKakaoGroupDataTb_advc(instCnt, kakaoVO, sendVO);
+				kakaoSendUtil.insertKakaoGroupDataTb_advc(instCnt, kakaoVO, sendVO);
 
 
 /** @biz_kakao_price에 insert (대체문자 환불관련 테이블)*/ 
@@ -1032,146 +1107,26 @@
 	
 
 
-	private void insertKakaoAtDataJsonInfo_advc(List<KakaoSendAdvcVO> kakaoSendAdvcListVO) {
-		// TODO Auto-generated method stub
-
-		// 측정할 메소드 호출 전 시간 기록
-		List<KakaoSendAdvcVO> jsonInfoData = new ArrayList<>(kakaoSendAdvcListVO);
-		jsonInfoData.removeIf(t -> StringUtils.isBlank(t.getJsonStr()));
-		log.info(" + jsonInfoData Insert :: [{}]", jsonInfoData.size());
-		if(jsonInfoData.size() > 0) {
-			kakaoAlimTalkDAO.insertKakaoAtDataJsonInfo_advc(jsonInfoData);
-		}
-
-	}
-
-	private void insertKakaoGroupDataTb_advc(int instCnt, KakaoVO kakaoVO, KakaoSendAdvcVO sendVO) throws Exception {
-		// TODO Auto-generated method stub
-
-//		log.info(" + insertKakaoGroupDataTb_advc kakaoVO :: \n[{}]", kakaoVO.toString());;
-//		log.info(" + insertKakaoGroupDataTb_advc kakaoSendAdvcVOList :: \n[{}]", sendVO.toString());
-		
-		sendVO.setMsgGroupCnt(Integer.toString(instCnt));
-		sendVO.setReserveYn(kakaoVO.getReserveYn());
-		sendVO.setBefCash(priceAndPoint.getBefCash(sendVO.getUserId()));
-		sendVO.setBefPoint(priceAndPoint.getBefPoint(sendVO.getUserId()));
-		
-		Float eachPrice = Float.parseFloat(sendVO.getEachPrice());
-
-		Float totPrice = eachPrice * instCnt;
-		sendVO.setTotPrice(String.format("%.1f", totPrice));
-		
-		sendVO.setAtDelayYn(kakaoVO.getAtSmishingYn()); 
-		sendVO.setBizKakaoResendOrgnlTxt(kakaoVO.getSubMsgTxt());
-		sendVO.setBizKakaoResendType(sendVO.getSubMsgType());
-		
-		kakaoAlimTalkDAO.insertKakaoGroupDataTb_advc(sendVO);
-		
-	}
-
-	/**
-	 * @methodName	: insertKakaoData_advc 
-	 * @author		: 이호영
-	 * @date		: 2025. 3. 20.
-	 * @description	: 카카오 batch 발송 => mj_msg_data
-	 * @return : int
-	 * @param kakaoSendAdvcVOList
-	 * @param parentLoopCount
-	 * @param isJsonNotEmpty
-	 * @param isJsonNameAllSame
-	 * @return
-	 * 
-	 */
-	private int insertKakaoData_advc(List<KakaoSendAdvcVO> kakaoSendAdvcVOList) {
-
-
-		// 시작 시간 측정
-		long totalStartTime = System.currentTimeMillis();
-
-		int totalSize = kakaoSendAdvcVOList.size(); // 총 데이터 개수
-		// Batch 크기 설정 (고정값)
-//		int batchSize = 10000; 465
-		int batchSize = 50000; // 9분 18초
-
-		log.info("총 데이터 개수 :: [{}] ", totalSize);
-		log.info("설정된 Batch 크기 :: [{}] ", batchSize);
-
-		// 총 insert 카운트
-		int instCnt = 0;
-		int batchCount = 0;
-
-		// 각 배치별 실행 시간 기록
-		List<Double> batchExecutionTimes = new ArrayList<>();
-
-	
-		// 첫 번째 배치에서만 삽입했는지 추적하는 플래그
-		for (int i = 0; i < totalSize; i += batchSize) {
-			// Batch 시작 시간 측정
-			long batchStartTime = System.currentTimeMillis();
-
-			// Batch 리스트 생성
-			List<KakaoSendAdvcVO> batchList = kakaoSendAdvcVOList.subList(i, Math.min(i + batchSize, totalSize));
-			System.out.println("Batch 시작 인덱스: " + i);
-
-			// mj_msg_data 테이블 insert
-			int insertedCount = kakaoAlimTalkDAO.insertKakaoAtDataInfo_advc(batchList);
-			
-			/** @kakaoSendUtil.populateSendLists  
-			 * 하단에서 
-			 * getJsonStr 데이터 처리 후 활용
-			 *  */
-			instCnt += insertedCount;
-
-			// Batch 종료 시간 측정 및 실행 시간 계산
-			long batchEndTime = System.currentTimeMillis();
-			double batchExecutionTimeInSeconds = (batchEndTime - batchStartTime) / 1000.0;
-
-			// 실행 시간 기록
-			batchExecutionTimes.add(batchExecutionTimeInSeconds);
-			batchCount++;
-		}
-
-		// 종료 시간 측정
-		long totalEndTime = System.currentTimeMillis();
-
-		// 총 실행 시간 계산 (밀리초 -> 초로 변환)
-		double totalExecutionTimeInSeconds = (totalEndTime - totalStartTime) / 1000.0;
-
-		// 실행 시간 출력
-		log.info("총 배치 실행 횟수 :: [{}] ", batchCount);
-		log.info("batchSize :: [{}] ", batchSize);
-		log.info("총 실행 시간 :: [{}] ", totalExecutionTimeInSeconds + "초");
-		log.info("총 삽입 건수 :: [{}] ", instCnt);
-
-		// 각 배치별 실행 시간 출력
-		for (int k = 0; k < batchExecutionTimes.size(); k++) {
-			System.out.println("배치 " + (k + 1) + " 실행 시간 :: " + batchExecutionTimes.get(k) + "초");
-		}
-
-		return instCnt;
-		
-	}
-
-	// 보유 금액이 충분한지 확인하는 메서드
-	private boolean isCashSufficient(String userId, List<KakaoSendAdvcVO> kakaoSendAdvcListVO) throws Exception {
-		
-		
-		String userMoney = priceAndPoint.getBefCash(userId);
-		// 쉼표 제거
-		userMoney = userMoney.replace(",", "");
-
-		// 사용자 보유 금액 BigDecimal 변환 (HALF_EVEN 적용)
-		BigDecimal befCash = new BigDecimal(userMoney).setScale(2, RoundingMode.HALF_EVEN);
-		
-		// 총 메시지 금액 계산 (HALF_EVEN 적용)
-		BigDecimal totalEachPrice = kakaoSendAdvcListVO.stream()
-			.map(msg -> new BigDecimal(String.valueOf(msg.getEachPrice()))) // 변환 오류 방지
-			.reduce(BigDecimal.ZERO, BigDecimal::add)
-			.setScale(2, RoundingMode.HALF_EVEN); // 일관성 유지
-
-		// 비교 수행
-		return befCash.compareTo(totalEachPrice) >= 0;
-	}
+//	// 보유 금액이 충분한지 확인하는 메서드
+//	private boolean isCashSufficient(String userId, List<KakaoSendAdvcVO> kakaoSendAdvcListVO) throws Exception {
+//		
+//		
+//		String userMoney = priceAndPoint.getBefCash(userId);
+//		// 쉼표 제거
+//		userMoney = userMoney.replace(",", "");
+//
+//		// 사용자 보유 금액 BigDecimal 변환 (HALF_EVEN 적용)
+//		BigDecimal befCash = new BigDecimal(userMoney).setScale(2, RoundingMode.HALF_EVEN);
+//		
+//		// 총 메시지 금액 계산 (HALF_EVEN 적용)
+//		BigDecimal totalEachPrice = kakaoSendAdvcListVO.stream()
+//			.map(msg -> new BigDecimal(String.valueOf(msg.getEachPrice()))) // 변환 오류 방지
+//			.reduce(BigDecimal.ZERO, BigDecimal::add)
+//			.setScale(2, RoundingMode.HALF_EVEN); // 일관성 유지
+//
+//		// 비교 수행
+//		return befCash.compareTo(totalEachPrice) >= 0;
+//	}
 	
 	
 	
 
src/main/java/itn/let/kakao/user/kakaoFt/service/KakaoFriendsTalkService.java (added)
+++ src/main/java/itn/let/kakao/user/kakaoFt/service/KakaoFriendsTalkService.java
@@ -0,0 +1,12 @@
+package itn.let.kakao.user.kakaoFt.service;
+
+import javax.servlet.http.HttpServletRequest;
+
+import itn.let.kakao.kakaoComm.KakaoVO;
+import itn.let.mail.service.StatusResponse;
+
+public interface KakaoFriendsTalkService {
+
+	StatusResponse insertKakaoFtSandAjax_advc(KakaoVO kakaoVO, HttpServletRequest request) throws Exception;
+
+}
 
src/main/java/itn/let/kakao/user/kakaoFt/service/impl/KakaoFriendsTalkServiceImpl.java (added)
+++ src/main/java/itn/let/kakao/user/kakaoFt/service/impl/KakaoFriendsTalkServiceImpl.java
@@ -0,0 +1,249 @@
+package itn.let.kakao.user.kakaoFt.service.impl;
+
+import java.time.Duration;
+import java.time.Instant;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.stream.Collectors;
+
+import javax.annotation.Resource;
+import javax.servlet.http.HttpServletRequest;
+
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.http.HttpStatus;
+import org.springframework.stereotype.Service;
+
+import egovframework.rte.fdl.cmmn.EgovAbstractServiceImpl;
+import egovframework.rte.fdl.idgnr.EgovIdGnrService;
+import egovframework.rte.fdl.security.userdetails.util.EgovUserDetailsHelper;
+import itn.com.cmm.LoginVO;
+import itn.com.utl.fcc.service.EgovStringUtil;
+import itn.let.kakao.kakaoComm.KakaoSendAdvcVO;
+import itn.let.kakao.kakaoComm.KakaoSendUtil;
+import itn.let.kakao.kakaoComm.KakaoVO;
+import itn.let.kakao.user.kakaoAt.service.impl.KakaoAlimTalkDAO;
+import itn.let.kakao.user.kakaoFt.service.KakaoFriendsTalkService;
+import itn.let.mail.service.StatusResponse;
+import itn.let.mjo.mjocommon.MjonCommon;
+import itn.let.mjo.msg.service.MjonMsgVO;
+import itn.let.mjo.msg.service.impl.MjonMsgDAO;
+import itn.let.module.base.PriceAndPoint;
+import itn.let.uss.umt.service.EgovUserManageService;
+import lombok.extern.slf4j.Slf4j;
+
+@Slf4j
+@Service("KakaoFriendsTalkService")
+public class KakaoFriendsTalkServiceImpl  extends EgovAbstractServiceImpl implements KakaoFriendsTalkService{
+
+	@Resource(name = "egovFriendstalkTemplateIdService")
+	private EgovIdGnrService idgenFriendTalkTmpId;
+
+	@Resource(name="kakaoFriendsTalkTemplateDAO")
+	private KakaoFriendsTalkTemplateDAO kakaoFriendsTalkTemplateDAO;
+
+	@Resource(name="mjonMsgDAO")
+	private MjonMsgDAO mjonMsgDAO;
+	
+	/** userManageService */
+	@Resource(name = "userManageService")
+	private EgovUserManageService userManageService;
+
+	@Resource(name = "egovMjonMsgGroupIdGnrService")
+	private EgovIdGnrService idgenMjonMsgGroupId;
+
+	
+	@Resource(name="kakaoAlimTalkDAO")
+	private KakaoAlimTalkDAO kakaoAlimTalkDAO;
+	
+	@Autowired
+	KakaoSendUtil kakaoSendUtil;
+	
+	@Autowired
+	private MjonCommon mjonCommon;
+	
+	@Autowired
+	private PriceAndPoint priceAndPoint;
+	
+	@Override
+	public StatusResponse insertKakaoFtSandAjax_advc(KakaoVO kakaoVO, HttpServletRequest request) throws Exception {
+		StatusResponse statusResponse = new StatusResponse();
+		
+		log.info(" + kakaoVO.toString() :: [{}]", kakaoVO.toString()); 
+		
+		log.info(" + kakaoVO.toString() :: [{}]", kakaoVO.ftToString()); 
+		
+
+		
+		// 측정할 메소드 호출 전 시간 기록
+		Instant start = Instant.now();
+//		KakaoSendAdvcVO
+		
+		Map<String, Object> returnMap = new HashMap<>();
+
+		LoginVO loginVO = EgovUserDetailsHelper.isAuthenticated()
+				? (LoginVO) EgovUserDetailsHelper.getAuthenticatedUser()
+				: null;
+		String userId = loginVO == null ? "" : EgovStringUtil.isNullToString(loginVO.getId());
+
+		if (userId.equals("")) {
+			return new StatusResponse(HttpStatus.BAD_REQUEST, "로그인 후 이용이 가능합니다.");
+		}
+		
+		kakaoVO.setUserId(userId);
+
+		/**
+		 * 회원 정지된 상태이면 문자 발송이 안되도록 처리함 현재 로그인 세션도 만료 처리함
+		 */
+		boolean mberSttus = userManageService.selectUserStatusInfo(userId);
+		if (!mberSttus) {
+			request.getSession().invalidate();
+			// UNAUTHORIZED : 인증되지 않은 사용자가 접근하려고 할 때
+			return new StatusResponse(HttpStatus.UNAUTHORIZED,
+					"현재 고객님께서는 문자온 서비스 이용이 정지된 상태로 친구톡을 발송하실 수 없습니다. 이용정지 해제를 원하시면 고객센터로 연락주시기 바랍니다.");
+		}
+		
+
+		/**
+		 * 친구톡은 발송 시간 제약이 있음
+		 */
+		if(kakaoSendUtil.isRestrictedFriendTalkTime(kakaoSendUtil.resolveBaseDate(kakaoVO))) {
+			// UNAUTHORIZED : 인증되지 않은 사용자가 접근하려고 할 때
+			return new StatusResponse(HttpStatus.BAD_REQUEST,
+					"친구톡은 20시 50분부터 익일 08시까지 발송이 제한됩니다.");
+		}
+		
+
+/** @isHolidayNotified 
+ * @false : 알림 X
+ * @true  : 알림 O */
+		boolean isNotified = mjonCommon.processUserAndCheckAT(kakaoVO);
+		
+		
+		
+/** @카카오톡 전송 list 셋팅 -------------------------------------------*/
+		List<KakaoSendAdvcVO> kakaoSendAdvcListVO = kakaoSendUtil.populateSendListsFT(kakaoVO, isNotified, statusResponse);
+		if (statusResponse.getStatus() != null && !statusResponse.getStatus().equals(HttpStatus.OK)) {
+			log.error(" + populateSendLists 처리 중 오류 발생: {}", statusResponse.getMessage());
+			return statusResponse;
+		}
+		
+		
+		
+/** @전송금액 확인 --------------------------------------------------*/
+		if (!kakaoSendUtil.isCashSufficient(userId, kakaoSendAdvcListVO)) {
+			log.error("Insufficient balance for message sending.");
+			return new StatusResponse(HttpStatus.BAD_REQUEST, "문자 발송에 필요한 보유 잔액이 부족 합니다.");
+		}
+		
+
+/** @json파일이 있을 떄 biz_attachments insert */
+		kakaoSendUtil.insertKakaoAtDataJsonInfo_advc(kakaoSendAdvcListVO);
+		
+		
+
+		Map<String, List<KakaoSendAdvcVO>> priceGroupedMessages = kakaoSendAdvcListVO.stream()
+				.collect(Collectors.groupingBy(KakaoSendAdvcVO::getEachPrice));
+		// instTotalCnt : 화면에서 보여줄 총 발송건수
+		int instTotalCnt = 0;
+		
+
+		// 임시
+		List<String> nextMsgGroupIdA = new ArrayList<>();
+			// 대안: entrySet() 직접 사용
+		for (Map.Entry<String, List<KakaoSendAdvcVO>> entry : priceGroupedMessages.entrySet()) {
+			    // entry 사용
+
+			List<KakaoSendAdvcVO> groupedMsgList = entry.getValue(); // 해당 가격의 메시지 리스트
+
+			String nextMsgGroupId = idgenMjonMsgGroupId.getNextStringId();
+			groupedMsgList.forEach(t -> t.setMsgGroupId(nextMsgGroupId));
+			
+
+			// 발송 데이터 삽입
+			int instCnt = kakaoSendUtil.insertKakaoData_advc(groupedMsgList);
+//			int instCnt = 6;
+			
+			if(instCnt > 0) {
+
+				instTotalCnt += instCnt;
+
+				KakaoSendAdvcVO sendVO = groupedMsgList.get(0);
+
+/** @groupData 테이블 insert */
+				kakaoSendUtil.insertKakaoGroupDataTb_advc(instCnt, kakaoVO, sendVO);
+
+
+				
+/** @biz_kakao_price에 insert (대체문자 환불관련 테이블)*/
+				priceAndPoint.insertBizFtKakaoPrice(kakaoVO.getUserId(), sendVO.getMsgGroupId()); 
+				
+				
+				priceAndPoint.insertCashAndPoint(kakaoVO.getUserId()
+						, -Float.parseFloat(sendVO.getTotPrice())
+						, "카카오 친구톡 총 "+groupedMsgList.size()+"건 중 " + instCnt + "건 발송"
+						, nextMsgGroupId
+					);
+
+
+/** @SLACK발송 */ 
+				/** @발송조건이되면 발송 */ 
+				if(isNotified) {
+					mjonCommon.getAdminKakaoAtSendSlack(sendVO);
+				}else if("Y".equals(kakaoVO.getAtSmishingYn())){
+				/** @발송조건이 안되면 DB INSERT */ 
+					mjonMsgDAO.insertSpamPassMsgData(MjonMsgVO.builder()
+							.msgGroupId(nextMsgGroupId)
+							.userId(kakaoVO.getUserId())
+							.reqDate(kakaoVO.getReqDate())
+							.smsTxt(groupedMsgList.get(0).getTemplateContent())
+							.totalCallCnt(instCnt)
+							.callFrom(kakaoVO.getCallFrom())
+							.msgType("8")
+							.reserveYn(kakaoVO.getReserveYn())
+							.build()
+						);
+				}
+				
+				nextMsgGroupIdA.add(nextMsgGroupId);
+				
+			}
+			
+		}
+
+		returnMap.put("resultSts", instTotalCnt);
+		returnMap.put("reserYn", kakaoVO.getReserveYn());
+		returnMap.put("groupIds", nextMsgGroupIdA);
+		
+		
+		// 측정할 메소드 호출 후 시간 기록
+		Instant end = Instant.now();
+
+		log.info(" + start :: [{}]", start);
+		// 실행 시간 계산 (나노초, 밀리초, 초)
+		long seconds = Duration.between(start, end).getSeconds();
+		log.info("메소드 실행 시간 (초): {} s", seconds);
+		double minutes = seconds / 60.0; // 소수점 포함을 위해 60.0으로 나눔
+
+		returnMap.put("second", seconds+" s");
+		returnMap.put("minutes", minutes+" min");
+		
+
+//		System.out.println("메소드 실행 시간 (분): " + minutes + " min");
+		
+		
+		
+		
+//		priceAndPoint.getBefCash(userId);
+		
+		
+		
+
+		statusResponse.setStatus(HttpStatus.OK);
+		statusResponse.setObject(returnMap);
+		
+		return statusResponse;
+	}
+	
+}
src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkSendController.java
--- src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkSendController.java
+++ src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkSendController.java
@@ -13,9 +13,11 @@
 import javax.servlet.http.HttpServletRequest;
 
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.http.ResponseEntity;
 import org.springframework.stereotype.Controller;
 import org.springframework.ui.ModelMap;
 import org.springframework.web.bind.annotation.ModelAttribute;
+import org.springframework.web.bind.annotation.RequestBody;
 import org.springframework.web.bind.annotation.RequestMapping;
 import org.springframework.web.servlet.ModelAndView;
 
@@ -29,7 +31,9 @@
 import itn.let.kakao.kakaoComm.KakaoVO;
 import itn.let.kakao.kakaoComm.kakaoApi.service.KakaoApiService;
 import itn.let.kakao.user.kakaoAt.service.KakaoAlimTalkService;
+import itn.let.kakao.user.kakaoFt.service.KakaoFriendsTalkService;
 import itn.let.kakao.user.kakaoFt.service.KakaoFriendsTalkTemplateService;
+import itn.let.mail.service.StatusResponse;
 import itn.let.mjo.mjocommon.MjonCommon;
 import itn.let.mjo.mjocommon.MjonHolidayApi;
 import itn.let.mjo.msgdata.service.MjonMsgDataService;
@@ -67,6 +71,9 @@
 	
 	@Resource(name = "kakaoFriendsTalkTemplateService")
 	private KakaoFriendsTalkTemplateService kakaoFtTemplateService;
+	
+	@Resource(name = "KakaoFriendsTalkService")
+	private KakaoFriendsTalkService kakaoFriendsTalkService;
 	
 	/** userManageService */
 	@Resource(name = "userManageService")
@@ -113,114 +120,109 @@
 		
 		model.addAttribute("loginVO", loginVO);
 		
-		try {
 			
-			if(!userId.equals("") && !author.equals("ROLE_ADMIN")) {
+		if(!"".equals(userId) && !"ROLE_ADMIN".equals(author)) {
+		
+			//사용자 등록 발신프로필 정보 조회해오기
+			kakaoVO.setUserId(userId);
+			List<KakaoVO> resultProfileList = kakaoApiService.selectKakaoProfileList(kakaoVO);
+			model.addAttribute("resultProfileList", resultProfileList);
 			
-				//사용자 등록 발신프로필 정보 조회해오기
-				kakaoVO.setUserId(userId);
-				List<KakaoVO> resultProfileList = kakaoApiService.selectKakaoProfileList(kakaoVO);
-				model.addAttribute("resultProfileList", resultProfileList);
-				
-				// 특수문자 리스트 불러오기
-				MjonSymbolVO symbolVO = new MjonSymbolVO();
-				List<MjonSymbolVO> symbolList = mjonSymbolService.selectMjonSymbolList(symbolVO);
-				model.addAttribute("symbolList", symbolList);
-				
-				//아이디 발신번호 리스트 불러오기.
-				List<String> resultSendPhonList = mjonMsgDataService.selectSendPhonNumList(userId);
-				List<String> resultPhonList = new ArrayList<String>();
-				MJUtil mjUtil = new MJUtil();
-				for(String phone : resultSendPhonList) {
-					resultPhonList.add(mjUtil.addDash(phone));
-				}
-				model.addAttribute("resultPhonList", resultPhonList);
-				
-				
-				MberManageVO mberManageVO = mjonMsgDataService.selectMberManageInfo(userId);
-				
-				model.addAttribute("atSmishingYn", mberManageVO.getAtSmishingYn());
-				
-				//3.사용자 개인단가 정보가 0이 아니면 개인단가 사용, 없으면 시스템 기본 단가 사용
-				/*Float shortPrice = mberManageVO.getShortPrice();
-				Float longPrice = mberManageVO.getLongPrice();
-				Float picturePrice = mberManageVO.getPicturePrice();
-				Float picture2Price = mberManageVO.getPicture2Price();
-				Float picture3Price = mberManageVO.getPicture3Price();*/
-				BigDecimal userMoney = new BigDecimal(mberManageVO.getUserMoney()).setScale(2, RoundingMode.HALF_EVEN);
-				
-				model.addAttribute("userMoney", userMoney);
-				
-				
-				//////////////////////////////////////////////////////////////////
-				
-				//최근 전송 내역
-				MjonMsgDataVO searchVO = new MjonMsgDataVO();
-				Calendar cal = Calendar.getInstance();
-				Date now = new Date();
-				SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd");
-				cal.setTime(now);
-				cal.add(Calendar.DATE, -3);
-				String chkDate = format.format(cal.getTime());
-				searchVO.setUserId(userId);
-				searchVO.setMyMsgStDt(chkDate);	//검색 시작일 저장 - 현재날짜로 부터 3일 이전 날짜로 시작
-				model.addAttribute("resultLatestMsgList", mjonMsgDataService.selectLatestMsgList(searchVO));
-				
-				//자주보내는 번호
-				model.addAttribute("resultBookMarkMsgList", mjonMsgDataService.selectBookMarkMsgList(searchVO));
-				
-				// 사용자 정의 단가 정보 불러오기(시스템 단가 혹은 협의 단가)
-				model.addAttribute("sendPrice", kakaoSendUtil.selectSendPriceOfKakaoAtAndSmsAndMms(userId));
-				
-				
-				//사용자 템플릿 정보 조회
-				String friendId = kakaoVO.getFriendId();
-				KakaoVO resultTemplateVO = new KakaoVO();
-				
-				if(friendId != null) {
-					resultTemplateVO = kakaoFtTemplateService.selectKakaoFriendsTemplateDetail(kakaoVO);
-				}
-				model.addAttribute("resultTemplateVO", resultTemplateVO);
-				
-				
-				//친구톡 발송시간 체크 하기 - 20:50 ~ 익일 08:00 사이에는 발송 금지
-				
-				SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
-				
-				Date nows = new Date();
-				String nowDate = sdf1.format(nows);
-				cal.setTime(nows);
-				
-				int hours = cal.get(Calendar.HOUR_OF_DAY);
-				int minuts = cal.get(Calendar.MINUTE);
+			// 특수문자 리스트 불러오기
+			MjonSymbolVO symbolVO = new MjonSymbolVO();
+			List<MjonSymbolVO> symbolList = mjonSymbolService.selectMjonSymbolList(symbolVO);
+			model.addAttribute("symbolList", symbolList);
+			
+			//아이디 발신번호 리스트 불러오기.
+			List<String> resultSendPhonList = mjonMsgDataService.selectSendPhonNumList(userId);
+			List<String> resultPhonList = new ArrayList<String>();
+			MJUtil mjUtil = new MJUtil();
+			for(String phone : resultSendPhonList) {
+				resultPhonList.add(mjUtil.addDash(phone));
+			}
+			model.addAttribute("resultPhonList", resultPhonList);
+			
+			
+			MberManageVO mberManageVO = mjonMsgDataService.selectMberManageInfo(userId);
+			
+			model.addAttribute("atSmishingYn", mberManageVO.getAtSmishingYn());
+			
+			//3.사용자 개인단가 정보가 0이 아니면 개인단가 사용, 없으면 시스템 기본 단가 사용
+			/*Float shortPrice = mberManageVO.getShortPrice();
+			Float longPrice = mberManageVO.getLongPrice();
+			Float picturePrice = mberManageVO.getPicturePrice();
+			Float picture2Price = mberManageVO.getPicture2Price();
+			Float picture3Price = mberManageVO.getPicture3Price();*/
+			BigDecimal userMoney = new BigDecimal(mberManageVO.getUserMoney()).setScale(2, RoundingMode.HALF_EVEN);
+			
+			model.addAttribute("userMoney", userMoney);
+			
+			
+			//////////////////////////////////////////////////////////////////
+			
+			//최근 전송 내역
+			MjonMsgDataVO searchVO = new MjonMsgDataVO();
+			Calendar cal = Calendar.getInstance();
+			Date now = new Date();
+			SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd");
+			cal.setTime(now);
+			cal.add(Calendar.DATE, -3);
+			String chkDate = format.format(cal.getTime());
+			searchVO.setUserId(userId);
+			searchVO.setMyMsgStDt(chkDate);	//검색 시작일 저장 - 현재날짜로 부터 3일 이전 날짜로 시작
+//				model.addAttribute("resultLatestMsgList", mjonMsgDataService.selectLatestMsgList(searchVO));
+			
+			//자주보내는 번호
+//				model.addAttribute("resultBookMarkMsgList", mjonMsgDataService.selectBookMarkMsgList(searchVO));
+			
+			// 사용자 정의 단가 정보 불러오기(시스템 단가 혹은 협의 단가)
+			model.addAttribute("sendPrice", kakaoSendUtil.selectSendPriceOfKakaoAtAndSmsAndMms(userId));
+			
+			
+			//사용자 템플릿 정보 조회
+			String friendId = kakaoVO.getFriendId();
+			KakaoVO resultTemplateVO = new KakaoVO();
+			
+			if(friendId != null) {
+				resultTemplateVO = kakaoFtTemplateService.selectKakaoFriendsTemplateDetail(kakaoVO);
+			}
+			model.addAttribute("resultTemplateVO", resultTemplateVO);
+			
+			
+			//친구톡 발송시간 체크 하기 - 20:50 ~ 익일 08:00 사이에는 발송 금지
+			
+			SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
+			
+			Date nows = new Date();
+			String nowDate = sdf1.format(nows);
+			cal.setTime(nows);
+			
+			int hours = cal.get(Calendar.HOUR_OF_DAY);
+			int minuts = cal.get(Calendar.MINUTE);
 
-				boolean sendStatus = true;
+			boolean sendStatus = true;
+			
+			if(hours >= 20) {
 				
-				if(hours >= 20) {
+				if(minuts >= 50) {
 					
-					if(minuts >= 50) {
-						
-						System.out.println("발송금지 시간" + hours + ":" + minuts); 
-						sendStatus = false;
-						
-					}
-					
-				}
-				
-				if(hours < 8) {
-					
-					System.out.println("발송금지 시간" + hours + ":" + minuts);
+					System.out.println("발송금지 시간" + hours + ":" + minuts); 
 					sendStatus = false;
 					
 				}
 				
-				System.out.println("발송상태는 ::: "+sendStatus);
-				model.addAttribute("sendStatus", sendStatus);
-			
 			}
 			
-		} catch (Exception e) {
-			System.out.println(" kakaoFriendsTalkMsgDataViewDataRegist Error ::: " + e);
+			if(hours < 8) {
+				
+				System.out.println("발송금지 시간" + hours + ":" + minuts);
+				sendStatus = false;
+				
+			}
+			
+			System.out.println("발송상태는 ::: "+sendStatus);
+			model.addAttribute("sendStatus", sendStatus);
+		
 		}
 		
 		return "web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView";
@@ -313,7 +315,29 @@
 		return modelAndView;
 	}
 	
-	
+
+	/**
+	 * @methodName	: sendMsgData_ft_advc 
+	 * @author		: 이호영
+	 * @date		: 2025. 4. 17.
+	 * @description	: 친구톡 발송기능 
+	 * @return : ResponseEntity<StatusResponse>
+	 * @param kakaoVO
+	 * @param request
+	 * @param model
+	 * @return
+	 * @throws Exception
+	 * 
+	 */
+	@RequestMapping(value = "/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgSendAjax_advc.do")
+	public ResponseEntity<StatusResponse> kakaoFriendsTalkMsgSendAjax_advc(
+			@RequestBody KakaoVO kakaoVO,
+			HttpServletRequest request
+			) throws Exception {
+		System.out.println(" :: sendMsgData_ft_advc :: ");
+		return ResponseEntity.ok().body(kakaoFriendsTalkService.insertKakaoFtSandAjax_advc(kakaoVO, request)) ;
+		
+	}
 	
 	/**
 	* @Method Name : kakaoFriendsTalkMsgSendAjax
src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkTemplateController.java
--- src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkTemplateController.java
+++ src/main/java/itn/let/kakao/user/kakaoFt/web/KakaoFriendsTalkTemplateController.java
@@ -1,5 +1,6 @@
 package itn.let.kakao.user.kakaoFt.web;
 
+import java.time.LocalDateTime;
 import java.util.ArrayList;
 import java.util.List;
 import java.util.Map;
@@ -8,11 +9,16 @@
 import javax.servlet.http.HttpServletRequest;
 import javax.servlet.http.HttpServletResponse;
 
+import org.apache.commons.lang3.StringUtils;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
 import org.springframework.stereotype.Controller;
+import org.springframework.ui.Model;
 import org.springframework.ui.ModelMap;
 import org.springframework.web.bind.annotation.ModelAttribute;
 import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.ResponseBody;
 import org.springframework.web.multipart.MultipartFile;
 import org.springframework.web.multipart.MultipartHttpServletRequest;
 import org.springframework.web.servlet.ModelAndView;
@@ -21,6 +27,7 @@
 import egovframework.rte.ptl.mvc.tags.ui.pagination.PaginationInfo;
 import itn.com.cmm.EgovMessageSource;
 import itn.com.cmm.LoginVO;
+import itn.com.cmm.RestResponse;
 import itn.com.cmm.service.EgovCmmUseService;
 import itn.com.utl.fcc.service.EgovStringUtil;
 import itn.let.kakao.kakaoComm.KakaoReturnVO;
@@ -32,10 +39,14 @@
 import itn.let.kakao.kakaoComm.kakaoApi.KakaoFTJsonSave;
 import itn.let.kakao.kakaoComm.kakaoApi.service.KakaoApiService;
 import itn.let.kakao.user.kakaoFt.service.KakaoFriendsTalkTemplateService;
+import itn.let.mail.service.StatusResponse;
 import itn.let.mjo.symbol.service.MjonSymbolService;
 import itn.let.mjo.symbol.service.MjonSymbolVO;
+import itn.let.uss.ion.cnt.service.CntManageVO;
 import itn.let.uss.umt.service.EgovUserManageService;
+import lombok.extern.slf4j.Slf4j;
 
+@Slf4j
 @Controller
 public class KakaoFriendsTalkTemplateController {
 
@@ -170,6 +181,63 @@
 		
 		return modelAndView;
 	}
+
+	
+	@RequestMapping(value="/web/pop/ft/kakaoTemplatePop.do")
+	public String siteContentIntro(@ModelAttribute CntManageVO cntManageVO, HttpServletRequest request, Model model) throws Exception {  
+
+		return "/web/pop/kakaoFtPop";
+	}
+	
+	@ResponseBody
+	@RequestMapping(value= {"/web/mjon/kakao/template/sendKakaoFriendsTemplateImageUploadAjax_advc.do"})
+	public ResponseEntity<StatusResponse> sendKakaoFriendsTemplateImageUploadAjax_advc(
+			@ModelAttribute("kakaoVO")  KakaoVO kakaoVO
+			, final MultipartHttpServletRequest multiRequest
+			) throws Exception {
+		ModelAndView modelAndView = new ModelAndView();		
+		modelAndView.setViewName("jsonView");
+		
+		try {
+			LoginVO loginVO = EgovUserDetailsHelper.isAuthenticated()? (LoginVO)EgovUserDetailsHelper.getAuthenticatedUser():null;
+			String userId = loginVO == null ? "" : EgovStringUtil.isNullToString(loginVO.getId());
+			
+			if(StringUtils.isEmpty(userId)) {	//KISA 보안취약점 조치 (2018-12-10, 이정은)
+				return ResponseEntity.ok(
+						new StatusResponse(HttpStatus.UNAUTHORIZED
+								, "로그인을 하셔야 이용 가능합니다."
+								, LocalDateTime.now()
+								)
+						);
+			}
+			
+			
+				
+				final Map<String, MultipartFile> files = multiRequest.getFileMap();
+				/*				if (!files.isEmpty()){
+									
+									Map<String, String> resultMap = kakaoApiImageUpload.kakaoApiImageUpload(kakaoVO, files, 0);
+									
+									//테스트 용 
+									modelAndView.addObject("code", resultMap.get("code"));
+									modelAndView.addObject("msg", resultMap.get("msg"));
+									modelAndView.addObject("imgUrl", resultMap.get("imgUrl"));
+								}
+								
+								modelAndView.addObject("result", "success");*/
+
+				return ResponseEntity.ok().body(kakaoApiImageUpload.kakaoApiImageUpload_advc(kakaoVO, files, 0));
+			} catch (Exception e) {
+				e.printStackTrace();
+				// TODO: handle exception
+				return ResponseEntity.ok().body(new StatusResponse(
+						HttpStatus.BAD_REQUEST
+						, "오류가 발생하였습니다."
+						, ""
+						));
+		}
+		
+	}
 	
 	
 	// 카카오 친구톡 템플릿 등록요청
@@ -189,9 +257,13 @@
 		@ModelAttribute("kakaoVO")  KakaoVO kakaoVO
 		, final MultipartHttpServletRequest multiRequest
 	) throws Exception {
+		
+		
 		ModelAndView modelAndView = new ModelAndView();		
 		modelAndView.setViewName("jsonView");
 		
+//		log.info(" + kakaoVO.getAdFlag() :: [{}]", kakaoVO.getAdFlag()); 
+		log.info(" +  ImgLink :: [{}]", kakaoVO.getImgLink()); 
 		try {
 			LoginVO loginVO = EgovUserDetailsHelper.isAuthenticated()? (LoginVO)EgovUserDetailsHelper.getAuthenticatedUser():null;
 			String userId = loginVO == null ? "" : EgovStringUtil.isNullToString(loginVO.getId());
@@ -207,16 +279,18 @@
 				kakaoVO.setLastUpdusrId(userId);
 				kakaoVO.setSendType("FT");
 				
+				/*
 				String imgUrl = kakaoVO.getTemplateImageUrl();
 				int buttonSize = kakaoVO.getButtonVOList().size();
 				
 				if(!imgUrl.equals("") || buttonSize > 0) {
-					
+				
+					log.info(" + kakaoVO.getAdFlag() :: [{}]", kakaoVO.getAdFlag());
 					//json 파일 생성 처리
 					String resultJsonPath = kakaoFTJsonSave.kakaoApiJsonSave(kakaoVO);
 					kakaoVO.setBizJsonName(resultJsonPath);
 					
-				}
+				}*/
 				
 				int result = kakaoFtTemplateService.insertKakaoFriendsTemplateData(kakaoVO);
 				
src/main/java/itn/let/kakao/user/sent/service/KakaoSentVO.java
--- src/main/java/itn/let/kakao/user/sent/service/KakaoSentVO.java
+++ src/main/java/itn/let/kakao/user/sent/service/KakaoSentVO.java
@@ -6,9 +6,11 @@
 import itn.let.uss.umt.service.UserDefaultVO;
 import lombok.Getter;
 import lombok.Setter;
+import lombok.ToString;
 
 @Getter
 @Setter
+@ToString
 public class KakaoSentVO extends UserDefaultVO{
 
 	private static final long serialVersionUID = 1L;
@@ -90,7 +92,7 @@
 	private String bizKakaoFtPrice;
 	private String bizSmsPrice;
 	private String bizMmsPrice;
-	
+
 	private int successCount;
 	private int waitCount;
 	private int failCount;
@@ -112,6 +114,7 @@
 	
 	private String successPrice;
 	private String kakaoResendSuccPrice;
+	private String bizKakaoImageType;
 	
 	private String divideYn;
 	
@@ -119,4 +122,6 @@
 	
 	private String yellowId;
 	
+	private String adFlag;
+	
 }
src/main/java/itn/let/kakao/user/sent/service/impl/KakaoSentServiceImpl.java
--- src/main/java/itn/let/kakao/user/sent/service/impl/KakaoSentServiceImpl.java
+++ src/main/java/itn/let/kakao/user/sent/service/impl/KakaoSentServiceImpl.java
@@ -39,7 +39,9 @@
 import itn.let.kakao.user.sent.service.KakaoSentService;
 import itn.let.kakao.user.sent.service.KakaoSentVO;
 import itn.let.mjo.msgsent.service.MjonMsgSentVO;
+import lombok.extern.slf4j.Slf4j;
 
+@Slf4j
 @Service("KakaoSentService")
 public class KakaoSentServiceImpl  extends EgovAbstractServiceImpl implements KakaoSentService{
 
@@ -140,6 +142,9 @@
 		List<KakaoSentVO> resultList = new ArrayList<KakaoSentVO>();
 		
 		resultList = kakaoSentDAO.selectAllKakaoSentList_advc(kakaoSentVO);
+
+		System.out.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
+		
 		
 		//totPrice 계산 및 상태코드 set
 		resultList = resultList.stream().map(t -> setPriceNCode(t)).collect(Collectors.toList());
@@ -287,15 +292,27 @@
 	 * */
 	private KakaoSentVO setPriceNCode(KakaoSentVO result) {
 		
+		log.info("=============================== setPriceNCode ================================");
+		
 		//성공 건수 세팅
 		KakaoSentVO eachCnt = new KakaoSentVO();
 		eachCnt.setMsgGroupId(result.getMsgGroupId());
 		eachCnt.setBizKakaoResendYn("Y".equals(result.getBizKakaoResendYn()) ? "Y" : "N");
+		eachCnt.setMsgType(result.getMsgType());
+		eachCnt.setBizKakaoImageType(result.getBizKakaoImageType());
+		eachCnt.setBizKakaoResendType(result.getBizKakaoResendType());
+		
 		try {
+			log.info("eachCnt.getMsgType() :: [{}]", eachCnt.getMsgType());
+			log.info("eachCnt.getBizKakaoImageType() :: [{}]", eachCnt.getBizKakaoImageType());
+			log.info("eachCnt.getBizKakaoResendType() :: [{}]", eachCnt.getBizKakaoResendType());
+			
 			eachCnt = kakaoSentDAO.selectKakaoSentCntEachCnt_advc(eachCnt);
 		} catch (Exception e) {
 			System.out.println("setPriceNCode error!!");
 		}
+		log.info(" + eachCnt.toString() :: [{}]", eachCnt.toString());
+//		log.info(" + eachCnt.getSuccessPrice() :: [{}]", eachCnt.getSuccessPrice());
 		
 		result.setSuccessCount(eachCnt.getSuccessCount());
 		result.setWaitCount(eachCnt.getWaitCount());
@@ -328,9 +345,9 @@
 		
 		//총금액 시작
 		//=======================================================
-		// TotPrice : 성공건수에 대한 금액 곱하기
-		BigDecimal atPrice = new BigDecimal(successPrice);
-		BigDecimal kakaoResendPrice = new BigDecimal(kakaoResendSuccPrice);
+		// TotPrice : 성공건수에 대한 금액 곱하기 ? : null 처리
+		BigDecimal atPrice = successPrice != null ? new BigDecimal(successPrice) : BigDecimal.ZERO;
+		BigDecimal kakaoResendPrice = kakaoResendSuccPrice != null ? new BigDecimal(kakaoResendSuccPrice) : BigDecimal.ZERO;
 		BigDecimal totalPrice = atPrice.add(kakaoResendPrice);
 		// 소수점 한 자리로 설정 (반올림)// totalPrice 값을 소수점 한 자리까지 반올림하여 roundedTotalPrice에 저장
 		// RoundingMode.HALF_UP: 반올림 방식으로, 소수점 기준 5 이상이면 올림, 그렇지 않으면 내림
src/main/java/itn/let/kakao/user/sent/web/KakaoSentController.java
--- src/main/java/itn/let/kakao/user/sent/web/KakaoSentController.java
+++ src/main/java/itn/let/kakao/user/sent/web/KakaoSentController.java
@@ -2,6 +2,7 @@
 
 import java.io.FileReader;
 
+import org.apache.commons.io.FilenameUtils;
 import org.apache.commons.lang3.StringUtils;
 import itn.com.cmm.util.DateUtils;
 import java.text.SimpleDateFormat;
@@ -38,6 +39,7 @@
 import itn.com.cmm.util.StringUtil;
 import itn.com.cmm.util.StringUtil2;
 import itn.com.utl.fcc.service.EgovStringUtil;
+import itn.let.cmm.vo.FileInfoVO;
 import itn.let.kakao.admin.kakaoAt.service.MjonKakaoATVO;
 import itn.let.kakao.kakaoComm.KakaoButtonVO;
 import itn.let.kakao.kakaoComm.KakaoReturnVO;
@@ -46,13 +48,20 @@
 import itn.let.kakao.user.sent.service.KakaoSentDetailVO;
 import itn.let.kakao.user.sent.service.KakaoSentService;
 import itn.let.kakao.user.sent.service.KakaoSentVO;
+import itn.let.mjo.msgsent.service.MjonMsgSentVO;
+import itn.let.mjo.msgsent.service.impl.MjonMsgSentDAO;
+import lombok.extern.slf4j.Slf4j;
 
+@Slf4j
 @Controller
 public class KakaoSentController {
 
 	@Resource(name = "KakaoSentService")
 	private KakaoSentService kakaoSentService;
 
+	@Resource(name="MjonMsgSentDAO")
+	private MjonMsgSentDAO mjonMsgSentDAO;
+	
 	@Autowired
 	KakaoApiTemplate kakaoApiTemplate;
 	
@@ -182,7 +191,7 @@
 		
 		//전체 발송 리스트 불러오기
 		List<KakaoSentVO> resultAllSentList = kakaoSentService.selectAllKakaoSentList(kakaoSentVO);
-		System.out.println("??");
+		
 		model.addAttribute("resultAllSentList", resultAllSentList);
 //		model.addAttribute("resultAllSentCnt", resultAllSentList.size());
 		
@@ -890,22 +899,21 @@
 		KakaoReturnVO returnVO = new KakaoReturnVO();
 		try {
 			
-			String tmpContent = kakaoATVO.getSmsTxtTrans();
-			String jsonFilePath = kakaoATVO.getBizKakaoJsonFile();
+			String jsonContents = kakaoATVO.getContents();
 			
 			//친구톡 이미지 또는 버튼 정보가 있다면 실행
-			if(jsonFilePath != null && jsonFilePath.length() > 0) {
+			if(jsonContents != null && jsonContents.length() > 0) {
 				
-				FileReader reader = new FileReader(jsonFilePath);         
+				/*FileReader reader = new FileReader(jsonFilePath);         
 				int ch;        
 				String resultStr = "";
 				while ((ch = reader.read()) != -1) {
 					 
 					resultStr = resultStr + (char)ch;
 				}
-				
+				*/
 				JSONParser parser = new JSONParser();
-				Object obj = parser.parse(resultStr);
+				Object obj = parser.parse(jsonContents);
 				JSONObject object = (JSONObject) obj;
 				
 				String image = (object.get("image") == null) ? null : object.get("image").toString();
@@ -964,7 +972,7 @@
 			}
 			
 			//친구톡 내용 셋팅
-			returnVO.setTemplateContent(tmpContent);
+			returnVO.setTemplateContent(kakaoATVO.getSmsTxt());
 
 		} catch (Exception e) {
 			e.printStackTrace();
@@ -1037,42 +1045,70 @@
 		
 		
 		
-		//발송 관리 문자발송 내용 상세보기 팝업 => 문자내용(MJ_MSG_DATA)
-		MjonKakaoATVO mjonKakaoATResultVO = kakaoSentService.selectKakaoSentDetailDataAjax(mjonKakaoATVO);
-		// 대체문자 엔터키 치환
-		mjonKakaoATResultVO.setSmsTxt(StringUtil2.replaceBR(mjonKakaoATResultVO.getSmsTxt()));  
-		model.addAttribute("resultMsgDetail", mjonKakaoATResultVO);
+		try {
 		
-		String msgType = mjonKakaoATResultVO.getMsgType();
+			//발송 관리 문자발송 내용 상세보기 팝업 => 문자내용(MJ_MSG_DATA)
+			MjonKakaoATVO mjonKakaoATResultVO = kakaoSentService.selectKakaoSentDetailDataAjax(mjonKakaoATVO);
+			// 대체문자 엔터키 치환
+			if(StringUtils.isNotEmpty(mjonKakaoATResultVO.getSmsTxt())) {
+				mjonKakaoATResultVO.setSmsTxt(StringUtil2.replaceBR(mjonKakaoATResultVO.getSmsTxt()));  
+			}
+			model.addAttribute("resultMsgDetail", mjonKakaoATResultVO);
+			
+			String msgType = mjonKakaoATResultVO.getMsgType();
+			
+			if(msgType.equals("8")) {//카카오 알림톡인 경우 상세정보 처리
+				
+				// 템플릿 api 가져오기
+				KakaoVO kakaoVO = new KakaoVO(); 
+				kakaoVO.setSenderKey(mjonKakaoATResultVO.getMsgNoticetalkSenderKey());
+				kakaoVO.setTemplateCode(mjonKakaoATResultVO.getMsgNoticetalkTmpKey());
+				
+				KakaoReturnVO kakaoTemplateInfo =kakaoApiTemplate.selectKakaoApiTemplateDetail(kakaoVO);
+				
+				model.addAttribute("kakaoTemplateInfo", kakaoTemplateInfo);
+				// //템플릿 api 가져오기
+				
+			}else if(msgType.equals("9")) {//카카오 친구톡인 경우 상세정보 처리
+				
+				//String smsTxt = mjonKakaoATResultVO.getSmsTxt();
+				
+				KakaoReturnVO kakaoTemplateInfo = getKakaoFTSendTemplateInfo(mjonKakaoATResultVO);
+				model.addAttribute("kakaoTemplateInfo", kakaoTemplateInfo);
+				
+				if(StringUtils.isNotEmpty( mjonKakaoATResultVO.getFilePath1() )) {
+					List<FileInfoVO> fileInfos = new ArrayList<>();
+					
+
+					// 확장자 제외한 파일명
+					String fileId = FilenameUtils.getBaseName(mjonKakaoATResultVO.getFilePath1());
+					
+					// 파일 정보 조회
+					MjonMsgSentVO info = mjonMsgSentDAO.selectFileInfo(fileId);
+
+					// FileInfo 객체 생성 및 추가
+					FileInfoVO fileInfo = new FileInfoVO();
+					fileInfo.setAtchFileId(info.getAtchFileId());
+					fileInfo.setFileSn(info.getFileSn());
+
+					fileInfos.add(fileInfo);
+					model.addAttribute("fileInfos", fileInfos);
+				}
+				//kakaoTemplateInfo.setTemplateContent(smsTxt);
+				
+				
+			}else {
+				
+				model.addAttribute("kakaoTemplateInfo", "");
+				
+			}
+			model.addAttribute("msgType", msgType);		
 		
-		if(msgType.equals("8")) {//카카오 알림톡인 경우 상세정보 처리
 			
-			// 템플릿 api 가져오기
-			KakaoVO kakaoVO = new KakaoVO(); 
-			kakaoVO.setSenderKey(mjonKakaoATResultVO.getMsgNoticetalkSenderKey());
-			kakaoVO.setTemplateCode(mjonKakaoATResultVO.getMsgNoticetalkTmpKey());
-			
-			KakaoReturnVO kakaoTemplateInfo =kakaoApiTemplate.selectKakaoApiTemplateDetail(kakaoVO);
-			
-			model.addAttribute("kakaoTemplateInfo", kakaoTemplateInfo);
-			// //템플릿 api 가져오기
-			
-		}else if(msgType.equals("9")) {//카카오 친구톡인 경우 상세정보 처리
-			
-			//String smsTxt = mjonKakaoATResultVO.getSmsTxt();
-			
-			KakaoReturnVO kakaoTemplateInfo = getKakaoFTSendTemplateInfo(mjonKakaoATResultVO);
-			//kakaoTemplateInfo.setTemplateContent(smsTxt);
-			
-			model.addAttribute("kakaoTemplateInfo", kakaoTemplateInfo);
-			
-		}else {
-			
-			model.addAttribute("kakaoTemplateInfo", "");
-			
+		} catch (Exception e) {
+			e.printStackTrace();
+			// TODO: handle exception
 		}
-		
-		model.addAttribute("msgType", msgType);		
 		return "web/kakao/sent/KakaoSentDetailPopAjax";
 	}
 }
src/main/java/itn/let/mail/service/StatusResponse.java
--- src/main/java/itn/let/mail/service/StatusResponse.java
+++ src/main/java/itn/let/mail/service/StatusResponse.java
@@ -5,6 +5,7 @@
 import org.springframework.http.HttpStatus;
 
 import itn.let.mjo.pay.service.RefundVO;
+import lombok.AllArgsConstructor;
 import lombok.Builder;
 import lombok.Getter;
 import lombok.NoArgsConstructor;
src/main/java/itn/let/mjo/msg/service/MjonMsgVO.java
--- src/main/java/itn/let/mjo/msg/service/MjonMsgVO.java
+++ src/main/java/itn/let/mjo/msg/service/MjonMsgVO.java
@@ -165,6 +165,7 @@
 	
 	private float smsPrice;		// sms 단가
 	private float mmsPrice;		// mms 단가
+	private float picturePrice;		// mms 단가
 	private float kakaoAtPrice;	// 카카오 알림톡 단가
 	private float kakaoFtPrice;	// 카카오 친구톡 단가
 	private float kakaoFtImgPrice;// 카카오 이미지 단가
@@ -301,6 +302,8 @@
 	private String bizLogCallStatusCode;	//다우기술 biz_log 테이블의 발송결과 코드 값 성공/실패/대기 코드값 변환(성공:S, 대기:W, 실패:F).
 	private String bizLogCallStatusTxt;	//다우기술 biz_log 테이블의 발송결과 내용 텍스트.
 	private String bizLogStatus;		//다우기술 biz_log 테이블의 전송상태값 
+	
+	private String bizKakaoImageType;	// 비즈 발송 img 값 
 
 	private String  accessKey; // 'API Key',
 
src/main/java/itn/let/mjo/msg/service/impl/MjonMsgDAO.java
--- src/main/java/itn/let/mjo/msg/service/impl/MjonMsgDAO.java
+++ src/main/java/itn/let/mjo/msg/service/impl/MjonMsgDAO.java
@@ -455,6 +455,11 @@
 		update("mjonMsgDAO.updateKakaoAtDelayCancelMsgDataFlag", mjonMsgVO);
 	}
 	
+	// mj_msg_data 테이블 지연 알림톡 취소 값 수정
+	public void updateRefundY(MjonMsgVO mjonMsgVO) {
+		update("mjonMsgDAO.updateRefundY", mjonMsgVO);
+	}
+	
 	// mj_msg_group_data 테이블 지연 알림톡 취소 값 수정
 	public void updateKakaoAtDelayCancelMsgGroupDataFlag(MjonMsgVO mjonMsgVO) {
 		update("mjonMsgDAO.updateKakaoAtDelayCancelMsgGroupDataFlag", mjonMsgVO);
 
src/main/java/itn/let/mjo/msgsent/service/MjonMsgDetailSentDTO.java (added)
+++ src/main/java/itn/let/mjo/msgsent/service/MjonMsgDetailSentDTO.java
@@ -0,0 +1,22 @@
+package itn.let.mjo.msgsent.service;
+
+import java.util.List;
+
+import itn.let.cmm.vo.FileInfoVO;
+import itn.let.uss.umt.service.UserDefaultVO;
+import lombok.Getter;
+import lombok.Setter;
+
+@Getter
+@Setter
+public class MjonMsgDetailSentDTO extends UserDefaultVO{
+
+	private static final long serialVersionUID = 1L;
+
+
+	private String userId;
+	private String callTo;
+	private String statusTxt;
+	
+	
+}
src/main/java/itn/let/mjo/msgsent/service/MjonMsgSentService.java
--- src/main/java/itn/let/mjo/msgsent/service/MjonMsgSentService.java
+++ src/main/java/itn/let/mjo/msgsent/service/MjonMsgSentService.java
@@ -65,7 +65,7 @@
 
 	public Map<String, Object> selectAllMsgSentDetailView(MjonMsgDetailSentVO mjonMsgDetailSentVO) throws Exception;
 
-	public List<MjonMsgDetailSentVO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO);
+	public List<MjonMsgDetailSentDTO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO);
 
 	public void msgSentExcelDownLoad(MjonMsgSentVO mjonMsgSentVO, HttpServletResponse response) throws IOException, Exception;
 
src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentDAO.java
--- src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentDAO.java
+++ src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentDAO.java
@@ -9,6 +9,7 @@
 import itn.let.fax.addr.service.FaxAddrGroupVO;
 import itn.let.mjo.addr.service.AddrGroupVO;
 import itn.let.mjo.block.service.MjonBlockVO;
+import itn.let.mjo.msgsent.service.MjonMsgDetailSentDTO;
 import itn.let.mjo.msgsent.service.MjonMsgDetailSentVO;
 import itn.let.mjo.msgsent.service.MjonMsgSWFDTO;
 import itn.let.mjo.msgsent.service.MjonMsgSentVO;
@@ -182,9 +183,9 @@
 		return (MjonMsgDetailSentVO) select("MjonMsgSentDAO.selectAllMsgSentDetailView", mjonMsgDetailSentVO);
 	}
 
-	public List<MjonMsgDetailSentVO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO) {
+	public List<MjonMsgDetailSentDTO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO) {
 		
-		return (List<MjonMsgDetailSentVO>) list("MjonMsgSentDAO.findByMsgDetailListAjax", mjonMsgDetailSentVO);
+		return (List<MjonMsgDetailSentDTO>) list("MjonMsgSentDAO.findByMsgDetailListAjax", mjonMsgDetailSentVO);
 	}
 
 	public List<String> findByReqDateWhereMsgGroupId(String msgGroupId) {
src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentServiceImpl.java
--- src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentServiceImpl.java
+++ src/main/java/itn/let/mjo/msgsent/service/impl/MjonMsgSentServiceImpl.java
@@ -39,6 +39,7 @@
 import itn.let.fax.addr.service.FaxAddrGroupVO;
 import itn.let.mjo.addr.service.AddrGroupVO;
 import itn.let.mjo.block.service.MjonBlockVO;
+import itn.let.mjo.msgsent.service.MjonMsgDetailSentDTO;
 import itn.let.mjo.msgsent.service.MjonMsgDetailSentVO;
 import itn.let.mjo.msgsent.service.MjonMsgSWFDTO;
 import itn.let.mjo.msgsent.service.MjonMsgSentService;
@@ -474,9 +475,9 @@
 	}
 
 	@Override
-	public List<MjonMsgDetailSentVO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO) {
+	public List<MjonMsgDetailSentDTO> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO) {
 		
-		List<MjonMsgDetailSentVO> list = mjonMsgSentDAO.findByMsgDetailListAjax(mjonMsgDetailSentVO);
+		List<MjonMsgDetailSentDTO> list = mjonMsgSentDAO.findByMsgDetailListAjax(mjonMsgDetailSentVO);
 		list.stream().forEach(t->{
 			t.setCallTo(StringUtil2.formatPhone(t.getCallTo())); 
 		});
src/main/java/itn/let/mjo/msgsent/web/MjonMsgSentController.java
--- src/main/java/itn/let/mjo/msgsent/web/MjonMsgSentController.java
+++ src/main/java/itn/let/mjo/msgsent/web/MjonMsgSentController.java
@@ -50,6 +50,7 @@
 import itn.let.mjo.addr.service.AddrVO;
 import itn.let.mjo.apikey.service.ApiKeyMngService;
 import itn.let.mjo.apikey.service.ApiKeyVO;
+import itn.let.mjo.msgsent.service.MjonMsgDetailSentDTO;
 import itn.let.mjo.msgsent.service.MjonMsgDetailSentVO;
 import itn.let.mjo.msgsent.service.MjonMsgSentCntVO;
 import itn.let.mjo.msgsent.service.MjonMsgSentService;
@@ -191,7 +192,17 @@
  	public ResponseEntity<StatusResponse> findByMsgDetailListAjax(MjonMsgDetailSentVO mjonMsgDetailSentVO) throws Exception {
 
 			
-		List<MjonMsgDetailSentVO> resultList = mjonMsgSentService.findByMsgDetailListAjax(mjonMsgDetailSentVO);
+		List<MjonMsgDetailSentDTO> resultList = new ArrayList<>();
+		try {
+
+			resultList = mjonMsgSentService.findByMsgDetailListAjax(mjonMsgDetailSentVO);	
+		} catch (Exception e) {
+			e.printStackTrace();
+			// TODO: handle exception
+		}
+		log.info("resultList :: [{}]", resultList.size());
+//		resultList = resultList.subList(0, 275000);
+//		log.info("resultList :: [{}]", resultList.size());
 		
 
 		return ResponseEntity.ok().body(new StatusResponse(HttpStatus.OK, "", resultList));
src/main/java/itn/let/mjo/pay/service/MjonPayVO.java
--- src/main/java/itn/let/mjo/pay/service/MjonPayVO.java
+++ src/main/java/itn/let/mjo/pay/service/MjonPayVO.java
@@ -236,6 +236,8 @@
 
 	private int  remainPoint;
 	
+	private String  userData;
+	
 	
 	private String totChgPay;			// 간편결제분류
 	
src/main/java/itn/let/mjo/pay/service/impl/MjonPayDAO.java
--- src/main/java/itn/let/mjo/pay/service/impl/MjonPayDAO.java
+++ src/main/java/itn/let/mjo/pay/service/impl/MjonPayDAO.java
@@ -5,6 +5,7 @@
 import org.springframework.stereotype.Repository;
 
 import egovframework.rte.psl.dataaccess.EgovAbstractDAO;
+import itn.let.kakao.kakaoComm.BizKakaoPriceVO;
 import itn.let.mjo.pay.service.MjonPayVO;
 import itn.let.mjo.pay.service.MjonVaMsgLogVO;
 import itn.let.mjo.pay.service.RefundVO;
@@ -45,6 +46,10 @@
 	public MjonPayVO selectCashVO(MjonPayVO mjonPayVO) throws Exception{
 		return (MjonPayVO)select("mjonPayDAO.selectCashVO", mjonPayVO);
 	}
+	
+	public BizKakaoPriceVO selectBizKakaoPrice(String msgGroupId) throws Exception{
+		return (BizKakaoPriceVO)select("mjonPayDAO.selectBizKakaoPrice", msgGroupId);
+	}
 
 	public void insertCash(MjonPayVO mjonPayVO) throws Exception{
 		
src/main/java/itn/let/module/base/PriceAndPoint.java
--- src/main/java/itn/let/module/base/PriceAndPoint.java
+++ src/main/java/itn/let/module/base/PriceAndPoint.java
@@ -5,18 +5,27 @@
 
 import javax.annotation.Resource;
 
+import org.apache.commons.lang.StringUtils;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Component;
 
+import egovframework.rte.fdl.cmmn.exception.FdlException;
 import egovframework.rte.fdl.idgnr.EgovIdGnrService;
+import itn.let.kakao.kakaoComm.KakaoSendAdvcVO;
+import itn.let.kakao.kakaoComm.KakaoSendUtil;
+import itn.let.kakao.kakaoComm.KakaoVO;
+import itn.let.kakao.user.kakaoAt.service.impl.KakaoAlimTalkDAO;
 import itn.let.mjo.event.service.MjonEventService;
 import itn.let.mjo.event.service.MjonEventVO;
 import itn.let.mjo.event.service.impl.MjonEventDAO;
 import itn.let.mjo.msg.service.MjonMsgVO;
+import itn.let.mjo.msgdata.service.MjonMsgDataService;
 import itn.let.mjo.msgdata.service.impl.MjonMsgDataDAO;
 import itn.let.mjo.pay.service.MjonPayVO;
 import itn.let.mjo.pay.service.impl.MjonPayDAO;
+import itn.let.sym.site.service.JoinSettingVO;
 import itn.let.uss.umt.service.MberManageVO;
+import lombok.extern.slf4j.Slf4j;
  
 /**
  * 
@@ -32,6 +41,7 @@
  * 
  * 
  */
+@Slf4j
 @Component
 public class PriceAndPoint {
  
@@ -46,6 +56,11 @@
 
     @Resource(name = "egovMjonCashIdGnrService")
     private EgovIdGnrService idgenMjonCashId;
+
+	@Resource(name="kakaoAlimTalkDAO")
+	private KakaoAlimTalkDAO kakaoAlimTalkDAO;
+
+
  
 	/** 
 	 * @methodName	: getBefCash 
@@ -114,19 +129,90 @@
 					, String msgGroupId
 			) throws Exception {
  
-		MjonPayVO mjonPayVO = new MjonPayVO();
-		mjonPayVO.setCashId(idgenMjonCashId.getNextStringId());
-		mjonPayVO.setUserId(userId);
-		System.out.println(" + totPrice :: "+ totPrice);
-		mjonPayVO.setCash(totPrice);
-		mjonPayVO.setFrstRegisterId(userId);
-		mjonPayVO.setMemo(memo);
-		mjonPayVO.setMsgGroupId(msgGroupId);
+//		MjonPayVO mjonPayVO = new MjonPayVO();
+//		mjonPayVO.setCashId(idgenMjonCashId.getNextStringId());
+//		mjonPayVO.setUserId(userId);
+//		System.out.println(" + totPrice :: "+ totPrice);
+//		mjonPayVO.setCash(totPrice);
+//		mjonPayVO.setFrstRegisterId(userId);
+//		mjonPayVO.setMemo(memo);
+//		mjonPayVO.setMsgGroupId(msgGroupId);
  
+		insertCashAndPoint(userId, totPrice, memo, msgGroupId, null);
+		
+//		mjonPayDAO.insertCash(mjonPayVO); //캐시 
+//		mjonPayDAO.updateMemberCash(mjonPayVO); //회원정보 업데이트
+		
+	}
+	/**
+	 * @methodName	: insertCashAndPoint 
+	 * @author		: 이호영
+	 * @date		: 2025. 7. 17.
+	 * @description	: insertCashAndPoint 에서 환불으로 인해 userData 추가
+	 * @return : void
+	 * @param userId
+	 * @param totPrice
+	 * @param memo
+	 * @param msgGroupId
+	 * @param userData
+	 * @throws Exception
+	 * 
+	 */
+	public void insertCashAndPoint(
+			String userId
+			, float totPrice
+			, String memo
+			, String msgGroupId
+			, String userData
+			) throws Exception {
+		
+		MjonPayVO mjonPayVO = buildPayVO(userId, totPrice, memo, msgGroupId);
+
+//		 환불로 인해 userData가 추후 사용될 경우 여기에 처리
+		 if (StringUtils.isNotEmpty(userData)) { mjonPayVO.setOrderId(userData); }
+		
 		mjonPayDAO.insertCash(mjonPayVO); //캐시 
 		mjonPayDAO.updateMemberCash(mjonPayVO); //회원정보 업데이트
 		
 	}
+	
+	private MjonPayVO buildPayVO(String userId, float totPrice, String memo, String msgGroupId) throws FdlException {
+		MjonPayVO vo = new MjonPayVO();
+		vo.setCashId(idgenMjonCashId.getNextStringId());
+		vo.setUserId(userId);
+		vo.setCash(totPrice);
+		vo.setFrstRegisterId(userId);
+		vo.setMemo(memo);
+		vo.setMsgGroupId(msgGroupId);
+		System.out.println(" + totPrice :: " + totPrice);
+		return vo;
+	}
+
+	public void insertBizFtKakaoPrice(String userId, String msgGroupId) throws Exception {
+		KakaoVO kakaoVO = new KakaoVO();
+
+		System.out.println("=======================");
+		// 사용자 개인 단가 정보 불러오기
+		MberManageVO mberManageVO = mjonMsgDataDAO.selectMberManageInfo(userId);
+		// 시스템 기본 단가 정보 불러오기
+		JoinSettingVO sysJoinSetVO = mjonMsgDataDAO.selectJoinSettingInfo();
+		KakaoSendUtil.getValidPrice(mberManageVO.getShortPrice(), sysJoinSetVO.getShortPrice());
+		// TODO Auto-generated method stub
+		kakaoVO.setMsgGroupId(msgGroupId);
+		
+		kakaoVO.setSmsPrice(KakaoSendUtil.getValidPrice(mberManageVO.getShortPrice(), sysJoinSetVO.getShortPrice()));     
+		kakaoVO.setMmsPrice(KakaoSendUtil.getValidPrice(mberManageVO.getLongPrice(), sysJoinSetVO.getLongPrice()));       
+		kakaoVO.setPicturePrice(KakaoSendUtil.getValidPrice(mberManageVO.getPicturePrice(), sysJoinSetVO.getPicturePrice())); 
+		
+		kakaoVO.setKakaoFtPrice(KakaoSendUtil.getValidPrice(mberManageVO.getKakaoFtPrice(), sysJoinSetVO.getKakaoFtPrice())); 
+		kakaoVO.setKakaoFtImgPrice(KakaoSendUtil.getValidPrice(mberManageVO.getKakaoFtImgPrice(), sysJoinSetVO.getKakaoFtImgPrice())); 
+		kakaoVO.setKakaoFtWideImgPrice(KakaoSendUtil.getValidPrice(mberManageVO.getKakaoFtWideImgPrice(), sysJoinSetVO.getKakaoFtWideImgPrice())); 
+		
+		
+		
+		kakaoAlimTalkDAO.insertKakaoSendPrice(kakaoVO);
+		
+	}
  
  
 }
(No newline at end of file)
src/main/java/itn/let/schdlr/service/impl/SchdlrManageServiceImpl.java
--- src/main/java/itn/let/schdlr/service/impl/SchdlrManageServiceImpl.java
+++ src/main/java/itn/let/schdlr/service/impl/SchdlrManageServiceImpl.java
@@ -444,7 +444,7 @@
 		kakaoAlimTalkService.selectKakaoAtSentRefundList();
 
 		// 카카오 친구톡 환불 처리
-//		kakaoAlimTalkService.selectKakaoFtSentRefundList();
+		kakaoAlimTalkService.selectKakaoFtSentRefundList();
 
 	}
 
src/main/java/itn/let/sym/site/service/JoinSettingVO.java
--- src/main/java/itn/let/sym/site/service/JoinSettingVO.java
+++ src/main/java/itn/let/sym/site/service/JoinSettingVO.java
@@ -1,5 +1,12 @@
 package itn.let.sym.site.service;
 
+import lombok.AllArgsConstructor;
+import lombok.Builder;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import lombok.Setter;
+import lombok.ToString;
+
 /**
  * 로그인정책에 대한 VO 클래스를 정의한다.
  * 로그인정책정보의 목록 항목을 관리한다.
@@ -18,13 +25,18 @@
  *
  * </pre>
  */
+
+@Getter
+@Setter
+@NoArgsConstructor
+@AllArgsConstructor
+@Builder
+@ToString
 public class JoinSettingVO  {
 
 	/**
 	 * serialVersionUID
 	 */
-	@SuppressWarnings("unused")
-	private static final long serialVersionUID = 1L;
 	
 	private float shortPrice; 				// 단문 단가
 	private float longPrice; 				//장문 단가
@@ -37,6 +49,9 @@
 	private float customTextPrice;			// 텍스트 단순수정
 	private float kakaoAtPrice;				// 카카오 알림톡 단가
 	private float kakaoFtPrice;				// 카카오 친구톡 단가
+	private float kakaoFtImgPrice;			// 카카오 친구톡 이미지 단가
+	private float kakaoFtWideImgPrice;		// 카카오 친구톡 와이드 단가
+	
 	private float faxPrice;					// 팩스 단가
 	
 	private float refundPer; //환불 비율
@@ -51,188 +66,5 @@
 	private String smishingNoti;	//첫결제(카드제외) 스미싱의심 알림 여부
 	private String holiSmishingNoti; //야간 스미싱알림 여부
 
-	public String getSmishingNoti() {
-		return smishingNoti;
-	}
-
-	public void setSmishingNoti(String smishingNoti) {
-		this.smishingNoti = smishingNoti;
-	}
-
-	public float getShortPrice() {
-		return shortPrice;
-	}
-
-	public void setShortPrice(float shortPrice) {
-		this.shortPrice = shortPrice;
-	}
-
-	public float getLongPrice() {
-		return longPrice;
-	}
-
-	public void setLongPrice(float longPrice) {
-		this.longPrice = longPrice;
-	}
-
-	public float getPicturePrice() {
-		return picturePrice;
-	}
-
-	public void setPicturePrice(float picturePrice) {
-		this.picturePrice = picturePrice;
-	}
-
-	public float getPicture2Price() {
-		return picture2Price;
-	}
-
-	public void setPicture2Price(float picture2Price) {
-		this.picture2Price = picture2Price;
-	}
-
-	public float getPicture3Price() {
-		return picture3Price;
-	}
-
-	public void setPicture3Price(float picture3Price) {
-		this.picture3Price = picture3Price;
-	}
-
-	public float getCustomSamplePrice() {
-		return customSamplePrice;
-	}
-
-	public void setCustomSamplePrice(float customSamplePrice) {
-		this.customSamplePrice = customSamplePrice;
-	}
-
-	public float getCustomEditPrice() {
-		return customEditPrice;
-	}
-
-	public void setCustomEditPrice(float customEditPrice) {
-		this.customEditPrice = customEditPrice;
-	}
-
-	public float getCustomEdit3Price() {
-		return customEdit3Price;
-	}
-
-	public void setCustomEdit3Price(float customEdit3Price) {
-		this.customEdit3Price = customEdit3Price;
-	}
-
-	public float getCustomTextPrice() {
-		return customTextPrice;
-	}
-
-	public void setCustomTextPrice(float customTextPrice) {
-		this.customTextPrice = customTextPrice;
-	}
-	
-	public float getRefundPer() {
-		return refundPer;
-	}
-
-	public void setRefundPer(float refundPer) {
-		this.refundPer = refundPer;
-	}
-
-	public float getJoinCash() {
-		return joinCash;
-	}
-
-	public void setJoinCash(float joinCash) {
-		this.joinCash = joinCash;
-	}
-
-	public float getPointPer() {
-		return pointPer;
-	}
-
-	public void setPointPer(float pointPer) {
-		this.pointPer = pointPer;
-	}
-
-	public String getLasUpdusrId() {
-		return lasUpdusrId;
-	}
-
-	public void setLasUpdusrId(String lasUpdusrId) {
-		this.lasUpdusrId = lasUpdusrId;
-	}
-
-	public String getLastUpdtPnttm() {
-		return lastUpdtPnttm;
-	}
-
-	public void setLastUpdtPnttm(String lastUpdtPnttm) {
-		this.lastUpdtPnttm = lastUpdtPnttm;
-	}
-
-	public String getJoinCertType() {
-		return joinCertType;
-	}
-
-	public void setJoinCertType(String joinCertType) {
-		this.joinCertType = joinCertType;
-	}
-
-	public String getSmsNoti() {
-		return smsNoti;
-	}
-
-	public void setSmsNoti(String smsNoti) {
-		this.smsNoti = smsNoti;
-	}
-
-	public String getEmailNoti() {
-		return emailNoti;
-	}
-
-	public void setEmailNoti(String emailNoti) {
-		this.emailNoti = emailNoti;
-	}
-	
-	public String getSlackNoti() {
-		return slackNoti;
-	}
-
-	public void setSlackNoti(String slackNoti) {
-		this.slackNoti = slackNoti;
-	}
-
-	public float getKakaoAtPrice() {
-		return kakaoAtPrice;
-	}
-
-	public void setKakaoAtPrice(float kakaoAtPrice) {
-		this.kakaoAtPrice = kakaoAtPrice;
-	}
-
-	public float getKakaoFtPrice() {
-		return kakaoFtPrice;
-	}
-
-	public void setKakaoFtPrice(float kakaoFtPrice) {
-		this.kakaoFtPrice = kakaoFtPrice;
-	}
-
-	public float getFaxPrice() {
-		return faxPrice;
-	}
-
-	public void setFaxPrice(float faxPrice) {
-		this.faxPrice = faxPrice;
-	}
-
-	public String getHoliSmishingNoti() {
-		return holiSmishingNoti;
-	}
-
-	public void setHoliSmishingNoti(String holiSmishingNoti) {
-		this.holiSmishingNoti = holiSmishingNoti;
-	}	
 	
 }
src/main/resources/egovframework/sqlmap/let/mjo/kakao/KakaoSent_SQL_Mysql.xml
--- src/main/resources/egovframework/sqlmap/let/mjo/kakao/KakaoSent_SQL_Mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/mjo/kakao/KakaoSent_SQL_Mysql.xml
@@ -909,7 +909,7 @@
 	
 	<!-- 발신 내용 상세보기 조회 (상세보기 버튼 클릭시)-->
 	<select id="KakaoSentDAO.selectKakaoSentDetailDataAjax" parameterClass="mjonKakaoATVO" resultClass="mjonKakaoATVO">
-
+        /* KakaoSentDAO.selectKakaoSentDetailDataAjax */
 		SELECT
 		    MGD.MSG_GROUP_ID				as msgGroupId
 			, MGD.USER_ID					as userId
@@ -925,9 +925,14 @@
 			, MD.BIZ_KAKAO_RESEND_DATA		as bizKakaoResendData 		/* 대체 문자 (치환O) */
 			, MGD.BIZ_KAKAO_RESEND_ORGNL_TXT   as bizKakaoResendOrgnlTxt /* 대체 문자( 치환X ) */
 			, MD.BIZ_KAKAO_JSON_FILE		as bizKakaoJsonFile			/* Json 파일 경로 */
+			, BA.CONTENTS as contents 
+			, MGD.AD_FLAG as adFlag
+			, MD.FILE_PATH1 as filePath1
 		FROM MJ_MSG_GROUP_DATA MGD
 				INNER JOIN MJ_MSG_DATA MD 
 					ON MGD.MSG_GROUP_ID = MD.MSG_GROUP_ID
+                LEFT JOIN BIZ_ATTACHMENTS BA  
+                    ON MD.BIZ_KAKAO_JSON_FILE  = BA.MSG_KEY 
 				AND MGD.USER_ID = MD.USER_ID
 		WHERE
 			MGD.USER_ID				= #userId#
@@ -1392,6 +1397,10 @@
 				       B.AT_DELAY_COMPLETE_YN      AS atDelayCompleteYn,
 				       A.MSG_NOTICETALK_SENDER_KEY AS msgNoticetalkSenderKey,
 				       A.BIZ_KAKAO_RESEND_YN       AS bizKakaoResendYn,
+						A.BIZ_KAKAO_RESEND_DATA as bizKakaoResendData,
+						B.AD_FLAG  as adFlag ,
+						B.BIZ_KAKAO_IMAGE_TYPE  as bizKakaoImageType ,
+						B.BIZ_KAKAO_RESEND_TYPE  as bizKakaoResendType ,
 				       CASE
 				           WHEN B.AT_DELAY_YN = 'Y' AND B.AT_DELAY_COMPLETE_YN = 'N'
 				               THEN DATE_ADD(B.REQ_DATE, INTERVAL -30 MINUTE)
@@ -1400,7 +1409,7 @@
 				       C.YELLOW_ID                 AS yellowId
 				FROM MJ_MSG_GROUP_DATA B
 			         INNER JOIN (
-					         select MSG_GROUP_ID, MSG_TYPE, DEL_FLAG, MSG_NOTICETALK_SENDER_KEY, BIZ_KAKAO_RESEND_YN
+					         select MSG_GROUP_ID, MSG_TYPE, DEL_FLAG, MSG_NOTICETALK_SENDER_KEY, BIZ_KAKAO_RESEND_YN, BIZ_KAKAO_RESEND_DATA
 						      from MJ_MSG_DATA
 						      where DEL_FLAG = 'N'
 						        <isNotEmpty property="tabType">
@@ -1663,23 +1672,96 @@
 	</select>
 	
 	<select id="KakaoSentDAO.selectKakaoSentCntEachCnt_advc" parameterClass="kakaoSentVO" resultClass="kakaoSentVO">
+		/* KakaoSentDAO.selectKakaoSentCntEachCnt_advc */
 		SELECT 
-		    a.MSG_GROUP_ID AS msgGroupId
-		    , a.successCount
-		    , a.waitCount
-		    , a.failCount
-		    , a.divideYn
-		    <isEqual property="bizKakaoResendYn" compareValue="Y">
-			    , a.kakaoResendSuccCount
-			    , a.kakaoResendFailCount
-			    , a.successCount * bkp.BIZ_KAKAO_AT_PRICE AS successPrice
-			    , (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE) AS kakaoResendSuccPrice
-		    </isEqual>
-		    <isNotEqual property="bizKakaoResendYn" compareValue="Y">
-		    	, 0 AS kakaoResendSuccCount
-			    , 0 AS kakaoResendFailCount
-			    , 0 AS successPrice
-			    , 0 AS kakaoResendSuccPrice
+			a.MSG_GROUP_ID AS msgGroupId
+			, a.successCount
+			, a.waitCount
+			, a.failCount
+			, a.divideYn
+			<isEqual property="bizKakaoResendYn" compareValue="Y">
+				, a.kakaoResendSuccCount
+				, a.kakaoResendFailCount
+				
+				
+				<isEqual property="msgType" compareValue="8">
+					, a.successCount * bkp.BIZ_KAKAO_AT_PRICE AS successPrice
+					, (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE) AS kakaoResendSuccPrice
+				
+				</isEqual>
+				
+				
+				
+				<isEqual property="msgType" compareValue="9">
+				
+					<!-- bizKakaoImageType 있음 + MMS -->
+					<isNotEmpty property="bizKakaoImageType">
+					
+					
+						<!-- 와이드 이미지인 경우 -->
+						<isEqual property="bizKakaoImageType" compareValue="W">
+							, a.successCount * bkp.BIZ_KAKAO_FT_WIDE_IMG_PRICE AS successPrice
+						</isEqual>
+					
+						<!-- 일반 이미지인 경우 -->
+						<isEqual property="bizKakaoImageType" compareValue="I">
+							, a.successCount * bkp.BIZ_KAKAO_FT_IMG_PRICE AS successPrice
+						</isEqual>
+					
+						, (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_PICTURE_PRICE) AS kakaoResendSuccPrice
+					</isNotEmpty>
+				
+
+					<!-- 이미지 없음 -->
+					<isEmpty property="bizKakaoImageType">
+					
+					
+						, a.successCount * bkp.BIZ_KAKAO_FT_PRICE AS successPrice
+						<isEqual property="bizKakaoResendType" compareValue="MMS">
+							, (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE) AS kakaoResendSuccPrice
+						</isEqual>
+				
+						<isEqual property="bizKakaoResendType" compareValue="SMS">
+							, (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * 0) AS kakaoResendSuccPrice
+						</isEqual>
+				
+						<!-- 예외: bizKakaoResendType 이 MMS, SMS 둘 다 아닐 경우 -->
+						<isNotEqual property="bizKakaoResendType" compareValue="MMS">
+							<isNotEqual property="bizKakaoResendType" compareValue="SMS">
+								, 0 AS kakaoResendSuccPrice
+							</isNotEqual>
+						</isNotEqual>
+					</isEmpty>
+				</isEqual>
+			</isEqual>
+			<isNotEqual property="bizKakaoResendYn" compareValue="Y">
+				, 0 AS kakaoResendSuccCount
+				, 0 AS kakaoResendFailCount
+				<isEqual property="msgType" compareValue="8">
+					, a.successCount * bkp.BIZ_KAKAO_AT_PRICE AS successPrice
+				</isEqual>
+				
+				<isEqual property="msgType" compareValue="9">
+					
+					<!-- 와이드 이미지인 경우 -->
+					<isEqual property="bizKakaoImageType" compareValue="W">
+						, a.successCount * bkp.BIZ_KAKAO_FT_WIDE_IMG_PRICE AS successPrice
+					</isEqual>
+				
+					<!-- 일반 이미지인 경우 -->
+					<isEqual property="bizKakaoImageType" compareValue="I">
+						, a.successCount * bkp.BIZ_KAKAO_FT_IMG_PRICE AS successPrice
+					</isEqual>
+				
+					<!-- 텍스트형(이미지 없음)인 경우 -->
+					<isNotEqual property="bizKakaoImageType" compareValue="W">
+						<isNotEqual property="bizKakaoImageType" compareValue="I">
+							, a.successCount * bkp.BIZ_KAKAO_FT_PRICE AS successPrice
+						</isNotEqual>
+					</isNotEqual>
+					
+				</isEqual>
+				, 0 AS kakaoResendSuccPrice
 		    </isNotEqual>
 		FROM (
 		    SELECT  
@@ -1738,8 +1820,19 @@
 		    , a.failCount
 		    , a.kakaoResendSuccCount
 		    , a.kakaoResendFailCount
-		    , a.successCount * bkp.BIZ_KAKAO_AT_PRICE                                        AS successPrice
-		    , (a.smsCnt      * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE)			 AS kakaoResendSuccPrice
+		    <!-- , a.successCount * bkp.BIZ_KAKAO_AT_PRICE                                        AS successPrice -->
+		    ,case
+	           when a.msgType = '8' then a.successCount * bkp.BIZ_KAKAO_AT_PRICE
+	           when a.msgType = '9' and a.bizKakaoImageType = 'I' then a.successCount * bkp.BIZ_KAKAO_FT_IMG_PRICE
+	           when a.msgType = '9' and a.bizKakaoImageType = 'W' then a.successCount * bkp.BIZ_KAKAO_FT_WIDE_IMG_PRICE
+	           when a.msgType = '9' and a.bizKakaoImageType != 'I' and a.bizKakaoImageType != 'W' then a.successCount * bkp.BIZ_KAKAO_FT_PRICE
+	       end successPrice
+		    <!-- , (a.smsCnt      * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE)			 AS kakaoResendSuccPrice -->
+		    ,case
+	           when a.msgType = '8' then (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE)
+	           when a.msgType = '9' and (a.bizKakaoImageType = 'I' or a.bizKakaoImageType = 'W') then a.mmsCnt * bkp.BIZ_PICTURE_PRICE
+	           when a.msgType = '9' and a.bizKakaoImageType != 'I' and a.bizKakaoImageType != 'W' then (a.smsCnt * bkp.BIZ_SMS_PRICE) + (a.mmsCnt * bkp.BIZ_MMS_PRICE)
+           end kakaoResendSuccPrice
 		    , a.divideYn
 		    , a.bizKakaoResendYn
 		    , MKPI.YELLOW_ID AS yellowId
@@ -1865,6 +1958,7 @@
 					, MD.BIZ_KAKAO_RESEND_YN as bizKakaoResendYn
 					, MD.MSG_NOTICETALK_SENDER_KEY
 					, MD.MSG_NOTICETALK_TMP_KEY
+					, MGD.BIZ_KAKAO_IMAGE_TYPE                                        as bizKakaoImageType
 				FROM     MJ_MSG_DATA MD
 		        inner join MJ_MSG_GROUP_DATA MGD on
 					MGD.MSG_GROUP_ID = MD.MSG_GROUP_ID
@@ -1896,6 +1990,7 @@
 	
 	<!-- 발신 내용 상세보기 조회 (상세보기 버튼 클릭시)-->
 	<select id="KakaoSentDAO.selectKakaoSentDetailViewPhoneAjax" parameterClass="mjonKakaoATVO" resultClass="mjonKakaoATVO">
+	/* KakaoSentDAO.selectKakaoSentDetailViewPhoneAjax */
 
 		SELECT
 		    MGD.MSG_GROUP_ID				as msgGroupId
src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_AT_SQL_Mysql.xml
--- src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_AT_SQL_Mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_AT_SQL_Mysql.xml
@@ -72,12 +72,16 @@
 			
 			, SMS_TXT
 			, BIZ_KAKAO_TITLE
+			, AD_FLAG
+			, FILE_PATH1
 			
 			, BIZ_KAKAO_RESEND_YN
 			, BIZ_KAKAO_RESEND_DATA
 			, BIZ_KAKAO_RESEND_TYPE
 			, BIZ_KAKAO_JSON_FILE
 			, REQ_DATE
+			
+			, FILE_CNT
 		)VALUES
 		<iterate conjunction=",">
 		(
@@ -95,12 +99,16 @@
 			
 			, #[].templateContent#
 			, #[].templateTitle#
+			, #[].adFlag#
+			, #[].filePath1#
 			
 			, #[].subMsgSendYn#
 			, #[].subMsgTxt#
 			, #[].subMsgType#
 			, #[].bizJsonName#
 			, #[].reqDate#
+			
+			, #[].fileCnt#
 		)
 		</iterate>
 	</insert>
@@ -131,6 +139,7 @@
             REQ_DATE,
             MSG_GROUP_CNT,
             MSG_TYPE,
+            AD_FLAG,
             
             AGENT_CODE,
             EACH_PRICE,
@@ -142,7 +151,8 @@
             
             AT_DELAY_YN,
             BIZ_KAKAO_RESEND_ORGNL_TXT,
-            BIZ_KAKAO_RESEND_TYPE
+            BIZ_KAKAO_RESEND_TYPE,
+            BIZ_KAKAO_IMAGE_TYPE
 		)VALUES
 		(
            #msgGroupId#,
@@ -153,6 +163,7 @@
            #reqDate#,
            #msgGroupCnt#,
            #msgType#,
+           #adFlag#,
 
            #agentCode#,
            #eachPrice#,
@@ -164,7 +175,8 @@
            
            #atDelayYn#,
            #bizKakaoResendOrgnlTxt#,
-           #bizKakaoResendType#
+           #bizKakaoResendType#,
+           #bizKakaoImageType#
 		)
 	</insert>
 	
@@ -178,6 +190,7 @@
 			, BIZ_KAKAO_FT_WIDE_IMG_PRICE
 			, BIZ_SMS_PRICE
 			, BIZ_MMS_PRICE
+			, BIZ_PICTURE_PRICE
 		)
 		VALUES
 		(
@@ -188,6 +201,7 @@
 			,#kakaoFtWideImgPrice#
 			,#smsPrice#
 			,#mmsPrice#
+			,#picturePrice#
 		)
 	</insert>
 	
@@ -222,6 +236,8 @@
 			MMD.USER_ID									AS userId
 			, MMD.MSG_GROUP_ID							AS msgGroupId
 			, MMD.MSG_SEQ								AS msgSeq
+			, MMGD.BIZ_KAKAO_IMAGE_TYPE 				AS bizKakaoImageType
+			, MMGD.EACH_PRICE			 				AS eachPrice
 			, MMD.USERDATA								AS userData
 			, MMD.REFUND_YN								AS refundYn
 			, MMD.RSLT_CODE								AS rsltCode
@@ -231,11 +247,14 @@
 			, DATE_FORMAT(MMD.RSLT_DATE,'%Y-%m-%d %T')	AS rsltDate
 			, MMD.BIZ_KAKAO_RESEND_YN					AS subMsgSendYn
 			, MMD.BIZ_KAKAO_RESEND_TYPE					AS subMsgType
-			
+			, MMD.FILE_CNT 								AS fileCnt
+			, MMD.BIZ_UMID								AS bizUmid
 		FROM
 			MJ_MSG_DATA MMD
 			INNER JOIN LETTNGNRLMBER MB
 			ON MMD.USER_ID = MB.MBER_ID
+			INNER JOIN mj_msg_group_data MMGD
+			on MMD.MSG_GROUP_ID = MMGD.MSG_GROUP_ID 
 		WHERE 1=1
 			AND	MMD.CUR_STATE          = '3'
 			AND	MMD.REFUND_YN          = 'N'
@@ -250,6 +269,15 @@
 			mj_msg_data
 		WHERE 1=1
 			AND USERDATA = #userData#
+	</select>
+	
+	<select id="kakaoAlimTalkDAO.selectBizLog" resultClass="kakaoVO" parameterClass="String">
+		SELECT
+			CALL_STATUS AS bizLogCallStatus
+		FROM
+			BIZ_LOG
+		WHERE 1=1
+			AND CMID = #umid#
 	</select>
 	
 	<!-- 카카오 전송 환불 프로시저 실행 (카카오 전송 성공 관련) -->
@@ -271,7 +299,7 @@
 	
 	<!-- 카카오 친구톡 전송 환불 프로시저 실행 (카카오 전송 성공 관련 - 대체문자 선택시 차액 환불 처리) -->
 	<procedure id="kakaoAlimTalkDAO.updateKakaoFtSend" parameterClass="kakaoVO">
-		{call kakaoFt_Send(#userId#, #msgGroupId#, #userData#)}
+		{call kakaoFt_Send(#userId#, #msgGroupId#, #userData#, #fileCnt#, #bizKakaoResendType#, #bizKakaoImageType#)}
 	</procedure>
 	
 	<!-- 카카오 친구톡 전송 환불 프로시저 실행 (카카오 전송 실패시 대체문자 관련 - 대체문자 발송 완료 된 경우) -->
src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_FT_SQL_Mysql.xml
--- src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_FT_SQL_Mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/mjo/kakao/Kakao_FT_SQL_Mysql.xml
@@ -37,7 +37,7 @@
 		       VALUES
 		       (	#friendId#
 		              , #userId#
-		              , #bizJsonName#
+		              , #atchFileId#
 		              , #templateName#
 		              , #imageFileName#
 		              , #templateImageUrl#
@@ -200,6 +200,7 @@
 		       A.IMAGE_TYPE                                    AS imageType,
 		       A.IMAGE_TITLE                                   AS imgTitle,
 		       A.IMAGE_LINK                                    AS imgLink,
+		       A.ATCH_FILE_ID                                  AS atchFileId,
 		       A.TEMPLATE_CONTENTS                             AS templateContent,
 		       A.AD_FLAG                                       AS adFlag,
 		       DATE_FORMAT(A.FRST_REGIST_PNTTM, '%Y-%m-%d %T') AS frstRegistPnttm,
@@ -215,6 +216,7 @@
 		               KFT.IMAGE_TYPE,
 		               KFT.IMAGE_TITLE,
 		               KFT.IMAGE_LINK,
+		               KFT.ATCH_FILE_ID,
 		               KFT.TEMPLATE_CONTENTS,
 		               KFT.AD_FLAG,
 		               KFT.FRST_REGIST_PNTTM,
src/main/resources/egovframework/sqlmap/let/msg/MjonMsgData_SQL_mysql.xml
--- src/main/resources/egovframework/sqlmap/let/msg/MjonMsgData_SQL_mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/msg/MjonMsgData_SQL_mysql.xml
@@ -2151,6 +2151,8 @@
 			, POINT_PER				AS pointPer
 			, KAKAO_AT_PRICE		AS kakaoAtPrice
 			, KAKAO_FT_PRICE		AS kakaoFtPrice
+			, KAKAO_FT_IMG_PRICE		AS kakaoFtImgPrice
+			, KAKAO_FT_WIDE_IMG_PRICE	AS kakaoFtWideImgPrice
 			, FAX_PRICE				AS faxPrice
 		FROM MJ_MBER_SETTING
 	
@@ -2169,6 +2171,8 @@
 			, PICTURE3_PRICE		AS picture3Price
 			, KAKAO_AT_PRICE		AS kakaoAtPrice
 			, KAKAO_FT_PRICE		AS kakaoFtPrice
+			, KAKAO_FT_IMG_PRICE	AS kakaoFtImgPrice
+			, KAKAO_FT_WIDE_IMG_PRICE	AS kakaoFtWideImgPrice
 			, FAX_PRICE				AS faxPrice
 			, USER_MONEY			AS userMoney
 			, USER_POINT			AS userPoint
@@ -7431,6 +7435,17 @@
 	
 	</update>
 	
+	<update id="mjonMsgDAO.updateRefundY" parameterClass="mjonMsgVO">
+		
+			UPDATE MJ_MSG_DATA
+				SET	REFUND_YN = 'Y'
+			WHERE
+				USER_ID				=	#userId#
+				AND USERDATA		=	#userData#
+				AND MSG_GROUP_ID	=	#msgGroupId#	
+	
+	</update>
+	
 	<update id="mjonMsgDAO.updateKakaoAtDelayCancelMsgGroupDataFlag" parameterClass="mjonMsgVO">
 		
 		UPDATE
src/main/resources/egovframework/sqlmap/let/msg/MjonMsgSent_SQL_mysql.xml
--- src/main/resources/egovframework/sqlmap/let/msg/MjonMsgSent_SQL_mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/msg/MjonMsgSent_SQL_mysql.xml
@@ -7,6 +7,7 @@
 <sqlMap namespace="Msg">
 	<typeAlias  alias="mjonMsgSWFDTO" type="itn.let.mjo.msgsent.service.MjonMsgSWFDTO"/>
 	<typeAlias  alias="mjonMsgDetailSentVO" type="itn.let.mjo.msgsent.service.MjonMsgDetailSentVO"/>
+	<typeAlias  alias="mjonMsgDetailSentDTO" type="itn.let.mjo.msgsent.service.MjonMsgDetailSentDTO"/>
 	<typeAlias  alias="mjonMsgSentVO" type="itn.let.mjo.msgsent.service.MjonMsgSentVO"/>
 	<typeAlias  alias="mjonMsgVO" type="itn.let.mjo.msg.service.MjonMsgVO"/>
 	<typeAlias  alias="addrGroupVO" type="itn.let.mjo.addr.service.AddrGroupVO"/>
@@ -389,7 +390,7 @@
 	
 	
 	<!-- 전체 발송결과 조회  (전송사별)-->
-	<select id="MjonMsgSentDAO.findByMsgDetailListAjax" parameterClass="mjonMsgDetailSentVO" resultClass="mjonMsgDetailSentVO">
+	<select id="MjonMsgSentDAO.findByMsgDetailListAjax" parameterClass="mjonMsgDetailSentVO" resultClass="mjonMsgDetailSentDTO">
 		/* MjonMsgSentDAO.findByMsgDetailListAjax*/
 		
 		SELECT
src/main/resources/egovframework/sqlmap/let/pay/MjonPay_SQL_mysql.xml
--- src/main/resources/egovframework/sqlmap/let/pay/MjonPay_SQL_mysql.xml
+++ src/main/resources/egovframework/sqlmap/let/pay/MjonPay_SQL_mysql.xml
@@ -3,10 +3,11 @@
   =========     =======    =================================================
   2021.03.01    신명섭 	
 -->
-<!DOCTYPE sqlMap PUBLIC "-//iBATIS.com//DTD SQL Map 2.0//EN" "http://www.ibatis.com/dtd/sql-map-config-2.dtd">
+<!DOCTYPE sqlMap PUBLIC "-//iBATIS.com//DTD SQL Map 2.0//EN" "http://ibatis.apache.org/dtd/sql-map-2.dtd">
 <sqlMap namespace="Pay">
 	<typeAlias  alias="mjonPayVO" type="itn.let.mjo.pay.service.MjonPayVO"/>
 	<typeAlias  alias="stVcVO" type="itn.let.mjo.pay.service.StVcVO"/>
+	<typeAlias  alias="bizKakaoPriceVO" type="itn.let.kakao.kakaoComm.BizKakaoPriceVO"/>
 	<typeAlias  alias="mjonVaMsgLogVO" type="itn.let.mjo.pay.service.MjonVaMsgLogVO"/>
 	
 	<!-- 공통 쿼리 부분 문자 발송 관련 -->
@@ -606,6 +607,7 @@
            MEMO,
            ORDER_ID,
            MSG_GROUP_ID
+           
        	)VALUES
        	(
            #userId#,
@@ -624,6 +626,22 @@
 	</update>
 	
 	
+	
+	<select id="mjonPayDAO.selectBizKakaoPrice" resultClass="bizKakaoPriceVO" parameterClass="String">
+			SELECT
+				BIZ_KAKAO_PRICE_ID          AS bizKakaoPriceId,
+				BIZ_KAKAO_AT_PRICE          AS bizKakaoAtPrice,
+				BIZ_KAKAO_FT_PRICE          AS bizKakaoFtPrice,
+				BIZ_KAKAO_FT_IMG_PRICE      AS bizKakaoFtImgPrice,
+				BIZ_KAKAO_FT_WIDE_IMG_PRICE AS bizKakaoFtWideImgPrice,
+				BIZ_SMS_PRICE               AS bizSmsPrice,
+				BIZ_MMS_PRICE               AS bizMmsPrice,
+				MSG_GROUP_ID                AS msgGroupId
+			FROM BIZ_KAKAO_PRICE
+			WHERE MSG_GROUP_ID = #msgGroupId#
+	</select>
+	
+	
 	<insert id="mjonPayDAO.insertMjPg" parameterClass="mjonPayVO">
 		INSERT INTO MJ_PG
        (
src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkSubMenuTap.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkSubMenuTap.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkSubMenuTap.jsp
@@ -90,6 +90,6 @@
     	<li id="tabStep" class="tab subTab"><button type="button" onclick="javascript:fnLinkPageTab('tabStep');">사용안내</button></li>
         <li id="tabProfile" class="tab subTab"><button type="button" onclick="javascript:fnLinkPageTab('tabProfile');">채널ID 등록/관리</button></li>
         <li id="tabTemplate" class="tab subTab"><button type="button" onclick="javascript:fnLinkPageTab('tabTemplate');">알림톡 템플릿 등록/관리</button></li>
-        <li id="tabSubFt" class="tab subTab"><button type="button" onclick="javascript:fnLinkPageTab('tabSubFt');">친구톡 템플릿 등록/관리</button></li>
+        <li id="tabSubFt" class="tab subTab"><button type="button" onclick="javascript:fnLinkPageTab('tabSubFt');">친구톡 템플릿 관리</button></li>
         
     </ul>
(No newline at end of file)
src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkTopMenuTap.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkTopMenuTap.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkTopMenuTap.jsp
@@ -143,7 +143,9 @@
 	<ul class="tabType4">
         <li id="tabAt" class="tab topTab"><button type="button" onclick="javascript:fnLinkPageTopTab('tabAlim');">알림톡</button></li>
 						<c:if test="${pageContext.request.serverName == 'localhost' 
-						            || pageContext.request.serverName == '119.193.215.98'}">
+						            || pageContext.request.serverName == '119.193.215.98'
+						            || pageContext.request.serverName == '192.168.0.176'
+						            }">
 	        <li id="tabFt" class="tab topTab"><button type="button" onclick="javascript:fnLinkPageTopTab('tabFriend');">친구톡</button></li>
        					 </c:if>
         <li id="tabConf" class="tab topTab"><button type="button" onclick="javascript:fnLinkPageTopTab('tabConf');">카카오톡 설정</button></li>
src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/at/KakaoAlimtalkMsgDataView.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/at/KakaoAlimtalkMsgDataView.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/at/KakaoAlimtalkMsgDataView.jsp
@@ -762,7 +762,7 @@
 // 				$('.loading_layer').removeClass('active');
 
 				// 프로그래스 바 종료
-				progressComplete();;
+				progressComplete();
 			}
 			,error: function (e) {
 				console.log("ERROR : ", e);
src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView.jsp
@@ -5,15 +5,17 @@
 <%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
 <%@ taglib prefix="ec" uri="/WEB-INF/tld/ecnet_tld.tld"%>
 <%@ page import="itn.com.cmm.LoginVO" %>
-<script src="/js/kakao/templateComm.js"></script>
+<script src="/js/kakao/ft/templateComm.js"></script>
 
 <script type="text/javascript" src="<c:out value='/js/MJUtill.js' />"></script>
 <script type="text/javascript" src="<c:out value='/js/kakao/ft/ftPriceClclt.js' />"></script>
-<script type="text/javascript" src="<c:out value='/js/kakao/ft/ftTabulator.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/ft/ftTabulator.js?v=20240328' />"></script>
 <script type="text/javascript" src="<c:out value='/js/txtSpecialReplace.js' />"></script>
-<script type="text/javascript" src="<c:out value='/js/kakao/at/addr.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/ft/addr.js' />"></script>
 <script type="text/javascript" src="<c:out value='/js/common/popup.js' />"></script>
 <script type="text/javascript" src="<c:out value='/js/kakao/ft/friendstalkExcel.js' />"></script>
+<!-- 주소록 유효성 체크 공통유틸로 인해 추가 -->
+<script type="text/javascript" src="<c:url value='/js/web/addr/cmn.js?date=202409021440'/>"></script>
 
 
 <script type="text/javascript">
@@ -25,12 +27,18 @@
 var fileExt = ""; // 첨부파일 확장자
 var excelAddr = []; //엑셀 불러오기에서 내용 저장하는 배열 변수
 
+var KAKAO_FT_PRICE;
+var KAKAO_FT_IMG_PRICE;
+var KAKAO_FT_WIDE_IMG_PRICE;
 $(document).ready(function (){
 	
+	KAKAO_FT_PRICE  = $('#kakaoFtPrice').val();
+	KAKAO_FT_IMG_PRICE  = $('#kakaoFtImgPrice').val();
+	KAKAO_FT_WIDE_IMG_PRICE  = $('#kakaoFtWideImgPrice').val();
 	//화면 초기 로딩시 화면 처리해주기 - 템플릿 불러오기 했을 경우를 위함
 	initFormChk();
 	
-	//이모티콘 삽입 기능 처리
+	//특수문자, 일괄변화문구 삽입 기능 처리
 	$(".symbolButton, .changeWord").on('click', function(){
 		
 		// 커서 위치에 삽입
@@ -40,18 +48,12 @@
 	
 	
 	//즉시 발송 라디오 버튼 선택시 숨김처리
-	$("#reserYnN").on('click', function(){
-		$('.rev_selected').hide();
-		$('.send_rev .send_content').css('padding-bottom','108px');
-		$('.send_btn .btnType:first-child').html('발송하기');
-		$('#bizForm #reserveYn').val($(this).val());
-	});
-	
-	//예약 발송 라디오 버튼 선택시 숨김 해제처리
-	$("#reserYnY").on('click', function(){
-		$('.rev_selected').show();
-		$('.send_rev .send_content').css('padding-bottom','0');
-		$('.send_btn .btnType:first-child').html('예약하기');
+	$('input[name="reserYn"]').on('change', function() {
+		const isReserve = $(this).val() === 'Y';
+
+		$('.rev_selected').toggle(isReserve);
+		$('.send_rev .send_content').css('padding-bottom', isReserve ? '0' : '108px');
+		$('.send_btn .btnType:first-child').html(isReserve ? '예약하기' : '발송하기');
 		$('#bizForm #reserveYn').val($(this).val());
 	});
 	
@@ -63,27 +65,29 @@
 		
 	});
 	
-	$("#excelAdd").click(function(){
-		
-		if(excelAddr.length > 0){
-			
-			//첨부파일로 불러온 데이터를 받는사람 목록에 추가해 준다.
-			addPhoneInfo(excelAddr);
-			
-			 //첨부파일 초기화 시켜주기
-		    $("#excelFile").val("");
-		    $("#excelNm").val("");
-			
-		}else{
-			
-			alert("추가 엑셀 데이터가 없습니다.");
-			return false;
-			
-		}
-		
+// 웹링크 버튼명 변경 시 미리보기에도 적용 (여러 개일 때 정확하게)
+	$(document).on('keyup', '#btnNmWeb', function(){
+		var inputTxt = $(this).val() || '웹링크';
+		// 웹링크용 인풋만 골라서 현재 인덱스를 구함
+		var $webNmInputs = $('input#btnNmWeb');
+		var idx = $webNmInputs.index(this);
+		// 미리보기 영역의 웹링크 버튼들 중, 동일한 인덱스의 버튼 텍스트를 바꿔줌
+		$('.btnViewArea .btn_kakao_type.btnEmpty[id^=btnViewWL]').eq(idx).text(inputTxt);
 	});
-	
-	
+	$(document).on('keyup', '#btnNmApp', function(){
+		var inputTxt = $(this).val() || '앱링크';
+		var $appNmInputs = $('input#btnNmApp');
+		var idx = $appNmInputs.index(this);
+		$('.btnViewArea .btn_kakao_type.btnEmpty[id^=btnViewAL]').eq(idx).text(inputTxt);
+	});
+
+	$(".friend_talk_wrap #ad_Y").click(function () {
+		if ($(this).is(":checked") == true) {
+
+			fnAgentCodeChg();
+		}
+		thisFnByteString($('#smsTxtArea').val());
+	});
 	$("#send_fail_check").change(function(){
 		if($("#send_fail_check").is(":checked")){
 
@@ -143,76 +147,298 @@
 	});
 	
 	
-	//대체문자 초기화 버튼 선택 처리
-	$("#failCheckInit").click(function(){
-		
-		$('#smsTxtArea').val("");
-		
-		// 금액 계산 fn 호출
-		totalFtPriceSum(tableL.getRows().length);
-		//fn_priceClclt();
-		// 문자 바이트수 체크
-		thisFnByteString($('#smsTxtArea').val());
-		// 에러버튼 체크 초기화
-		fn_insertErrorYN('N');
-		
+
+	// 광고포함여부 체크 이벤트
+	$("#ad_Y").click(function () {
+		const isChecked = $(this).prop("checked");
+		$('#adFlag').val(isChecked ? 'Y' : 'N');
 	});
+	
+
+	$("input[name='img_file_add']").change(function () {
+
+		imageInit();
+		
+
+		// 첨부가 없으면 
+		if ($('#sendFailImgSrc').attr('src') != null) {
+			$('#sendFailImg').hide();
+			$('#sendFailImgSrc').attr('src', '');
+			$('#atchFileId').val('');
+			thisFnByteString($('#smsTxtArea').val());
+		}
+		
+		
+		
+
+		//결제 금액 구하기
+	    totalFtPriceSum(tableL.getRows().length);
+	});
+	
+	// 최근 전송내역
+	resultLatestMsgList();
+	//자주보내는 번호
+	resultBookMarkMsgList();
+	
+	
+	
+
+	// 대체문자 초기화
+	$('#failCheckInit').on('click', function(){
+		console.log('::failCheckInit::');
+		$('#smsTxtArea').val('');
+		thisFnByteString('');
+		
+	 });
+	 
 	
 	
 });
 
-function initFormChk(){
+function imageInit(){
+
+	var imageUrl = $("#templateImageUrl").val();
 	
-	//첫로딩시 우측 미리보기 화면 숨김처리
-	$('.emphasis_title_text').hide();	//강조유형 타이틀
-	$('.emphasis_side_text').hide();	//강조유형 보조문구
-	//$('.template_text').hide();			//내용 미리보기
-	$('.side_info_text').hide();		//부가정보 내용
-	$('.channel_info_text').hide();		//채널추가 텍스트 내용
-	$('.btnAd').hide();					//샘플 채널추가 버튼
-	//$('.btnEmpty').hide();				//샘플 버튼
+	// 이미지가 없으면 function 종료
+	if(imageUrl == ''){
+		return false;
+	}
+	
+	// 이미지가 있으면 이미지 창 및 미리보기 창에 이미지 삭제 
+	$("#imgLink").val("");
+	$("#imgFile").val("");
+
+	$('#imgTitle').addClass('file_none')
+					.text("이미지 없음");
+
+	$('#imgLink').prop('disabled', false);  // 회색 처리 제거 + 입력 불가 제거
+	
+	$("#templateImageUrl").val("");
+	$('.kakao_image').css("display", "none");
+	$("#kakaoImg").attr("src", "");
+	$("#imageFileName").val("");	//이미지 파일명 삭제
+	
+}
+/* 
+function validateButtons() {
+	var isValid = true;
+
+	console.log($('#buttonAddWrap').html());
+	var val = '';
+	// button_add_wrap 하위 input들 중에서 비어있는 항목이 있는지 확인
+	$('#buttonAddWrap').find('input[type="text"]').each(function() {
+
+		var val = $(this).val().trim();
+		if (val === '') {
+			console.log($(this).prop('outerHTML'));
+			isValid = false;
+			return false; // .each 루프 중단
+		}
+	});
+	
+	if (!isValid) {
+		alert('버튼명 및 링크를 입력해주세요.');
+		return false;
+	}else if (!val.startsWith('http://') && !val.startsWith('https://') ) {
+		alert('버튼의 링크는 http:// 또는 https:// 형식으로 입력해주세요.');
+		$(this).focus();
+		return false;
+	}
+
+	return true;
+} */
+
+function validateButtons() {
+	var isValid = true;
+
+	$('.kakaoBtnNmList').each(function() {
+		var val = $(this).val().trim();
+
+		// 빈 값 체크
+		if (val === '') {
+			alert('버튼명을 모두 입력해주세요.');
+			$(this).focus();
+			isValid = false;
+			return false; // .each 루프 중단
+		}
+	
+	});
+	
+	if(isValid){
+		$('.kakaoBtnList').each(function() {
+			var val = $(this).val().trim();
+	
+			// 빈 값 체크
+			if (val === '') {
+				alert('링크를 모두 입력해주세요.');
+				$(this).focus();
+	 			isValid = false;
+				return false; // .each 루프 중단
+			}
+	
+			// http 형식 체크
+			if (!val.startsWith('http://') && !val.startsWith('https://')) {
+				alert('버튼의 링크는 http:// 또는 https:// 형식으로 입력해주세요.');
+				$(this).focus();
+	 			isValid = false;
+				return false; // .each 루프 중단
+			}
+		});
+	}
+
+
+	return isValid;
+}
+
+//최근 전송내역
+function resultLatestMsgList(){
+
+  
+  $.ajax({
+      type:"POST",
+      url:"/web/mjon/msgdata/resultLatestMsgListAjax.do",
+      data:{},
+      dataType:'json',
+      // timeout:(1000*30),
+      success:function(data){
+          console.log('resultLatestMsgList : ',data.object);
+          
+
+          // 가져온 데이터 배열
+          let resultList = data.object; 
+          let $latestMsgUl = $('#latestMsgUl'); // 기존 리스트 UL
+
+          // 기존 내용을 비우기
+          $latestMsgUl.empty();
+
+          // 데이터가 있는 경우
+          if (resultList && resultList.length > 0) {
+              resultList.forEach(function(item, index) {
+                  let listItem = 
+                      '<li id="latestLi">' +
+                          '<input type="checkbox" id="addrChk_' + (index + 1) + '" name="latAddrChk" value="' + item.callTo + '">' +
+                          '<label for="addrChk_' + (index + 1) + '" class="label">최근 전송내역</label>' +
+                          '<p>' + item.callTo + '</p>' +
+                          '<button type="button" id="latestAddrDel">' +
+                              '<img src="/publish/images/popup/close3.png" alt="전화번호 삭제">' +
+                          '</button>' +
+                      '</li>';
+                  
+                  $latestMsgUl.append(listItem);
+              });
+          } else {
+              // 데이터가 없는 경우
+              $latestMsgUl.append('<li><p>최근 발송 내역이 없습니다.</p></li>');
+          }
+      },
+      error:function(request , status, error){
+          console.log(' error ?');
+          console.log('request : ', request);
+          console.log('status : ', status);
+      }
+  });
+}
+
+//자주보내는 번호
+function resultBookMarkMsgList(){
+
+  $.ajax({
+      type:"POST",
+      url:"/web/mjon/msgdata/resultBookMarkMsgList.do",
+      data:{},
+      dataType:'json',
+      // timeout:(1000*30),
+      success:function(data){
+          console.log('resultBookMarkMsgList : ',data.object);
+          
+
+          // 가져온 데이터 배열
+          let resultList = data.object; 
+          let $bookMsgUl = $('#bookMsgUl'); // 기존 리스트 UL
+
+          // 기존 내용을 비우기
+          $bookMsgUl.empty();
+
+          // 데이터가 있는 경우
+          if (resultList && resultList.length > 0) {
+              resultList.forEach(function(item, index) {
+//                console.log(item.addrPhoneNo + " : " + item.addrPhoneNo);
+                  let listItem = 
+                      '<li id="bookMarkLi">' +
+                          '<input type="checkbox" id="bokAddrChk_' + (index + 1) + '" name="bookAddrChk" value="' + item.addrPhoneNo + '">' +
+                          '<label for="bokAddrChk_' + (index + 1) + '" class="label">최근 전송내역</label>' +
+//                           '<p>' + item.addrPhoneNo + '</p>' +
+                          '<p>' + item.addrPhoneNo + '</p>' +
+                          '<button type="button" id="bookMarkAddrDel">' +
+                              '<img src="/publish/images/popup/close3.png" alt="전화번호 삭제">' +
+                          '</button>' +
+                      '</li>';
+                  
+                  $bookMsgUl.append(listItem);
+              });
+          } else {
+              // 데이터가 없는 경우
+              $bookMsgUl.append('<li><p>등록된 자주 보내는 번호 내역이 없습니다..</p></li>');
+          }
+      },
+      error:function(request , status, error){
+          console.log(' error ?');
+          console.log('request : ', request);
+          console.log('status : ', status);
+      }
+  });
+}
+
+function initFormChk(){
+	console.log(':: initFormChk ::');
 	
 	//채널ID 정보 미리보기에 표시해주기
 	fnAgentCodeChg();
 	
 	//템플릿 등록 이미지가 있는 경우 이미지 정보 표시해 주기
-	var imgType = $("input[name=img_file_add]:checked").val();
-	
-	if(!imgType == ''){
+	var imgType = $('#imageType').val();
+
+	$("#img_file_0").prop('checked',true);
+	if(imgType){
 		
-		if($("input[name='img_file_add']:checked").attr("id")=="img_file_1"){
-			$(".basic_img_add_wrap").show().siblings(".img_add_info_wrap").hide();
-			$(".img_file_add_wrap").show();
-		}else if($("input[name='img_file_add']:checked").attr("id")=="img_file_2"){
-			$(".wide_img_add_wrap").show().siblings(".img_add_info_wrap").hide();
-			$(".img_file_add_wrap").show();
-		} else{
+		if(imgType == 'I'){
+			
+			$('#img_file_1').click();
+			
+		}else if(imgType == 'W'){
+			$('#img_file_2').click();
+		} 
+
+		// 이미지명 입력
+		$('#imgTitle').removeClass('file_none')
+						.text($('#imageFileName').val());
+		
+		
+		$('#imgLink').val($('#imgLinkTemp').val());  // 이미지 link
+		$('#imgLink').prop('disabled', true);  // 회색 처리 + 입력 불가
+		// 이미지 URL은 해당 input에 있음.
+		
+		/* else{
 			$(".img_add_info_wrap, .img_file_add_wrap").hide();
-		}
-		
-	}else{
-		
-		$("#img_file_0").prop('checked',true);
+		} */
 		
 	}
-	var imageFileName = '${resultTemplateVO.imageFileName}';
-	$("#imgNm").text(imageFileName);
 	
-	//미리보기에 이미지 표시해주기
-	var imgUrl = '${resultTemplateVO.templateImageUrl}';
+	var imgUrl = $('#templateImageUrl').val();
 	
 	if(imgUrl != ''){
 		$('.kakao_image').css("display", "block");
 		$("#kakaoImg").attr("src", imgUrl);
+		$("#kakaoImg").attr("src", imgUrl);
+
+		$('#sendFailImg').show();
+		$('#sendFailImgSrc').attr('src', imgUrl);
+		
 	}
 	
 	//초기 템플릿 내용 글자수 처리해주기
 	var tmpContents = $('#inputTemplateContent').val();
 	setContentsLengForFriends(tmpContents);
-	
-	//초기 광고포함 여부 처리
-	var adFlagVal = $("input[name=ad_flag]:checked").val();
-	advTextChange(adFlagVal);
 	
 	var reserYn = $("input[name=reserYn]:checked").val();
 	if(reserYn == 'N'){
@@ -224,6 +450,16 @@
 		
 	}
 	
+	
+	// 광고포함여부 체크
+	// 기본
+	// 	- adFlag = Y
+	// 	- ad_Y = Y
+	if($('#adFlag').val() == 'N'){
+		$('#ad_Y').click();
+	}
+	
+	
 }
 
 //우측 미리보기 화면 채널 아이디 정보 변경해 주기
@@ -231,16 +467,42 @@
 	var yellowIdVal = $("#selectAgentCode option:checked").val();
 	var yellowId = $("#selectAgentCode option:checked").text();
 	var yellowIdAt = yellowId.replaceAll('@','');
+	
+	// 체크 여부에 따라 채널ID 문구 설정
+	
 	if(yellowIdVal != ''){
 		
-		$('#spnYellowid').text(yellowId);			//미리보기화면 채널ID
-		$('#spnYellowidAt').text(yellowIdAt);		//미리보기화면 채널ID @제거
-		$('#yellowIdCon').text(yellowIdAt);			//내용 입력 부분 채널ID 
+		$('#spnYellowid').text(yellowId);	// 미리보기화면 최상단 채널ID
+		
+		var cleanYellowId = yellowId.replace(/^@/, ""); // 앞에 @ 기호 제거
+		if ($("#ad_Y").is(":checked")) {
+			
+			$("#adTxt").show(); // 내용
+// 			$("#adTxt p").html("(광고) " + cleanYellowId + ""); // 내용 
+			$("#adTxt p").html("(광고)"); // 내용 
+			
+// 			$("#talkTitle").html("<span>(광고)</span> " + cleanYellowId); // 미리보기 화면 상단
+			$("#talkTitle").html("<span>(광고)</span> "); // 미리보기 화면 상단
+		} 
+		
 	}else{
 		
-		$('#spnYellowid').text("채널ID");			//미리보기화면 채널ID
-		$('#spnYellowidAt').text("채널ID");			//미리보기화면 채널ID @제거
-		$('#yellowIdCon').text("채널ID");			//내용 입력 부분 채널ID 
+		var channelId = "채널ID";
+		
+		$('#spnYellowid').text(channelId);	// 미리보기화면 최상단 채널ID
+		
+		if ($("#ad_Y").is(":checked")) {
+			
+			$("#adTxt").show(); // 내용
+// 			$("#adTxt p").html("(광고) " + channelId + ""); // 내용 
+			$("#adTxt p").html("(광고)"); // 내용 
+			
+// 			$("#talkTitle").html("<span>(광고)</span> " + channelId); // 미리보기 화면 상단
+			$("#talkTitle").html("<span>(광고)</span>"); // 미리보기 화면 상단
+		} else {
+			$("#adTxt").hide(); // 내용
+			$("#talkTitle").html(channelId); // 미리보기 화면 상단
+		}
 	}
 }
 
@@ -248,12 +510,12 @@
 function upImgClick(){
 	var selectChennelID = $("select[name='selectAgentCode']").val();	//채널ID 정보
 	var imageType = $("input[name=img_file_add]:checked").val();		//이미지 종류 정보(일반, 와이드)
-	var title = $("#imgTitle").val();
+	var title = $("#imgTitle").text();
 	var link = $("#imgLink").val();
 	
 	if(selectChennelID == ''){
 		
-		$("#imgNm").text("");
+		$("#imgTitle").text("");
 		$("#imgFile").val("");
 		alert("채널ID를 선택해 주세요.");
 		return false;
@@ -266,43 +528,41 @@
 		
 	}else{
 		
-		$("#imgNm").text("");
+		$("#imgTitle").text("");
 		$("#imgFile").val("");
 		alert("이미지 종류를 선택해 주세요.");
 		return false;
 		
 	}
-	
+	/* 
 	if(title == ''){
 		
-		$("#imgNm").text("");
+		$("#imgTitle").text("");
 		$("#imgFile").val("");
 		alert("이미지 제목을 입력해 주세요.");
 		return false;
 		
-	}
+	} */
 	
 	//이미지 선택시 링크 정보 및 url 패턴 검사
-	if(link == ''){
-		
-		$("#imgNm").text("");
-		$("#imgFile").val("");
-		alert("이미지 클릭시 이동할 URL 주소를 http:// 또는 https:// 포함하여 입력해 주세요.");
-		return false;
-		
-	}else{
-		
-		if(link.search("http://") == -1 && link.search("https://") == -1){
-			
-			$("#imgNm").text("");
-			$("#imgFile").val("");
-			alert("이미지 URL 주소에는 http:// 또는 https://를 포함하여 입력해야 합니다.");
-			return false;
-			
+	console.log('link: ', link);
+	if (!link) {
+		if(!confirm("먼저 URL을 입력하지 않으면 이미지에 URL이 저장되지 않습니다. 계속하시겠습니까?")){
+			return false;	
 		}
-		
+	}else if(!link.startsWith('http://') && !link.startsWith('https://')){
+		alert('이동할 URL은 http:// 또는 https:// 형식으로 입력해주세요.');
+		return false;	
 	}
-	
+	/* 
+	else if(link.search("http://") == -1 && link.search("https://") == -1){
+
+		$("#imgTitle").text("");
+		$("#imgFile").val("");
+		alert("이미지 URL 주소에는 http:// 또는 https://를 포함하여 입력해야 합니다.");
+		return false;
+	}
+ */	
 	//첨부파일 선택 팝업 호출해주기
 	$("#imgFile").click();
 }
@@ -321,7 +581,7 @@
 			return false;
 	   	}
 	   
-		$("#imgNm").text(fileName); 
+		$("#imgTitle").text(fileName); 
 	}
 	
 	//알림톡 이미지 등록 요청하기
@@ -350,18 +610,17 @@
 
    	}
 	
-	var url = "";
 	
-	if(fileExt == "jpg" || fileExt == "jpeg" || fileExt == "png"){
-		
-		url = "/web/mjon/kakao/template/sendKakaoFriendsTemplateImageUploadAjax.do";
-		
-	}else{
-		
-		alert('jpg, jpeg 파일만 업로드 할수 있습니다.');
-		return;
-		
+// 	if(fileExt == "jpg" || fileExt == "jpeg" || fileExt == "png"){
+	console.log('fileExt : ', fileExt);
+	if(fileExt != null && fileExt !== "jpg" && fileExt !== "jpeg" && fileExt !== "png"){
+	    alert("jpg, jpeg, png 파일만 업로드 할 수 있습니다.");
+	    return;
 	}
+
+	
+    var url = url = "/web/mjon/kakao/template/sendKakaoFriendsTemplateImageUploadAjax_advc.do";
+//     var url = url = "/web/mjon/kakao/template/sendKakaoFriendsTemplateImageUploadAjax.do";
 	
 	$.ajax({
 		type : 'POST'
@@ -373,54 +632,51 @@
 		, processData: false
 		, contentType: false
 		, cache : false
-		, success : function(returnData, status){
-			if(returnData.result == "success") {
+		, success : function(returnData){
+			console.log('returnData : ', returnData);
+			
+
+
+			if (returnData.status === 'OK' || returnData.status === 200) {
+				alert(returnData.message);
 				
-				var code = returnData.code;
-				var msg = returnData.msg;
-				var imgUrl = returnData.imgUrl;
+				$('#imgTitle').removeClass('file_none')
+								.text(fileName);
+				$('#imgLinkTemp').val($('#imgLink').val());  // 회색 처리 + 입력 불가
+				$('#imgLink').prop('disabled', true);  // 회색 처리 + 입력 불가
+
+
+				$("#imageFileName").val(fileName);
 				
-				if(code == "200"){//등록 성공시 메시지 및 이미지 표시 처리
-					alert("이미지 등록이 완료 되었습니다.");
-					$("#templateImageName").val(fileName);
-					$("#templateImageUrl").val(imgUrl);
 				
-					$('.kakao_image').css("display", "block");
-					$("#kakaoImg").attr("src", imgUrl);
-					$("#imageFileName").val(fileName);
+
+				// imgUrl은 object에 포함된 경우만 처리
+				if (returnData.object) {
+					$("#templateImageUrl").val(returnData.object.imgUrl);
+					$(".kakao_image").css("display", "block");
+					$("#kakaoImg").attr("src", returnData.object.imgUrl);
 					
-				}else if(code == "2002"){//일반 이미지 가로 세로 크기 오류 발생
 					
-					alert("권장 이미지 사이즈는  가로 720px 세로 720px(가로:세로 비율이 2:1 아닐 경우, 가로 500px, 세로 250px 이하일 경우 업로드 불가), 최대용량 500KB 입니다.");
-					
-					//첨부파일 정보를 지워준다.
-					$("#imgFile").val("");
-				    $("#imgNm").text("");  
-					return false;
-					
-				}else if(code == "405"){
-					
-					alert("권장 이미지 사이즈는  가로 800px 세로 600px(가로:세로 비율이 2:1 아닐 경우, 가로 500px, 세로 250px 이하일 경우 업로드 불가), 최대용량 2MB 입니다.");
-					
-					//첨부파일 정보를 지워준다.
-					$("#imgFile").val("");
-				    $("#imgNm").text("");  
-					return false;
-					
-				}else{
-					
-					alert(msg);
-					
-					//첨부파일 정보를 지워준다.
-					$("#imgFile").val("");
-				    $("#imgNm").text(""); 
-					return false;
+					// 대체문자 이미지
+					$("#atchFileId").val(returnData.object.atchFileId);
+					// 대체문자
+					$('#sendFailImg').show();
+					$('#sendFailImgSrc').attr('src', returnData.object.imgUrl);
+					// 대체문자 포토형으로 수정 
+// 					$('.put_left').removeClass("short"); //내용 입력 박스 클래스 삭제
+// 					$('.put_left').removeClass("long"); //내용 입력 박스 클래스 삭제
+// 					$('.put_left').addClass('photo'); 
+					setSendFailTemplate('P')
 				}
-				
-			}else if(returnData.result == "loginFail"){
-				location.href="/web/user/login/login.do";
-			}else{
-				alert("이미지 등록 중 오류가 발생하였습니다.!!!");
+
+			} else {
+				alert(returnData.message || "이미지 등록 중 오류가 발생하였습니다.");
+
+				// 첨부파일 정보 초기화
+				$("#imgFile").val("");
+
+				$('#imgTitle').addClass('file_none')
+								.text("이미지 없음");
 				return false;
 			}
 		},
@@ -441,13 +697,37 @@
 	$("#imgFile").val("");
 }
 
+function setSendFailTemplate(p_type){
+	if(p_type == 'P'){
+		// 대체문자 포토형으로 수정 
+		$('.put_left').removeClass("short"); //내용 입력 박스 클래스 삭제
+		$('.put_left').removeClass("long"); //내용 입력 박스 클래스 삭제
+		$('.put_left').addClass('photo'); 
+
+		$('.msg_com').removeClass("msg_short"); //단문 클래스 삭제하고
+		$('.msg_com').removeClass("msg_long"); //단문 클래스 삭제하고
+		$('.msg_com').addClass("msg_photo"); //단문 클래스 삭제하고
+		
+
+		$('#limitLeng').html("2000");
+		$('.msg_com').html("그림");
+		
+	}
+	
+}
+
 function fnImageDel(){
 	
-	var selectAgentCode = $("select[name='selectAgentCode']").val();	// 선택 채널ID
-	var imageUrl = $("#templateImageUrl").val();
+	if(confirm("등록된 이미지를 삭제하시겠습니까?")){
+		imageInit();
+	}
+	
+	
+	
+	/* var imageUrl = $("#templateImageUrl").val();
 	var bizForm = document.bizForm;
 	
-	bizForm.senderKey.value = selectAgentCode;
+	bizForm.senderKey.value = $("select[name='selectAgentCode']").val();	// 선택 채널ID
 	
 	if(imageUrl == ''){
 		
@@ -457,7 +737,7 @@
 	}
 	
 	var tmpFriendId = $("#tmpFriendId").val();
-	
+	console.log(' + tmpFriendId :: ', tmpFriendId);
 	if(tmpFriendId != ''){
 		
 		if(!confirm("불러온 템플릿의 이미지도 함께 삭제 됩니다. 계속하시겠습니까?")){
@@ -491,15 +771,25 @@
 					
 					if(code == "200"){
 						alert("등록된 이미지가 삭제되었습니다.");
-						$("#imgTitle").val("");
 						$("#imgLink").val("");
 						$("#imgFile").val("");
-						$("#imgNm").text("첨부파일 이미지");
+
+						$('#imgTitle').addClass('file_none')
+										.text("이미지 없음");
+
+						$('#imgLink').prop('disabled', false);  // 회색 처리 제거 + 입력 불가 제거
+						
 						$("#templateImageUrl").val("");
 						$('.kakao_image').css("display", "none");
 						$("#kakaoImg").attr("src", "");
 						$("#imageFileName").val("");	//이미지 파일명 삭제
 						$("#img_file_0").click();		//이미지 첨부 안함 선택
+						
+						
+
+						
+						
+						
 						return false;
 						
 					}else{
@@ -516,7 +806,7 @@
 				alert("이미지 삭제에 오류가 발생하였습니다.");
 			}
 		});
-	}
+	} */
 }
 
 //알림톡 내용 초기화 해주기
@@ -539,16 +829,9 @@
 	
 }
 
-//광고성 정보 포함 선택에 따른 텍스트 표시 부분 처리
-$(document).on('change', '.inputAdFlag', function(){
-	
-	var adFlagVal = $(this).val();
-	advTextChange(adFlagVal);
-	
-});
-
+/* 
 function advTextChange(adFlagVal){
-	
+	console.log('adFlagVal : ', adFlagVal);
 	if(typeof adFlagVal == "undefined" || adFlagVal == null || adFlagVal == ""){
 		
 		$("#ad_Y").prop('checked',true);
@@ -557,18 +840,16 @@
 		
 		$('.adFlag').hide();
 		$('.kakao_block_text').hide();
-		$('#adFlagFront').hide();
 		$('.adFlagAfter').hide();
 		
 	}else{
 		$('.adFlag').show();
 		$('.kakao_block_text').show();
-		$('#adFlagFront').show();
 		$('.adFlagAfter').show();
 	}
 	
 }
-
+ */
 
 //친구톡 내용 템플릿으로 저장하기 - 내용 저장하기
 function myTemplateSave(){
@@ -577,9 +858,8 @@
 	var yellowId = $("select[name='selectAgentCode'] option:selected").text().replaceAll('@','');			// 선택 채널명
 	var inputTemplateName = $("#inputTemplateName").val();				// 입력 템플릿 이름
 	var imageType = $("input[name=img_file_add]:checked").val();		// 첨부 이미지 종류(없음, 일반, 와이드 이미지)
-	var imageTitle = $("#imgTitle").val();								//첨부이미지 제목
+	var imageTitle = $("#imgTitle").text();								//첨부이미지 제목
 	var imageLink = $("#imgLink").val();								//첨부이미지 클릭시 이동 링크 주소
-	var inputTemplateAd = $("input[name=ad_flag]:checked").val();	// 광고성메시지 선택 여부
 	var inputTemplateContent = $("#inputTemplateContent").val();		// 템플릿 내용
 	
 	var inputTemplateImageName = $("#templateImageName").val();	// 템플릿 이미지 파일명
@@ -606,7 +886,7 @@
 			
 		}
 		
-		if(imageLink == ''){
+	/* 	if(imageLink == ''){
 			
 			alert("이미지 클릭시 이동할 URL을 입력해 주세요.");
 			return false;
@@ -620,7 +900,7 @@
 				
 			}
 			
-		}
+		} */
 		
 		if(inputTemplateImageUrl == ''){
 			
@@ -642,12 +922,18 @@
 	//bizForm.yellowId.value = yellowId;
 	bizForm.templateName.value = inputTemplateName;
 	//bizForm.imageType.value = imageType;
-	//bizForm.imgTitle.value = imageTitle;
-	//bizForm.imgLink.value = imageLink;
-	bizForm.adFlag.value = inputTemplateAd;
+	
+	bizForm.imgLink.value = imageLink;
+	console.log('imageLink : ', imageLink);
+	console.log('bizForm.imgLink.value : ', bizForm.imgLink.value);
+	
 	bizForm.templateContent.value = inputTemplateContent;
 	
 	var data = new FormData(bizForm);
+	data.append("imgLink", imageLink);
+	data.forEach((value, key) => {
+		console.log(key, value);
+	});
 	
 	if(confirm("템플릿을 저장하시겠습니까?")){
 		
@@ -740,7 +1026,7 @@
 		
 		if(resMin >= 50 && resMin <= 59){
 			
-			alert("친구톡은 20:50 ~ 익일 08:00까지 발송할 수 없습니다.");
+			alert("친구톡은 20시 50분부터 익일 08시까지 발송이 제한됩니다.");
 			$("#msgResMin").val("40").prop("selected", true);
 			
 		}
@@ -769,6 +1055,12 @@
 
 //친구톡 데이터 전송 처리
 function fn_sendMsgData(){
+	
+	// 유효성 검사 실패 시 저장 동작 중단
+/* 	if (!validateLinks()) {
+		return false;
+	}
+	 */
 	
 	var senderKey = $("#selectAgentCode option:checked").val();
 	
@@ -819,238 +1111,13 @@
 	}
 	
 
-	//수신번호 리스트 체크하기
-	var numCnt = 0;
-	var nameList = [];		//치환문자 이름
-	var phoneNum = [];	//받는사람
-	var rep1List = [];		//치환문자1
-	var rep2List = [];		//치환문자2
-	var rep3List = [];		//치환문자3
-	var rep4List = [];		//치환문자4
-	
-	var varValList = [];		//치환문자 연결시킬 변수 셋팅
 	
 	var selectedData = tableL.getRows();
-	
-	var varValStatus = true; //치환분자 데이터 체크용
 	
 	if(selectedData == "" || selectedData == null){
 		
 		alert("받는사람 주소를 한 건 이상 입력해주세요.");
 		return false;
-	
-	}else{ // 선택한 Row '-' 문자 삭제하기
-		
-		var txtReplYn = $("#txtReplYn").val();
-		
-		if(txtReplYn == 'Y'){//치환문자가 있는 경우 변수 치환 처리
-		
-			for(var i=0; i < selectedData.length; i++){
-				
-				var nmStatus = false;
-				var rep1Status = false;
-				var rep2Status = false;
-				var rep3Status = false;
-				var rep4Status = false;
-				
-				
-				if(tmpContents.indexOf("\#{이름}")  > -1){
-					
-					nmStatus = true;
-				}
-				
-				if(tmpContents.indexOf("\#{1}")  > -1){
-					
-					rep1Status = true;
-				}
-				
-				if(tmpContents.indexOf("\#{2}")  > -1){
-					
-					rep2Status = true;
-				}
-				
-				if(tmpContents.indexOf("\#{3}")  > -1){
-					
-					rep3Status = true;
-				}
-				
-				if(tmpContents.indexOf("\#{4}")  > -1){
-					
-					rep4Status = true;
-				}
-				
-				//일괄변환 문자에 콤마(,)가 들어가있으면 배열로 넘길때 문제가 발생하여 특수문자(§)로 치환하여 넘겨주도록 한다.
-				var name = tableL.getRows()[i].getData().name;
-				var phone = removeDash(tableL.getRows()[i].getData().phone);
-				var rep1 = tableL.getRows()[i].getData().rep1;
-				var rep2 = tableL.getRows()[i].getData().rep2;
-				var rep3 = tableL.getRows()[i].getData().rep3;
-				var rep4 = tableL.getRows()[i].getData().rep4;
-				var varValStr = "";	//¶ 구분자
-				
-				if(phone == ""){
-					
-					alert("수신 목록에 핸드폰 번호가 없는 항목이 있습니다.");
-					return false;
-					
-				}else if(!checkHpNum(phone)){
-					
-					alert("수신 목록에 잘 못된 핸드폰 번호가 있습니다. 핸드폰 번호 : " + phone + " 입니다.");
-					return false;
-					
-				}else{
-					
-					if(typeof(name) != 'undefined' && name != null && name !=""){
-						if(!emojiCheck(name)){//이모지 체크 해주기
-							return false;
-						}
-						
-						//이름 치환변수가 있으면 저장
-						if(nmStatus){
-							//nameList[i] = name.replaceAll(",","§");
-							if(varValStr == ''){
-								
-								varValStr = name.replaceAll(",","§");
-							}else{
-								
-								varValStr = varValStr + "¶" +  name.replaceAll(",","§");
-								
-							}
-						}
-					}else{
-						if(nmStatus){
-							varValStatus = false;
-						}
-					}
-
-					
-					if(phone != '' && phone != null){
-						//연락처 변수 저장하기
-						phoneNum[i] = phone;
-						if(varValStr == ''){
-							varValStr = phone;	
-						}else{
-							varValStr = varValStr + "¶" +  phone;
-						}
-					}else{
-						varValStatus = false;
-					}
-					
-					if(typeof(rep1) != 'undefined' && rep1 != null && rep1 !=""){
-						if(!emojiCheck(rep1)){//이모지 체크 해주기
-							return false;
-						}
-						
-						if(rep1Status){
-							
-							if(varValStr == ''){
-								//rep1List[i] = rep1.replaceAll(",","§");
-								varValStr = rep1.replaceAll(",","§");
-							}else{
-								//rep1List[i] = rep1.replaceAll(",","§");
-								varValStr = varValStr + "¶" +  rep1.replaceAll(",","§");
-							}
-						}
-					}else{
-						if(rep1Status){
-							varValStatus = false;
-						}
-					}
-					
-					if(typeof(rep2) != 'undefined' && rep2 != null && rep2 !=""){
-						if(!emojiCheck(rep2)){//이모지 체크 해주기
-							return false;
-						}
-						
-						if(rep2Status){
-							
-							if(varValStr == ''){
-								varValStr = rep2.replaceAll(",","§");
-							}else{
-								//rep2List[i] = rep2.replaceAll(",","§");
-								varValStr = varValStr + "¶" +  rep2.replaceAll(",","§");
-							}
-						}
-					}else{
-						if(rep2Status){
-							varValStatus = false;
-						}
-					}
-					
-					
-					if(typeof(rep3) != 'undefined' && rep3 != null && rep3 !=""){
-						if(!emojiCheck(rep3)){//이모지 체크 해주기
-							return false;
-						}
-						
-						if(rep3Status){
-							
-							if(varValStr == ''){
-								varValStr = rep3.replaceAll(",","§");
-							}else{
-								//rep3List[i] = rep3.replaceAll(",","§");
-								varValStr = varValStr + "¶" +  rep3.replaceAll(",","§");
-							}
-						}
-					}else{
-						if(rep3Status){
-							varValStatus = false;
-						}
-					}
-					
-					
-					if(typeof(rep4) != 'undefined' && rep4 != null && rep4 !=""){
-						if(!emojiCheck(rep4)){//이모지 체크 해주기
-							return false;
-						}
-						
-						if(rep4Status){
-							
-							if(varValStr == ''){
-								varValStr = rep4.replaceAll(",","§");
-							}else{
-								//rep4List[i] = rep4.replaceAll(",","§");
-								varValStr = varValStr + "¶" +  rep4.replaceAll(",","§");
-							}
-						}
-					}else{
-						if(rep4Status){
-							varValStatus = false;
-						}
-					}
-					
-				}
-				
-				varValList[i] = varValStr;
-				
-			}
-		
-		}else{//치환문자가 없는 경우 휴대폰 번호만 입력
-			
-			for(var i=0; i < selectedData.length; i++){
-				
-				//일괄변환 문자에 콤마(,)가 들어가있으면 배열로 넘길때 문제가 발생하여 특수문자(§)로 치환하여 넘겨주도록 한다.
-				var phone = removeDash(tableL.getRows()[i].getData().phone);
-				
-				if(phone == ""){
-					
-					alert("수신 목록에 핸드폰 번호가 없는 항목이 있습니다.");
-					return false;
-					
-				}else if(!checkHpNum(phone)){
-					
-					alert("수신 목록에 잘 못된 핸드폰 번호가 있습니다. 핸드폰 번호 : " + phone + " 입니다.");
-					return false;
-					
-				}else{
-					
-					phoneNum[i] = phone;
-					
-				}
-				
-			}
-		
-		}
 	
 	}
 	
@@ -1091,164 +1158,248 @@
 	
 	//대체문자 선택 및 내용 체크
 	var subMsgSendYn = "N";
-	
 	if($("#send_fail_check").is(":checked")){
-		
-		subMsgSendYn = "Y";
-		$("#subMsgSendYn").val("Y");
-		
-		//대체문자 체크사항 함수 호출
-		if(!fn_subMsgCheck()){ 
-			
-			/* $("#send_fail_check").prop("checked", false);
-			$(".replace_send_wrap").slideUp(400);
-			// 초기화 버튼 클릭
-			$('#failCheckInit').click();
-			$('.send_top .send_right .phone').css({'top': '0','transition': 'top .4s linear'}); */
-			return false;
-		
-		}
-		
-	}else{
-		
-		subMsgSendYn = "N";
-		$("#subMsgSendYn").val("N");
-		
+		subMsgSendYn = 'Y'
+		$('#callFrom').val($('#callFromList').val()) 
 	}
+	$("#subMsgSendYn").val(subMsgSendYn);
+	if(subMsgSendYn == 'Y'){
+		$("#subMsgTxt").val( $('#smsTxtArea').val());
+	}
+		
 	
-	//수신전화번호 목록
-	$("#callToList").val(phoneNum);
 	
-	//치환변수 데이터 정보 목록(수신번호 포함, tabulator 정보 모두 )
-	$("#varValList").val(varValList);
-
 	//템플릿 내용 입력
 	$("#templateContent").val(tmpContents);
 	
 	//광고포함 여부
-	var adFlag = $("input[name=ad_flag]:checked").val();
-	$("#adFlag").val(adFlag);
+/* 	var adFlag = $("input[name=adFlag]:checked").val();
+	$("#adFlag").val(adFlag); */
 	
-	//치환문자에 대한 데이터 누락 체크
-	if(!varValStatus){
-		
-		alert("특정문구 일괄변환에 대한 일부 데이터가 누락된 부분이 있습니다. 데이터를 확인해 주세요.");
-		return false;
-		
+	
+	if (!validateButtons()) {
+		return;
 	}
+	
+	
 	
 	if(!confirm("친구톡을 발송하시겠습니까?")){
-		
 		return false;
-		
 	}
 	
-	var spamChk = true;
+		
+
+	// 타블레이터 호출
+	var $selectedData = tableL.getData(); // 데이터 가져오기
 	
-	var spmData = new FormData(document.bizForm);
-	$.ajax({
-		type: "POST"
-		, url: "/web/mjon/kakao/friendstalk/selectSpamKakaoFriendsTalkMsgChkAjax.do"
-		, data: spmData
-		, dataType:'json'
-		, async: false
-		, processData: false
-		, contentType: false
-		, cache: false
-		, success: function (returnData, status) {
-			if(status == 'success'){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
-				
-				if("fail" == returnData.result){
-					alert(returnData.message);
-					spamChk = false;
-					return false;
-				}else if("loginFail" == returnData.result){
-					alert(returnData.message);
-					spamChk = false;
-					return false;
-				}else if("spams" == returnData.result){
-					//alert("전송 내용에  스팸문구가 포함되어 있습니다.")
-					$("#spamStatus").val("Y");
-					return false;
-				}else{
-					spamChk = true;
-					return false;
-				}
-				
-			} else if(status== 'fail'){
-				alert(returnData.message);
-				return false;
-			}
+	var data = $('#bizForm');
+	var formDataArray = data.serializeArray();
+	
+	// 배열을 객체로 변환
+	var formData = {};
+	$.each(formDataArray, function(index, field) {
+		formData[field.name] = field.value;
+	});
+
+	// 2. buttonVOList 수동으로 수집
+	var buttonList = [];
+	$('input[name^="buttonVOList"]').each(function() {
+		let nameAttr = $(this).attr('name');
+		let match = nameAttr.match(/buttonVOList\[(\d+)\]\.(\w+)/);
+		if (match) {
+			let index = parseInt(match[1]);
+			let key = match[2];
+			let value = $(this).val();
+		
+			if (!buttonList[index]) buttonList[index] = {};
+			buttonList[index][key] = value;
 		}
-		, error: function (e) {
-			alert("문자 발송에 실패하였습니다.");
-			console.log("ERROR : ", e);
-			return false;
+	});
+
+	// 3. formData에 배열로 추가
+	formData["buttonVOList"] = buttonList;
+	  // 회색 처리 + 입력 불가
+
+	// 4. 기존의 buttonVOList[0].xxx 형태 제거
+	Object.keys(formData).forEach(function(key) {
+		if (/^buttonVOList\[\d+\]\./.test(key)) {
+		    delete formData[key];
 		}
 	});
 	
-	if(spamChk){
-		var data = new FormData(document.bizForm);
-		$.ajax({
-			type: "POST"
-			, url: "/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgSendAjax.do"
-			, data: data
-			, dataType: 'json'
-			, async: true
-			, processData: false
-			, contentType: false
-			, cache: false
-			, success: function (returnData, status) {
-				if(status == 'success'){
-					if("loginFail" == returnData.result){
-						
-						alert(returnData.message);
-						return false;
-						
-					}else  if('fail' == returnData.result){
-						
-						alert(returnData.message);
-						return false;
-						
-					}else if('authFail' == returnData.result){
-						
-						alert(returnData.message);
-						location.reload();
-						
-					} else if(status == 'success'){
-						
-						var kakaoSendCnt = returnData.resultSts;
-						
-						$('.pop_msg_success').css({'display':'block','opacity':'1','left':'50%','top':'50%','transform':'translate(-50%,-50%)'});
-						
-						//예약발송 건의 경우 결과 팝업 문구 변경
-						if(reserYn == 'Y'){
-							$('.pop_msg_success .msg_text').html("예약 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 예약 되었습니다.");
-						}else{
-							$('.pop_msg_success .msg_text').html("발송 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 발송 되었습니다.");
-						}
-						
-						$('.mask').addClass('on');
+	// VO에 정의되어있지 않는 필요없는 값은 제거
+	["img_file_add", "userMoney", "callToList"].forEach(function(key) {
+		  delete formData[key];
+	});
+
+	formData["imgLink"] = $('#imgLinkTemp').val();
+
+	// 빈 값 제거
+	removeEmptyValues(formData);
+	// 선택된 데이터 추가
+	formData["mjonFTSendVOList"] = $selectedData;
+	// JSON 데이터 확인
+	console.log("최종 formData:", JSON.stringify(formData));
+
+
+    // 프로그래스파 시간을 위한 계산
+    var estimtedTime = calculateEstimatedTime(tableL.getRows().length);
+	
+	
+	$.ajax({
+		type: "POST"
+		, url: "/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgSendAjax_advc.do"
+		, data: JSON.stringify(formData)
+		, contentType: 'application/json'
+		, dataType: 'json'
+		, success: function (data) {
+            console.log('data : ', data);
+            
+            var status = data.status;
+            if("OK" == status){
+                var resultSts = data.object.resultSts;
+                var reserYn = data.object.reserYn;
+                var resText = (reserYn === 'Y') ? '예약' : '발송';
+                    
+                $('.pop_msg_success').css({'display':'block','opacity':'1','left':'50%','top':'50%','transform':'translate(-50%,-50%)'});
+                $('.pop_msg_success .msg_text').html(resText+" 성공 : <strong>"+ resultSts + "</strong>건의<br>친구톡이 " + resText + " 되었습니다.");
+                
+            }else if("UNAUTHORIZED" == status){
+                alert(data.message);
+                location.reload();
+            }else{
+                alert(data.message);
+                return false;
+            }
+			
+			/* 
+			if(status == 'success'){
+				if("loginFail" == returnData.result){
+					
+					alert(returnData.message);
+					return false;
+					
+				}else  if('fail' == returnData.result){
+					
+					alert(returnData.message);
+					return false;
+					
+				}else if('authFail' == returnData.result){
+					
+					alert(returnData.message);
+					location.reload();
+					
+				} else if(status == 'success'){
+					
+					var kakaoSendCnt = returnData.resultSts;
+					
+					$('.pop_msg_success').css({'display':'block','opacity':'1','left':'50%','top':'50%','transform':'translate(-50%,-50%)'});
+					
+					//예약발송 건의 경우 결과 팝업 문구 변경
+					if(reserYn == 'Y'){
+						$('.pop_msg_success .msg_text').html("예약 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 예약 되었습니다.");
+					}else{
+						$('.pop_msg_success .msg_text').html("발송 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 발송 되었습니다.");
 					}
+					
+					$('.mask').addClass('on');
 				}
-			}
-			,beforeSend : function(xmlHttpRequest) {
-				//로딩창 show
-				$('.loading_layer').addClass('active');
-			}
-			,complete : function(xhr, textStatus) {
-				//로딩창 hide
-				$('.loading_layer').removeClass('active');
-			}
-			,error: function (e) {
-				console.log("ERROR : ", e);
-				alert("카카오 친구톡 전송에 실패하였습니다.");
-			}
-		}); 
-	}
+			} */
+		}
+		,beforeSend : function(xmlHttpRequest) {
+			//로딩창 show
+// 			$('.loading_layer').addClass('active');
+            progressStart(estimtedTime);
+		}
+		,complete : function(xhr, textStatus) {
+			//로딩창 hide
+// 			$('.loading_layer').removeClass('active');
+            progressComplete();
+		}
+		,error: function (e) {
+			console.log("ERROR : ", e);
+			alert("카카오 친구톡 전송에 실패하였습니다.");
+		}
+	}); 
 	
 }
 
+//프로그레스바 
+var start, change;
+var progressInterval = null; // 전역 변수로 타이머 ID 관리
+function progressStart(time) {
+  // 기존 타이머 정지 및 초기화
+  if (progressInterval !== null) {
+      clearInterval(progressInterval); // 이전 타이머 정지
+      progressInterval = null; // 타이머 ID 초기화
+  }
+  resetProgressBar(); // 프로그레스바 초기화
+
+  // 프로그레스바 보이기
+  $(".progress_bar_wrap").css("display", "flex");
+
+  // 프로그레스바 요소 가져오기
+  var timeText = document.querySelector(".time_text");
+  var bar = document.querySelector(".change_bar");
+
+  // 초기 상태 설정
+  var width = 1;
+  var totalTime = time * 1000; // 총 실행 시간 (밀리초)
+  var cmpWid = totalTime / 100; // width 증가 간격 (밀리초)
+
+  // 새 타이머 시작
+  progressInterval = setInterval(changeWidth, cmpWid);
+
+  function changeWidth() {
+      if (width >= 100) {
+          // 프로그레스바 100% 도달
+          clearInterval(progressInterval); // 타이머 종료
+          progressInterval = null; // 타이머 ID 초기화
+
+          timeText.innerHTML = "100%";
+
+          setTimeout(function () {
+              // 100% 표시 후 "잠시만 기다려주세요" 변경
+              timeText.innerHTML = "잠시만 기다려주세요...";
+              $(".time_text").addClass("animation");
+          }, 1000);
+      } else {
+          // 프로그레스바 진행
+          width++;
+          bar.style.width = width + "%";
+          timeText.innerHTML = width + "%";
+      }
+  }
+}
+
+//선택된 데이터의 길이에 따라 예상 시간 계산 함수
+function calculateEstimatedTime(selectedCount) {
+  //기준값
+  // const processTimePerBatch = 130; // 130초
+  
+  
+  // 30만건 기준 10분으로 기준을 잡아서 
+  // 시간계산함
+  const processTimePerBatch = 600;
+  const batchSize = 300000;
+  
+  // 1건당 처리 시간
+  const timePerRecord = processTimePerBatch / batchSize;
+
+  // 예상 시간 계산
+  const estimatedTimeInSeconds = selectedCount * timePerRecord;
+
+  return estimatedTimeInSeconds.toFixed(2);
+}
+
+//빈 값이나 null 값을 제거하는 함수
+function removeEmptyValues(obj) {
+  Object.keys(obj).forEach(function(key) {
+      if (obj[key] === null || obj[key] === '') {
+          delete obj[key];
+      }
+  });
+}
 
 /**
  * @description 대체문자 오류체크 funciton
@@ -1284,74 +1435,30 @@
 	$('#errorChk').val(val);
 }
 
-/*
- *대체문자 필수 항목 체크 
- *
- **/
- 
-function fn_subMsgCheck(){
+//링크 유효성 검증 함수
+function validateLinks() {
 
-	if($('#callFromList').val() === ''){
-		
-		if(confirm('대체문자 전송을 위한 발신번호가 등록되지 않았습니다. \n대체문자 발신번호를 지금 등록하시겠습니까?')){
-			window.location="<c:out value='/web/user/sendNumberManage.do' />";
-		}
-		$("#send_fail_check").prop("checked", false);
-		
-		return false;
-		
-	}else{
+	console.log('검사 대상 수:', $('.kakaoBtnList').length);
 	
-		//발신번호 입력 처리
-		$("#callFrom").val(removeDash($('#callFromList').val()));
-		
-	}
-	
-	if($("#smsTxtArea").val() === ''){
-		
-		alert("대체문자 내용을 입력해 주세요.");
-		return false;
-		
-	}else{//대체문자에 치환문자 여부 체크
-		
-		var smsTxtArea = $("#smsTxtArea").val();
-		var replStatus = false;
-	
-		if(smsTxtArea.indexOf("\#{이름}")  > -1){
-			replStatus = true;
+	let isValid = true;
+
+	$('.kakaoBtnList').each(function () {
+		const val = $(this).val().trim();
+
+		console.log('val : ', val);
+		if (val !== '' && !val.startsWith('http://') && !val.startsWith('https://')) {
+			alert(`"${val}" 은(는) http:// 또는 https:// 로 시작해야 합니다.`);
+			$(this).focus();
+			isValid = false;
+			return false; // each 루프 중단
 		}
-		
-		if(smsTxtArea.indexOf("\#{1}")  > -1){
-			replStatus = true;
-		}
-		
-		if(smsTxtArea.indexOf("\#{2}")  > -1){
-			replStatus = true;
-		}
-		
-		if(smsTxtArea.indexOf("\#{3}")  > -1){
-			replStatus = true;
-		}
-		
-		if(smsTxtArea.indexOf("\#{4}")  > -1){
-			replStatus = true;
-		}
-		
-		if(replStatus){
-			$("#subMsgTxtReplYn").val("Y");
-		}else{
-			$("#subMsgTxtReplYn").val("N");
-		}
-		
-		$("#subMsgTxt").val(smsTxtArea);
-		
-	}
-	
-	
-	
-	return true;
-	
+	});
+
+	return isValid;
 }
+
+
+
 
 //문자 바이트수 계산하기 함수
 function thisFnByteString(contents){
@@ -1363,69 +1470,80 @@
 
 	$('#msgLeng').html("");
 	$('#limitLeng').html("");
+	
+	/* if ($("#ad_Y").is(":checked")) {
+		contents = "(광고)"+contents+"\n무료거부 0808800858"
+	} */
+	contents = adYChkAndMakeContents(contents);
+	console.log(contents);
 	var conLeng = conByteLeng(contents); // 내용 문자 입력 바이트 수 계산하기
 	
 		
 	$('#msgLeng').text(conLeng);
 	
 	//문자 길이 변수에 저장해주기
-	$('#smsLen').val(conLeng);
+// 	$('#smsLen').val(conLeng);
+	
+	var imgType = $('#sendFailImgSrc').attr('src');
 	
 	
-	if(conLeng > 90){
+	if(imgType != null && "" != imgType){
+
+		setSendFailTemplate('P')
+	}
+	else if(conLeng > 90){
 		
-		$('#msgLeng').html(conLeng + " / ");
 		$('#limitLeng').html("2000");
 		$('.msg_com').html("장문");
 		$('#msgType').val("6"); // 메세지 타입 설정
 		
 		$('.msg_com').removeClass("msg_short"); //단문 클래스 삭제하고
 		$('.put_left').removeClass("short"); //내용 입력 박스 클래스 삭제
+		$('.put_left').removeClass("photo"); //내용 입력 박스 클래스 삭제
 		$('.msg_com').addClass("msg_long"); // 장문 클래스 삽입
 		$("#subMsgType").val("LMS");
 		
 	}else{
 		
-		$('#msgLeng').html(conLeng + " / ");
 		$('#limitLeng').html("90");
 		$('.msg_com').html("단문");
 		$('#msgType').val("4"); // 메세지 타입 설정
+		
 		$('.msg_com').removeClass("msg_long"); //단문 클래스 삭제하고
 		$('.put_left').removeClass("long"); //내용 입력 박스 클래스 삭제
+		$('.put_left').removeClass("photo"); //내용 입력 박스 클래스 삭제
 		$('.msg_com').addClass("msg_short"); // 장문 클래스 삽입
 		$("#subMsgType").val("SMS");
 		
 	}
+	
+	$('#msgLeng').html(conLeng + " / ");
+	totalFtPriceSum(tableL.getRows().length);
 		
 }
 
 function msgSuccessClose(obj){
-	$(obj).closest('.pop_msg_success').attr('style','');
-	location.reload(true);
-	$('html').scrollTop(0);
+// 	$(obj).closest('.pop_msg_success').attr('style','');
+// 	location.reload(true);
+// 	$('html').scrollTop(0);
+
+	location.href = '/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgDataView.do';
 }
 
 function getFtImageType(imgChk){
 	
 	if(imgChk == 'I' || imgChk == 'W'){
 		
-		var imgTitle = $("#imgTitle").val();
 		var imgLink = $("#imgLink").val();
 		var imgFileName = $("#imageFileName").val();
 		
-		if(imgTitle == ''){
-			
-			alert("친구톡 이미지 제목정보를 입력해 주세요.");
-			return false;
-			
-		}
-		
+		/* 
 		if(imgLink == ''){
 			
 			alert("친구톡 이미지 선택시 이동할 링크 주소를 입력해 주세요.");
 			return false;
 			
-		}
+		} */
 		
 		if(imgFileName == ''){
 			
@@ -1516,154 +1634,6 @@
 		
 		alert("받는사람 주소를 한 건 이상 입력해주세요.");
 		return false;
-	
-	}else{
-		
-		//치환문구 변환 
-		var txtReplYn = $("#txtReplYn").val();
-		
-		if(txtReplYn == 'Y'){
-			
-			var name = tableL.getRows()[0].getData().name;
-			var phone = removeDash(tableL.getRows()[0].getData().phone);
-			var rep1 = tableL.getRows()[0].getData().rep1;
-			var rep2 = tableL.getRows()[0].getData().rep2;
-			var rep3 = tableL.getRows()[0].getData().rep3;
-			var rep4 = tableL.getRows()[0].getData().rep4;
-			
-			var varValList = [];		//치환문자 연결시킬 변수 셋팅
-			
-			
-			var nmStatus = false;
-			var rep1Status = false;
-			var rep2Status = false;
-			var rep3Status = false;
-			var rep4Status = false;
-			
-			var varValStr = "";
-			var varValStatus = true;
-
-			
-			if(tmpContents.indexOf("\#{이름}")  > -1){
-				nmStatus = true;
-			}
-			
-			if(tmpContents.indexOf("\#{1}")  > -1){
-				rep1Status = true;
-			}
-			
-			if(tmpContents.indexOf("\#{2}")  > -1){
-				rep2Status = true;
-			}
-			
-			if(tmpContents.indexOf("\#{3}")  > -1){
-				rep3Status = true;
-			}
-			
-			if(tmpContents.indexOf("\#{4}")  > -1){
-				rep4Status = true;
-			}
-			
-	
-			if(nmStatus && (typeof(name) != 'undefined' && name != null && name !="")){
-				
-				if(varValStr == ''){
-					varValStr = name.replaceAll(",","§");
-				}else{
-					varValStr = varValStr + "¶" +  name.replaceAll(",","§");
-				}
-				
-			}else{
-				
-				if(nmStatus){
-					varValStatus = false;
-				}
-				
-			}
-			
-			if(varValStr == ''){
-				varValStr = phone;
-			}else{
-				varValStr = varValStr + "¶" +  phone;
-			}
-
-			if(rep1Status && (typeof(rep1) != 'undefined' && rep1 != null && rep1 !="")){
-				
-				if(varValStr == ''){
-					varValStr = rep1.replaceAll(",","§");
-				}else{
-					varValStr = varValStr + "¶" +  rep1.replaceAll(",","§");
-				}
-				
-			}else{
-				
-				if(rep1Status){
-					varValStatus = false;
-				}
-				
-			}
-			
-			
-			if(rep2Status && (typeof(rep2) != 'undefined' && rep2 != null && rep2 !="")){
-			
-				if(varValStr == ''){
-					varValStr = rep2.replaceAll(",","§");
-				}else{
-					varValStr = varValStr + "¶" +  rep2.replaceAll(",","§");
-				}
-				
-			}else{
-				
-				if(rep2Status){
-					varValStatus = false;
-				}
-				
-			}
-			
-			if(rep3Status && (typeof(rep3) != 'undefined' && rep3 != null && rep3 !="")){
-				
-				if(varValStr == ''){
-					varValStr = rep3.replaceAll(",","§");
-				}else{
-					varValStr = varValStr + "¶" +  rep3.replaceAll(",","§");
-				}
-				
-			}else{
-				
-				if(rep3Status){
-					varValStatus = false;
-				}
-				
-			}
-			
-			if(rep4Status && (typeof(rep4) != 'undefined' && rep4 != null && rep4 !="")){
-				
-				if(varValStr == ''){
-					varValStr = rep4.replaceAll(",","§");
-				}else{
-					varValStr = varValStr + "¶" +  rep4.replaceAll(",","§");
-				}
-			
-			}else{
-				
-				if(rep4Status){
-					varValStatus = false;
-				}
-				
-			}
-			
-			if(!varValStatus){
-				
-				alert("특정문구 일괄변환에 대한 일부 데이터가 누락된 부분이 있습니다. 데이터를 확인해 주세요.");
-				return false;
-				
-			}
-			
-			varValList[0] = varValStr;
-			
-			$("#varValList").val(varValList);
-		}
-		
 	}
 
 	form.method = "post"; 
@@ -1676,14 +1646,15 @@
 
 function msgResultLink(){
 	var reserYn = $("input[name=reserYn]:checked").val(); // 예약 발송 여부 확인
-	if(reserYn == 'Y'){
-		
-		location.href="/web/mjon/reservmsg/selectReservKaKaoView.do";
-		
-	}else{
-		location.href="/web/kakao/sent/selectKakaoSentView.do";
-	}
+	location.href="/web/kakao/sent/selectKakaoSentView.do";
 }
+
+function updateButtons(){
+	
+}
+
+
+
 
 </script>
 
@@ -1695,6 +1666,19 @@
 	</div>
 </div>
 
+
+<div class="progress_bar_wrap">
+    <div class="progress_box">
+        <p class="time_text">0%</p>
+        <div class="bar">
+            <span class="change_bar"></span>
+        </div>
+    </div>
+    <div class="btn_wrap">
+    </div>
+
+</div>
+
 <div class="inner">
 	<div class="send_top">
 		<!-- tab button -->
@@ -1702,8 +1686,9 @@
 		<!-- // tab button -->
 		<div class="top_content kakaotalksend_cont current pay_tab_wrap">
 			<div class="heading">
-				<h2>친구톡 전송</h2>
-				<button type="button" class="button info">친구톡 사용방법</button>
+				<h2>친구톡 전송</h2><!-- KakaoFriendsTalkMsgDataView -->
+<!-- 				<button type="button" class="button info">친구톡 사용방법</button> -->
+				<button type="button" class="button info" onclick="infoListPop('friendTalkSend','792','340');">사용안내</button>
 			</div>
 			
 			<!-- 유저 보유잔액 -->
@@ -1711,8 +1696,12 @@
 			
 			<!-- 각 금액 단가  -->
 			<input type="hidden" id="kakaoFtPrice" value="<c:out value='${sendPrice.kakaoFtPrice}' />"> <!-- 친구톡 단가  -->
+			<input type="hidden" id="kakaoFtImgPrice" value="<c:out value='${sendPrice.kakaoFtImgPrice}' />"> <!-- 친구톡 이미지 단가  -->
+			<input type="hidden" id="kakaoFtWideImgPrice" value="<c:out value='${sendPrice.kakaoFtWideImgPrice}' />"> <!-- 친구톡 와이드이미지 단가  -->
 			<input type="hidden" id="longPrice" value="<c:out value='${sendPrice.longPrice}' />"> <!-- mms 단가  -->
 			<input type="hidden" id="shortPrice" value="<c:out value='${sendPrice.shortPrice}' />"> <!-- sms 단가  -->
+			<input type="hidden" id="picturePrice" value="<c:out value='${sendPrice.picturePrice}' />"> <!-- sms 단가  -->
+			<input type="hidden" id="imgLinkTemp" name="imgLinkTemp" value="<c:out value='${resultTemplateVO.imgLink}'/>"/>
 			
 			<form id="bizForm" name="bizForm" method="multipart/form-data">
 				<input type="hidden" id="menuTopTab" name="menuTopTab" value="tabFriend">
@@ -1722,8 +1711,11 @@
 				<input type="hidden" id="templateImageUrl" name="templateImageUrl" value="<c:out value='${resultTemplateVO.templateImageUrl}'/>">
 				<input type="hidden" id="tmpFriendId" name="friendId" value="<c:out value='${resultTemplateVO.friendId}'/>">
 				<input type="hidden" id="templateName" name="templateName" value=""/>
-				<input type="hidden" id="adFlag" name="adFlag" value="N"/>
+				<input type="hidden" id="adFlag" name="adFlag" value="<c:out value='${resultTemplateVO.adFlag}' default='Y'/>">
 				<input type="hidden" id="templateContent" name="templateContent" value=""/>
+				
+				<!-- 이미지 있을때 대체문자 발송 참고 이미지 -->
+				<input type="hidden" id="atchFileId" name="atchFileId" value="<c:out value='${resultTemplateVO.atchFileId}'/>"/>
 				
 				<!-- 예약발송 여부 -->
 				<input type="hidden" id="reserveYn" name="reserveYn" value="N"/>
@@ -1753,9 +1745,8 @@
 				<input type="hidden" id="callFrom" name="callFrom" value=""/>
 				
 				<!-- 친구톡 발송 단가 정보 -->
-				<input type="hidden" id="kakaoFtPrice" name="kakaoFtPrice" value="<c:out value='${sendPrice.kakaoFtPrice}' />"/>
-				<input type="hidden" id="eachPrice" name="eachPrice" value="0"/>
-				<input type="hidden" id="totPrice" name="totalPrice" value="0"/>
+<!-- 				<input type="hidden" id="eachPrice" name="eachPrice" value="0"/> -->
+<!-- 				<input type="hidden" id="totPrice" name="totalPrice" value="0"/> -->
 				
 				<!-- 친구톡 발송 json 파일 필요 유무 -->
 				<input type="hidden" id="bizJsonYn" name="bizJsonYn" value="N"/>
@@ -1786,63 +1777,76 @@
 								<tr>
 	                                <th>템플릿명</th>
 	                                <td>
-	                                    <input type="text" class="template_name" id="inputTemplateName" name="inputTemplateName" value="<c:out value='${resultTemplateVO.templateName}'/>" placeholder="템플릿명을 입력해주세요. (최대 50자)">
-	                                    <p class="template_name_cf">최대 50자, 템플릿 관리용</p>
+										<div class="template_name_wrap">
+											<input type="text" id="inputTemplateName" name="inputTemplateName" value="<c:out value='${resultTemplateVO.templateName}'/>" placeholder="최대 50자, 템플릿 관리용">
+											
+											
+												<button
+													type="button"
+													class="btnType btnType8"
+													onclick="${not empty loginVO.id 
+														? "window.open('/web/mjon/kakao/template/selectKakaoFriendsTemplateListPopupAjax.do','_blank','width=930, height=780, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')" 
+														: "alert('로그인 후 이용이 가능합니다.'); return false;"}">
+													템플릿 불러오기
+												</button>
+											
+<!-- 											<button type="button" class="btnType btnType8" onclick="window.open('/publish/popup_friendtalk_template_choice.html','_blank','width=930, height=780, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">템플릿 불러오기</button> -->
+										</div>
 	                                </td>
 	                            </tr>
 								<tr>
 									<th>이미지 첨부</th>
 									<td>
-										<div class="img_sort_wrap">
-											<input type="radio" name="img_file_add" id="img_file_0" value="" <c:if test="${resultTemplateVO.imageType eq ''}">checked</c:if> > <label for="img_file_0">첨부안함</label>
-											<input type="radio" name="img_file_add" id="img_file_1" value="I" <c:if test="${resultTemplateVO.imageType eq 'I'}">checked</c:if> > <label for="img_file_1">이미지 첨부</label>
-											<input type="radio" name="img_file_add" id="img_file_2" value="W" <c:if test="${resultTemplateVO.imageType eq 'W'}">checked</c:if> > <label for="img_file_2">와이드 이미지 첨부</label>
-											<div class="img_add_info_wrap basic_img_add_wrap">
-												<p class="info_title_text"><span class="c_e40000">*</span> 이미지 첨부 안내</p>
-												<ul class="info_text">
-												<li>- 권장사이즈 : 720px * 720px</li>
+										<input type="radio" name="img_file_add" id="img_file_0" value="" checked="checked">
+										<label for="img_file_0">첨부 안함</label>
+										<input type="radio" name="img_file_add" id="img_file_1" value="I">
+										<label for="img_file_1">이미지 첨부</label>
+										<input type="radio" name="img_file_add" id="img_file_2" value="W">
+										<label for="img_file_2">와이드 이미지 첨부</label>
+										
+										<div class="img_file_add_wrap basic_img_add_wrap">
+											<p class="info_title_text"><span class="c_e40000">*</span> 이미지 첨부 안내</p>
+											<ul class="info_text">
+												<li>- 권장사이즈 : 800px * 400px</li>
 												<li>- 제한사이즈 : 가로 500px 미만, 가로:세로 비율이 2:1 미만 또는 3:4 초과시 업로드 불가</li>
-												<li>- 파일형식 : jpg, png (최대 500kb)</li>
-												<li>- 이미지 첨부시 메시지 내용은 최대 400자까지 입력할 수 있습니다.</li>
-												</ul>
-											</div>
-											<div class="img_add_info_wrap wide_img_add_wrap">
-												<p class="info_title_text"><span class="c_e40000">*</span> 와이드 이미지 첨부 안내</p>
-												<ul class="info_text">
-												<li>- 권장사이즈 : 800px * 600px</li>
-												<!-- <li>- 제한사이즈 : 가로 500px 미만, 가로:세로 비율이 2:1 미만 또는 3:4 초과시 업로드 불가</li> -->
-												<li>- 파일형식 : jpg, png (최대 2mb)</li>
-												<li>- 와이드 이미지 첨부시 메시지 내용은 최대 76자, 버튼 1개까지 입력할 수 있습니다.</li>
-												</ul>
-											</div>
-										</div>
-										<div class="img_file_add_wrap">
-											<ul class="img_file_info_wrap">
-												<li>
-													<p>이미지 제목</p>
-													<input type="text" id="imgTitle" name="imgTitle" value="<c:out value='${resultTemplateVO.imgTitle}'/>" class="img_url" maxLength="50">
-												</li>
-												<li>
-													<p>이미지 클릭시 이동할 URL</p>
-													<input type="text" id="imgLink" name="imgLink" value="<c:out value='${resultTemplateVO.imgLink}'/>" placeholder="http://" class="img_url" maxLength="1000">
-												</li>
-												<li>
-													<p class="file_name"  id="imgNm" name="imgNm">첨부파일 이미지</p>
-													<button type="button" class="btn_del btnImgDel" onclick="buttonTypeDel(this);">
-														<img src="/publish/images/btn_delete.png" id="imgDel" alt="이미지 삭제" onclick="fnImageDel();">
-													</button>
-												</li>
+												<li>- 파일형식 : jpg, png (최대 5MB)</li>
+												<li>- 이미지 첨부 시 메시지 내용은 최대 400자, 버튼 5개까지 입력할 수 있습니다.</li>
 											</ul>
-											<button type="button" class="btnType btnType8 btn_img_upload" title="이미지 불러오기" onclick="javascript:upImgClick(); return false;">이미지 불러오기</button>
-											<input type="file" id="imgFile" accept=".jpg, .jpeg, .png" onchange="imgResizeInfo(event); return false;" style="display:none"/>
 										</div>
+										<div class="img_file_add_wrap wide_img_add_wrap">
+											<p class="info_title_text"><span class="c_e40000">*</span> 와이드 이미지 첨부 안내</p>
+											<ul class="info_text">
+												<li>- 권장사이즈 : 800px * 600px</li>
+												<li>- 파일형식 : jpg, png (최대 5MB)</li>
+												<li>- 와이드 이미지 첨부시 메시지 내용은 최대 76자, 버튼 1개까지 입력할 수 있습니다.</li>
+											</ul>
+											
+											
+										</div>
+										<ul class="img_file_info_wrap">
+											<li>
+												<button type="button" class="btnType btnType8 btn_img_upload" onclick="javascript:upImgClick(); return false;">이미지 불러오기</button>
+												<input type="file" id="imgFile" accept=".jpg, .jpeg, .png" onchange="imgResizeInfo(event); return false;" style="display:none"/> 
+												<div class="img_file_wrap">
+													<p id="imgTitle"  class="file_name file_none">이미지 없음</p>
+													<button type="button" class="btn_del" onclick="buttonTypeDel(this);">
+														<img src="/publish/images/btn_delete.png" alt="이미지 삭제" onclick="fnImageDel();">
+													</button>
+												</div>
+											</li>
+											<li>
+												<p>이미지 클릭시 이동할 URL</p> 
+												<input type="text" id="imgLink" name="imgLink" value="<c:out value='${resultTemplateVO.imgLink}'/>"  placeholder="(선택사항) https://" class="img_url">
+											</li>
+										</ul>
+											
 									</td>
 								</tr>
 								<tr>
 									<th>광고포함 여부</th>
 									<td>
-										<input type="radio" class="inputAdFlag" name="ad_flag" id="ad_Y" value="Y" <c:if test="${resultTemplateVO.adFlag eq 'Y'}">checked</c:if> ><label for="ad_Y">광고성 정보 포함</label>
-										<%-- <input type="radio" class="inputAdFlag" name="ad_flag" id="ad_N" value="N" <c:if test="${resultTemplateVO.adFlag eq 'N'}">checked</c:if> ><label for="ad_N">포함 안함</label> --%>
+										<input type="checkbox" id="ad_Y" style="margin: -4px 5px 0 0;" checked="checked" value="Y">
+										<label for="ad_Y">광고성 정보 포함</label>
 									</td>
 								</tr>
 								<tr>
@@ -1850,14 +1854,14 @@
 									<td class="kakao_template_text">
 										<div class="put_left short">
 											<div class="put_text_wrap">
-												<div class="ad_txt">
-													<p><span id="adFlagFront">(광고)</span> <span id="yellowIdCon" >채널ID</span></p>
+												<div class="ad_txt" id="adTxt">
+													<p>(광고) 채널ID</p>
 												</div>
 												<label for="inputTemplateContent" class="label"></label>
 												<textarea id="inputTemplateContent" name="inputTemplateContent" class="put_text" placeholder="내용을 입력해주세요."><c:out value="${resultTemplateVO.templateContent}"/></textarea>
 												<div class="text_length">
-													<div>
-														<p class="adFlagAfter">수신거부 : 홈 > 채널차단</p>
+													<div class="sub_ad_text">
+														<p>수신거부 : 홈 > 채널차단</p>
 													</div>
 													<div>
 														<p><span class="fwMd nowChar">0 /</span><span class="c_002c9a fwMd totChar">1000</span>자</p>
@@ -1960,13 +1964,13 @@
 															<p>주소록, 엑셀에 입력된 내용을 이용해 수신자마다 다른 내용의 메시지를 발송하는 기능</p>
 														</div>
 														<div class="convers_middle">
-															<a href="javascript:void(0)" class="changeWord" value="\#{이름}"><c:out value="\#{이름}"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="[*이름*]"><c:out value="[*이름*]"/></a>
 														</div>
 														<div class="convers_bottom">
-															<a href="javascript:void(0)" class="changeWord" value="\#{1}"><c:out value="\#{1}"/></a>
-															<a href="javascript:void(0)" class="changeWord" value="\#{2}"><c:out value="\#{2}"/></a>
-															<a href="javascript:void(0)" class="changeWord" value="\#{3}"><c:out value="\#{3}"/></a>
-															<a href="javascript:void(0)" class="changeWord" value="\#{4}"><c:out value="\#{4}"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="[*1*]"><c:out value="[*1*]"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="[*2*]"><c:out value="[*2*]"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="[*3*]"><c:out value="[*3*]"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="[*4*]"><c:out value="[*4*]"/></a>
 														</div>
 													</div>
 													<button type="button" class="btn_close" onclick="miniPopup(this)">닫기</button>
@@ -1980,25 +1984,16 @@
 												</div>
 											</div>
 											<div class="btn_popup_wrap template_call_wrap">
-												<c:choose>
-													<c:when test="${not empty loginVO.id}">
-														<button type="button" class="btnType btnType8" onclick="window.open('/web/mjon/kakao/template/selectKakaoFriendsTemplateListPopupAjax.do','_blank','width=930, height=780, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">템플릿 불러오기</button>
-													</c:when>
-													<c:otherwise>
-														<button type="button" class="btnType btnType8" onclick="javascript:fnEmptyId();">템플릿 불러오기</button>
-													</c:otherwise>
-												</c:choose>
 											</div>
 											<div class="send_btnWrap">
-												<c:choose>
-													<c:when test="${not empty loginVO.id}">
-														<button type="button" class="btnType btn_text_save" onclick="javascript:myTemplateSave();">내용 저장하기</button>
-													</c:when>
-													<c:otherwise>
-														<button type="button" class="btnType btn_text_save" onclick="javascript:fnEmptyId();">내용 저장하기</button>
-													</c:otherwise>
-												</c:choose>
-												
+												<button
+													type="button"
+													class="btnType btnType9 btn_text_save"
+													onclick="${not empty loginVO.id 
+														? 'myTemplateSave()' 
+														: 'alert(\'로그인 후 이용이 가능합니다.\'); return false;'}">
+													템플릿 저장
+												</button>
 												<button type="button" class="btnType btnType9" onclick="javascript:fnContentsReset();">초기화</button>
 											</div>
 										</div>
@@ -2009,16 +2004,13 @@
 									<td class="template_button">
 										<select name="tmpBtnSelect" id="tmpBtnSelect" class="select_gray_type template_button_select">
 											<option value="button_type_0">버튼타입 선택</option>
-											<option value="button_type_1">배송조회</option>
+<!-- 											<option value="button_type_1">배송조회</option> -->
 											<option value="button_type_2">웹링크</option>
 											<option value="button_type_3">앱링크</option>
-											<option value="button_type_5">메시지전달</option>
+<!-- 											<option value="button_type_5">메시지전달</option> -->
 										</select>
 										<button type="button" class="btnType btnType6" onclick="friendTemplateButtonAdd();">추가</button>
-										<p class="cf_text fwRg">
-											<span class="c_e40000 fwBold">*</span> 버튼 타입중 <span class="c_e40000">봇키워드, 메시지전달</span>은 <span class="c_222">카카오톡 채널 관리자센터(https://center-pf.kakao.com)</span>에서 설정을 직접 한 후 이용하셔야 동작합니다. (최대 5개까지 등록가능)
-										</p>
-										<div class="button_add_wrap">
+										<div class="button_add_wrap" id="buttonAddWrap">
 	                                    	<c:forEach var="buttonList" items="${resultTemplateVO.buttonVOList}" varStatus="status">
 	                                    	
 	                                    		<c:if test="${buttonList.linkType eq 'DS'}">
@@ -2153,7 +2145,7 @@
 										<div class="clearfix receipt_num">
 											<div class="receipt_num_top">
 												<label for="callTo" class="label">받는 번호입력</label>
-												<input type="text" placeholder="번호를 입력하세요" id="callTo" name="callTo" onfocus="this.placeholder=''" onblur="this.placeholder='번호를 입력하세요'" style="width:340px;">
+												<input type="text" value="" placeholder="번호를 입력하세요" id="callTo" name="callTo" onfocus="this.placeholder=''" onblur="this.placeholder='번호를 입력하세요'" style="width:340px;">
 												<button type="button" class="btnType btnType6 addCallToF">번호추가</button>
 												<span><span class="vMiddle">*</span> 중복번호는 한번만 발송됩니다.</span>
 											</div>
@@ -2183,7 +2175,7 @@
 														<button type="button" data-tooltip="popup06" class="btnType btnType7 popupAddr">주소록 불러오기</button>
 													</div>
 													<div class="btn_popup_wrap">
-														<button type="button" data-tooltip="popup02" class="btnType btnType7">엑셀 불러오기</button>
+														<button type="button" data-tooltip="popup07" class="btnType btnType7">엑셀 불러오기</button>
 													</div>
 													<div class="btn_popup_wrap">
 														<button type="button" data-tooltip="popup03" class="btnType btnType7 tab1">최근 전송내역</button>
@@ -2191,13 +2183,13 @@
 													<div class="btn_popup_wrap">
 														<button type="button" data-tooltip="popup03" class="btnType btnType7 tab2">자주보내는 번호</button>
 													</div>
-													<div class="btn_popup_wrap check_validity_wrap">
+													<!-- <div class="btn_popup_wrap check_validity_wrap">
 														<button type="button" class="btnType btnType7 check_validity">오류 검사<i class="qmMark error_qm"></i></button>
 														<div class="error_hover_cont send_hover_cont">
 															<p>휴대폰 번호 입력 시 해당 휴대폰 번호에 대한 형식이 어긋나거나 휴대폰 번호에 오류가 있는지 등을 검사하는 기능</p>
 															<span>(예시) 010-1234-0001(O) / 010-12345-0001(X)</span>
 														</div>
-													</div>
+													</div> -->
 												</div>
 											</div>
 											<div class="list_bottom clearfix">
@@ -2280,11 +2272,26 @@
 															<div class="clearfix">
 																<div class="put_left short">
 																	<!-- 업로드한 이미지의 썸네일 영역 -->
-																	<ul class="thumb_wrap liOnImg ui-sortable"></ul>
-																	<!-- //업로드한 이미지의 썸네일 영역 -->
+																	<ul class="thumb_wrap liOnImg ui-sortable">
+																		<li class="delLi" id="sendFailImg" style="display: none">
+																			<div>
+																				<img src="" id="sendFailImgSrc" class="thumb_img edit_y" alt="thumb1">
+																			</div>
+<!-- 																			<button type="button" class="file_close"><img src="/publish/images/content/thumb_del.png" alt="첨부파일 삭제"></button> -->
+<!-- 																			<button type="button" class="file_close_on"><img src="/publish/images/content/thumb_del_on.png" alt="첨부파일 삭제"></button> -->
+																		</li>
+																		
+																	</ul>
 																	<label for="smsTxtArea" class="label"></label>
-																	<textarea id="smsTxtArea" name="smsTxtArea" class="put_text"></textarea>
+																	<!-- //업로드한 이미지의 썸네일 영역 -->
+																	<div class="ad_txt" style="">
+																		<p>(광고)</p>
+																	</div>
+																	<textarea id="smsTxtArea" class="put_text"></textarea>
 																	<div class="text_length">
+																		<div class="deny_txt sub_ad_text">
+																			<p>무료거부 0808800858</p>
+																		</div>
 																		<div name="afterDeny">
 																			<p>
 																				<span class="fwMd" id="msgLeng">0 /</span>
@@ -2296,7 +2303,7 @@
 																</div>
 																<div class="put_right">
 																	<button type="button" class="btnType btnType9" id="failCheckInit">초기화</button>
-																	<button type="button" class="btnType btnType7" onclick="javascript:fn_errorChk(); return false;">오류검사<i class="qmMark"></i></button>
+<!-- 																	<button type="button" class="btnType btnType7" onclick="javascript:fn_errorChk(); return false;">오류검사<i class="qmMark"></i></button> -->
 																</div>
 															</div>
 														</td>
@@ -2310,7 +2317,7 @@
 						</table>
 					</div>
 					<div class="send_right">
-						<div class="phone" style="top: 201px;">
+						<div class="phone">
 							<div class="phoneIn">
 								<p class="prev_p">
 									<img src="/publish/images/content/kakao_prev_icon.png" alt="">
@@ -2318,9 +2325,9 @@
 								</p>
 								<!-- 텍스트 미리보기 -->
 								<div class="text_preview">
-									<p class="friend_talk_title">
-										<span class="adFlag">(광고)</span>
-										<span id="spnYellowidAt">채널ID</span>
+									<p class="friend_talk_title" id="talkTitle">
+<!-- 										<span>(광고)</span> 채널ID -->
+										<span>(광고)</span>
 									</p>
 									<div class="allimtalk_content">
 										<div class="kakao_image"  style="display:none;">
@@ -2443,62 +2450,86 @@
 
 <!-- 주소록 불러오기 -->
 <div class="tooltip-wrap">
-	<div class="popup-com import_layer popup06" tabindex="0" data-tooltip-con="popup06" data-focus="popup06" data-focus-prev="popup06-close" style="width: 1000px">
+	<div class="popup-com import_layer popup06 adr_call_popup" tabindex="0"
+		data-tooltip-con="popup06" data-focus="popup06"
+		data-focus-prev="popup06-close" style="width: 1000px">
 		<div class="popup_heading">
-			<p><span>주소록 불러오기</p>
+			<p>
+				<span>주소록 불러오기
+			</p>
 			<button type="button" onClick="javascript:addrClose(); return false;">
-			<img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+				<img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기">
+			</button>
 		</div>
 		<div class="layer_in">
 			<div class="titBox titBox_pad">
-				<p>- 주소록 수정 및 변경은 <span>[주소록 관리]</span>에서만 가능합니다.</p>
-				<button type="button" class="adr_admin" onClick="location.href='/web/mjon/addr/selectAddrList.do'">주소록 관리</button>
+				<p>
+					- 주소록 수정 및 변경은 <span>[주소록 관리]</span>에서만 가능합니다.
+				</p>
+				<button type="button" class="adr_admin"
+					onClick="location.href='/web/mjon/addr/selectAddrList.do'">주소록
+					관리</button>
 			</div>
 			<div class="adr_wrap">
-				<form id="searchAddrGrpForm" name="searchAddrGrpForm" method="post" style="display: flex; justify-content: space-between;">
-					<input type="hidden" id="searchAddrGrpId" name="searchAddrGrpId" value=""/>
-					<input type="hidden" id="type" name="type" value="all"/>
-					<input type="hidden" id="searchKeyword" name="searchKeyword" value=""/>
-					<input type="hidden" name="searchCondition" id="searchCondition" value="0" />
+				<form id="searchAddrGrpForm" name="searchAddrGrpForm" method="post"
+					style="display: flex; justify-content: space-between;">
+					<input type="hidden" id="searchAddrGrpId" name="searchAddrGrpId"
+						value="" /> <input type="hidden" id="type" name="type" value="all" />
+					<input type="hidden" id="searchKeyword" name="searchKeyword"
+						value="" /> <input type="hidden" name="searchCondition"
+						id="searchCondition" value="0" />
 					<div class="adr_pop_left">
 						<div class="adr_left_search">
-							<label for="searchGrpKeyword" class="label">그룹명 검색</label>
-							<input type="text" name="searchGrpKeyword" id="searchGrpKeyword"  placeholder="그룹명 검색" onfocus="this.placeholder=''" onblur="this.placeholder='그룹명 검색'" class="inputLight">
-							<button type="button" onClick="javascrit:fnAddrGrpSearch(); return false;"><img src="/publish/images/popup/search.png" alt="검색"></button>
+							<label for="searchKeyword" class="label">그룹명 검색</label> <input
+								type="text" name="searchGrpKeyword" id="searchGrpKeyword"
+								placeholder="그룹명 검색" onfocus="this.placeholder=''"
+								onblur="this.placeholder='그룹명 검색'" class="inputLight">
+							<button type="button"
+								onClick="javascrit:fnAddrGrpSearch(); return false;">
+								<img src="/publish/images/popup/search.png" alt="검색">
+							</button>
 						</div>
 						<div class="adr_pop_box">
-							<div id="addrGroupLoad">
-							</div>
+							<div id="addrGroupLoad"></div>
 						</div>
 						<!-- <div class="popup_btn">
-							<button type="button" class="btnType" onClick="javascript:fnSelectAddrGrpList(); return false;">선택 그룹 추가</button>
-						</div> -->
+                                <button type="button" class="btnType" onClick="javascript:fnSelectAddrGrpList(); return false;">선택 그룹 추가</button>
+                            </div> -->
 					</div>
 					<div class="adr_pop_right">
 						<div class="clearfix">
 							<div class="btnWrap_last">
-								<label for="searchAddrCondition" class="label">카테고리 선택</label>
-								<select id="searchAddrCondition" name="searchAddrCondition" class="selType2">
+								<label for="searchAddrCondition" class="label">카테고리 선택</label> <select
+									id="searchAddrCondition" name="searchAddrCondition"
+									class="selType2">
 									<option value='0'>전체</option>
 									<option value='1'>그룹명</option>
 									<option value='2'>이름</option>
 									<option value='3'>핸드폰번호</option>
-								</select>
-								<label for="searchAddrKeyword" class="label">검색어 입력</label>
-								<input type="text" id="searchAddrKeyword" name="searchAddrKeyword" placeholder="검색어를 입력하세요"  onfocus="this.placeholder=''" onblur="this.placeholder='검색어를 입력하세요'" >
-								<button type="button" class="btnType btnType17"  onClick="javascrit:fnAddrSearch(); return false;">검색</button>
+								</select> <label for="searchAddrKeyword" class="label">검색어 입력</label> <input
+									type="text" id="searchAddrKeyword" name="searchAddrKeyword"
+									placeholder="검색어를 입력하세요" onfocus="this.placeholder=''"
+									onblur="this.placeholder='검색어를 입력하세요'">
+								<button type="button" class="btnType btnType17"
+									onClick="javascrit:fnAddrSearch(); return false;">검색</button>
 							</div>
 							<!-- table -->
-							<div class="adr_excel adr_pop_list2 callAddr_box">
-							</div>
+							<div class="adr_excel adr_pop_list2 callAddr_box"
+								style="max-width: 722px"></div>
 							<!--// table -->
 						</div>
 						<div class="popup_btn_wrap2">
-							<button type="button" onClick="javascript:addrToList(); return false;">추가</button>
-							<button type="button" onClick="javascript:addrClose(); return false;">닫기</button>
+							<!--                                <button type="button" onClick="javascript:addrToList(); return false;">추가</button> -->
+							<button type="button"
+								onClick="javascript:addrToList_advc('all'); return false;">전체추가</button>
+							<button type="button"
+								onClick="javascript:addrToList_advc('select'); return false;">선택추가</button>
+							<button type="button"
+								onClick="javascript:addrClose(); return false;">닫기</button>
 						</div>
 						<%-- 주소록 레이어 팝업 닫기 실행 코드 --%>
-						<input type="hidden" name="btnAddrClose" id="btnAddrClose" class="tooltip-close closeAddr" data-focus="popup06-close" />         
+						<input type="hidden" name="btnAddrClose" id="btnAddrClose"
+							class="tooltip-close closeAddr" data-focus="popup06-close" />
 					</div>
 				</form>
 			</div>
@@ -2508,22 +2539,25 @@
 <!--// 주소록 불러오기 -->
 
 <!-- 엑셀 불러오기 -->
-<form id="excelToolTipForm" name="excelToolTipForm" method="post">
+<form id="excelForm" name="excelForm" method="post">
 	<div class="tooltip-wrap">
-		<div class="popup-com import_layer popup02" tabindex="0" data-tooltip-con="popup02" data-focus="popup02" data-focus-prev="popup02-close">
-			<div class="popup_heading">
+<!-- 	<div class="popup-com import_layer popup07" tabindex="0" data-tooltip-con="popup07" data-focus="popup07" data-focus-prev="popup07-close"> -->
+        <div class="popup-com adr_layer popup07" tabindex="0" data-tooltip-con="popup07" data-focus="popup07" data-focus-prev="popup07-close" style="width: 1000px;">
+            <%@include file="/WEB-INF/jsp/web/kakao/msgdata/include/ftDataIncludeExcel.jsp" %>
+<%--           <%@include file="/WEB-INF/jsp/web/msgdata/include/msgDataIncludeExcel.jsp" %> --%>
+			<!-- <div class="popup_heading">
 				<p><span>엑셀</span> 불러오기</p>
-				<button type="button" class="tooltip-close" data-focus="popup02-close"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+				<button type="button" class="tooltip-close" data-focus="popup07-close"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
 			</div>
 			<div class="layer_in">
-				<!-- 엑셀파일 불러오기 -->
+				엑셀파일 불러오기
 				<div class="hascont">
 					<div class="titBox">
 						<p>- 최대 2만 건까지 등록할 수 있습니다.</p>
 						<p>- [엑셀 불러오기]시 문서의 A, B열을 불러옵니다.(지원하는 파일 형식 : xls, xlsx)</p>
 						<p>- 휴대폰 항목은 숫자, 하이픈(-)만 인식하며, 번호 앞에 0이 생략되어도 정상 등록됩니다.
 						</p>
-						<!-- <button type="button" class="excel_btn" onclick="location.href='/cmm/fms/FileDown.do?atchFileId=FILE_000000000011651&fileSn=1'"><i></i>샘플파일 다운로드</button> -->
+						<button type="button" class="excel_btn" onclick="location.href='/cmm/fms/FileDown.do?atchFileId=FILE_000000000011651&fileSn=1'"><i></i>샘플파일 다운로드</button>
 						<button type="button" class="excel_btn" onclick="location.href='/download/msg/엑셀주소록_등록양식.xlsx'"><i class="downroad"></i>샘플파일 다운로드</button>
 					</div>
 					<div class="attachedFile">
@@ -2531,19 +2565,113 @@
 						<input type="text" id="excelNm" value="" readonly>
 						<input type="file" id="excelFile" accept=".xls, .xlsx" onchange="excelExport(event); return false;" style="display:none"/>
 						<button type="button" class="btnType btnType6 c1">찾아보기</button>
-<!-- 						<p><span class="vMiddle">*</span> 첨부된 파일은 <span class="c_e40000">[추가]버튼을 클릭</span>하셔야 받는 사람에 등록됩니다.</p> -->
+						<p><span class="vMiddle">*</span> 첨부된 파일은 <span class="c_e40000">[추가]버튼을 클릭</span>하셔야 받는 사람에 등록됩니다.</p>
 						<p><span class="vMiddle">*</span> 첨부된 파일은 <span class="c_e40000">[추가]버튼을 클릭</span>하셔야 받는 사람에 등록됩니다.</p>
 					</div>
-				</div><!--// 엑셀파일 불러오기 -->
+				</div>// 엑셀파일 불러오기
 				<div class="popup_btn_wrap2">
-					<button type="button" class="tooltip-close" data-focus="popup02-close"  data-focus-next="popup02" id="excelAdd">추가</button>
-					<button type="button" class="tooltip-close" data-focus="popup02-close"  data-focus-next="popup02">닫기</button>                      
+					<button type="button" class="tooltip-close" data-focus="popup07-close"  data-focus-next="popup07" id="excelAdd">추가</button>
+					<button type="button" class="tooltip-close" data-focus="popup07-close"  data-focus-next="popup07">닫기</button>                      
 				</div>
-			</div>
+			</div> -->
 		</div>
 	</div><!--// 엑셀 불러오기 -->
 </form>
 
+
+<div class="tooltip-wrap">
+	<div class="popup-com history_layer popup03" tabindex="0"
+		data-tooltip-con="popup03" data-focus="popup03"
+		data-focus-prev="popup03-close">
+		<div class="popup_heading">
+			<p>
+				전송내역
+			</p>
+			<button type="button" class="tooltip-close"
+				data-focus="popup03-close" id="btnLatestAddPhoneClose">
+				<img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기">
+			</button>
+		</div>
+		<div class="layer_in">
+			<!-- tab button -->
+			<ul class="tabType6">
+				<li class="tab active"><button type="button"
+						onclick="TabType(this,'1');">최근 전송내역</button></li>
+				<li class="tab"><button type="button"
+						onclick="TabType(this,'2');">자주보내는 번호</button></li>
+			</ul>
+			<!--// tab button -->
+			<!-- 최근 전송내역 -->
+			<div class="history_cont hascont current">
+				<div class="histroy_trans latestMsgArea" id="latestMsgArea">
+					<ul id="latestMsgUl">
+						<%-- <c:choose>
+                                <c:when test="${not empty resultLatestMsgList}">
+                                    <c:forEach var="latestMsgList" items="${resultLatestMsgList}" varStatus="status">
+                                        <li id="latestLi">
+                                            <input type="checkbox" id="addrChk_${status.count}" name="latAddrChk" value="<c:out value='${latestMsgList.callTo}'/>">
+                                            <label for="addrChk_${status.count}" class="label">최근 전송내역</label>
+                                            <p><c:out value="${latestMsgList.callTo}"/></p>
+                                            <button type="button" id="latestAddrDel">
+                                                <img src="/publish/images/popup/close3.png"  alt="전화번호 삭제">
+                                            </button>
+                                        </li>
+                                    </c:forEach>
+                                </c:when>
+                                <c:otherwise>
+                                    <li>
+                                        <p>최근 발송 내역이 없습니다.</p>
+                                    </li>
+                                </c:otherwise>
+                            </c:choose> --%>
+					</ul>
+				</div>
+				<div class="popup_btn_wrap2 hisroy_btn" style="width: 230px;">
+					<button type="button" id="latestAddPhoneAll">전체추가</button>
+					<button type="button" id="latestAddPhone">선택추가</button>
+					<button type="button" id="latestCancelPhone">선택취소</button>
+				</div>
+			</div>
+			<!--// 최근 전송내역 -->
+			<!-- 자주보내는 번호 -->
+			<div class="history_cont hascont">
+				<div class="histroy_trans" id="bookMarkMsgArea">
+					<ul id="bookMsgUl">
+						<li>
+							<p>데이터 로딩중입니다.</p>
+						</li>
+						<%-- <c:choose>
+                                <c:when test="${not empty resultBookMarkMsgList}">
+                                    <c:forEach var="bookMarkMsgList" items="${resultBookMarkMsgList}" varStatus="status">
+                                        <li id="bookMarkLi">
+                                            <input type="checkbox" id="bokAddrChk_${status.count}" name="bookAddrChk" value="<c:out value='${bookMarkMsgList.addrPhoneNo}'/>">
+                                            <label for="addrChk_${status.count}" class="label">최근 전송내역</label>
+                                            <p><c:out value="${bookMarkMsgList.addrPhoneNo}"/></p>
+                                            <button type="button" id="bookMarkAddrDel"><img src="/publish/images/popup/close3.png"  alt="전화번호 삭제"></button>
+                                        </li>
+                                    </c:forEach>
+                                </c:when>
+                                <c:otherwise>
+                                    <li>
+                                        <p>등록된 자주 보내는 번호 내역이 없습니다.</p>
+                                    </li>
+                                </c:otherwise>
+                            </c:choose> --%>
+					</ul>
+				</div>
+				<div class="popup_btn_wrap2 hisroy_btn" style="width: 230px;">
+					<button type="button" id="bookMarkAddPhoneAll">전체추가</button>
+					<button type="button" id="bookMarkAddPhone">선택추가</button>
+					<button type="button" id="bookMarkCancelPhone">선택취소</button>
+				</div>
+			</div>
+			<!--// 자주보내는 번호  -->
+		</div>
+	</div>
+</div>
+<!--// 전송내역 팝업 -->
+
+
 <form id="templateForm" name="templateForm" method="post">
 	<input type="hidden" id="friendId" name="friendId" value="<c:out value='${resultTemplateVO.friendId}'/>"/>
 </form>
(No newline at end of file)
 
src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView_advcbackup_20250414.jsp (added)
+++ src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/ft/KakaoFriendsTalkMsgDataView_advcbackup_20250414.jsp
@@ -0,0 +1,2549 @@
+<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
+<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
+<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"%>
+<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %>
+<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
+<%@ taglib prefix="ec" uri="/WEB-INF/tld/ecnet_tld.tld"%>
+<%@ page import="itn.com.cmm.LoginVO" %>
+<script src="/js/kakao/templateComm.js"></script>
+
+<script type="text/javascript" src="<c:out value='/js/MJUtill.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/ft/ftPriceClclt.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/ft/ftTabulator.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/txtSpecialReplace.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/at/addr.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/common/popup.js' />"></script>
+<script type="text/javascript" src="<c:out value='/js/kakao/ft/friendstalkExcel.js' />"></script>
+
+
+<script type="text/javascript">
+
+var loginVO = '${loginVO}';
+/* 파일등록 */
+var _fileIdx = 0;
+var _fileForm2 = new Array();
+var fileExt = ""; // 첨부파일 확장자
+var excelAddr = []; //엑셀 불러오기에서 내용 저장하는 배열 변수
+
+$(document).ready(function (){
+	
+	//화면 초기 로딩시 화면 처리해주기 - 템플릿 불러오기 했을 경우를 위함
+	initFormChk();
+	
+	//이모티콘 삽입 기능 처리
+	$(".symbolButton, .changeWord").on('click', function(){
+		
+		// 커서 위치에 삽입
+		setCursorInsertText("inputTemplateContent", $(this).attr("value"));
+		setContentsLeng($('#inputTemplateContent').val());
+	});
+	
+	
+	//즉시 발송 라디오 버튼 선택시 숨김처리
+	$("#reserYnN").on('click', function(){
+		$('.rev_selected').hide();
+		$('.send_rev .send_content').css('padding-bottom','108px');
+		$('.send_btn .btnType:first-child').html('발송하기');
+		$('#bizForm #reserveYn').val($(this).val());
+	});
+	
+	//예약 발송 라디오 버튼 선택시 숨김 해제처리
+	$("#reserYnY").on('click', function(){
+		$('.rev_selected').show();
+		$('.send_rev .send_content').css('padding-bottom','0');
+		$('.send_btn .btnType:first-child').html('예약하기');
+		$('#bizForm #reserveYn').val($(this).val());
+	});
+	
+	//템플릿 내용 입력 글자수 체크 해주기
+	$("#inputTemplateContent").keyup(function(e){
+		
+		var contents = $('#inputTemplateContent').val();
+		setContentsLengForFriends(contents);
+		
+	});
+	
+	$("#excelAdd").click(function(){
+		
+		if(excelAddr.length > 0){
+			
+			//첨부파일로 불러온 데이터를 받는사람 목록에 추가해 준다.
+			addPhoneInfo(excelAddr);
+			
+			 //첨부파일 초기화 시켜주기
+		    $("#excelFile").val("");
+		    $("#excelNm").val("");
+			
+		}else{
+			
+			alert("추가 엑셀 데이터가 없습니다.");
+			return false;
+			
+		}
+		
+	});
+	
+	
+	$("#send_fail_check").change(function(){
+		if($("#send_fail_check").is(":checked")){
+
+			
+			if(loginVO == "" || loginVO == null){
+				alert("로그인 후 사용 가능한 기능입니다.");
+				location.href="<c:url value='/web/user/login/login.do'/>";
+				return false;
+				
+				$("#send_fail_check").prop("checked", false);
+			}
+			
+			if($('#callFromList').val() === ''){
+				
+				if(confirm('대체문자 전송을 위한 발신번호가 등록되지 않았습니다. \n대체문자 발신번호를 지금 등록하시겠습니까?')){
+					window.location="<c:out value='/web/user/sendNumberManage.do' />";
+				}
+				$("#send_fail_check").prop("checked", false);
+				
+			}else{
+				// 에러버튼 체크 초기화
+				fn_insertErrorYN('N');
+				
+				$('#smsTxtArea').val('');
+// 				// 미리보기 텍스트를 가져와 줄바꿈 처리 후 대체문자 내용으로 입력
+				$('#smsTxtArea').val(
+					$('.template_text').html().trim().replace(/(<br>|<br\/>|<br \/>)/g, '\r\n')
+				);
+
+// 				//문자 내용 입력시 바이트수 계산하기
+				
+				$(".replace_send_wrap").slideDown(400);
+				thisFnByteString($('#smsTxtArea').val());
+			}
+		}else{
+			$(".replace_send_wrap").slideUp(400);
+			// 초기화 버튼 클릭
+			$('#failCheckInit').click();
+			$('.send_top .send_right .phone').css({'top': '0','transition': 'top .4s linear'});
+		}
+
+		// 금액 계산 fn 호출
+		totalFtPriceSum(tableL.getRows().length);
+		//fn_priceClclt();
+	});
+	
+	// 대체문자 내용 수정
+	$('#smsTxtArea').keyup(function(){
+		// 금액 계산 fn 호출
+		totalFtPriceSum(tableL.getRows().length);
+		//fn_priceClclt();
+		// 문자 바이트수 체크
+		thisFnByteString($('#smsTxtArea').val());
+		// 에러버튼 체크 초기화
+		fn_insertErrorYN('N');
+		
+	});
+	
+	
+	//대체문자 초기화 버튼 선택 처리
+	$("#failCheckInit").click(function(){
+		
+		$('#smsTxtArea').val("");
+		
+		// 금액 계산 fn 호출
+		totalFtPriceSum(tableL.getRows().length);
+		//fn_priceClclt();
+		// 문자 바이트수 체크
+		thisFnByteString($('#smsTxtArea').val());
+		// 에러버튼 체크 초기화
+		fn_insertErrorYN('N');
+		
+	});
+	
+	
+});
+
+function initFormChk(){
+	
+	//첫로딩시 우측 미리보기 화면 숨김처리
+	$('.emphasis_title_text').hide();	//강조유형 타이틀
+	$('.emphasis_side_text').hide();	//강조유형 보조문구
+	//$('.template_text').hide();			//내용 미리보기
+	$('.side_info_text').hide();		//부가정보 내용
+	$('.channel_info_text').hide();		//채널추가 텍스트 내용
+	$('.btnAd').hide();					//샘플 채널추가 버튼
+	//$('.btnEmpty').hide();				//샘플 버튼
+	
+	//채널ID 정보 미리보기에 표시해주기
+	fnAgentCodeChg();
+	
+	//템플릿 등록 이미지가 있는 경우 이미지 정보 표시해 주기
+	var imgType = $("input[name=img_file_add]:checked").val();
+	
+	if(!imgType == ''){
+		
+		if($("input[name='img_file_add']:checked").attr("id")=="img_file_1"){
+			$(".basic_img_add_wrap").show().siblings(".img_add_info_wrap").hide();
+			$(".img_file_add_wrap").show();
+		}else if($("input[name='img_file_add']:checked").attr("id")=="img_file_2"){
+			$(".wide_img_add_wrap").show().siblings(".img_add_info_wrap").hide();
+			$(".img_file_add_wrap").show();
+		} else{
+			$(".img_add_info_wrap, .img_file_add_wrap").hide();
+		}
+		
+	}else{
+		
+		$("#img_file_0").prop('checked',true);
+		
+	}
+	var imageFileName = '${resultTemplateVO.imageFileName}';
+	$("#imgNm").text(imageFileName);
+	
+	//미리보기에 이미지 표시해주기
+	var imgUrl = '${resultTemplateVO.templateImageUrl}';
+	
+	if(imgUrl != ''){
+		$('.kakao_image').css("display", "block");
+		$("#kakaoImg").attr("src", imgUrl);
+	}
+	
+	//초기 템플릿 내용 글자수 처리해주기
+	var tmpContents = $('#inputTemplateContent').val();
+	setContentsLengForFriends(tmpContents);
+	
+	//초기 광고포함 여부 처리
+	var adFlagVal = $("input[name=ad_flag]:checked").val();
+	advTextChange(adFlagVal);
+	
+	var reserYn = $("input[name=reserYn]:checked").val();
+	if(reserYn == 'N'){
+		
+		$('.rev_selected').hide();
+		$('.send_rev .send_content').css('padding-bottom','108px');
+		$('.send_btn .btnType:first-child').html('발송하기');
+		$('#bizForm #reserveYn').val('N');
+		
+	}
+	
+}
+
+//우측 미리보기 화면 채널 아이디 정보 변경해 주기
+function fnAgentCodeChg(){
+	var yellowIdVal = $("#selectAgentCode option:checked").val();
+	var yellowId = $("#selectAgentCode option:checked").text();
+	var yellowIdAt = yellowId.replaceAll('@','');
+	if(yellowIdVal != ''){
+		
+		$('#spnYellowid').text(yellowId);			//미리보기화면 채널ID
+		$('#spnYellowidAt').text(yellowIdAt);		//미리보기화면 채널ID @제거
+		$('#yellowIdCon').text(yellowIdAt);			//내용 입력 부분 채널ID 
+	}else{
+		
+		$('#spnYellowid').text("채널ID");			//미리보기화면 채널ID
+		$('#spnYellowidAt').text("채널ID");			//미리보기화면 채널ID @제거
+		$('#yellowIdCon').text("채널ID");			//내용 입력 부분 채널ID 
+	}
+}
+
+//파일첨부 버튼 클릭시 파일 첨부 실행
+function upImgClick(){
+	var selectChennelID = $("select[name='selectAgentCode']").val();	//채널ID 정보
+	var imageType = $("input[name=img_file_add]:checked").val();		//이미지 종류 정보(일반, 와이드)
+	var title = $("#imgTitle").val();
+	var link = $("#imgLink").val();
+	
+	if(selectChennelID == ''){
+		
+		$("#imgNm").text("");
+		$("#imgFile").val("");
+		alert("채널ID를 선택해 주세요.");
+		return false;
+		
+	}
+	
+	if(imageType != ''){
+		
+		bizForm.imageType.value = imageType;
+		
+	}else{
+		
+		$("#imgNm").text("");
+		$("#imgFile").val("");
+		alert("이미지 종류를 선택해 주세요.");
+		return false;
+		
+	}
+	
+	if(title == ''){
+		
+		$("#imgNm").text("");
+		$("#imgFile").val("");
+		alert("이미지 제목을 입력해 주세요.");
+		return false;
+		
+	}
+	
+	//이미지 선택시 링크 정보 및 url 패턴 검사
+	if(link == ''){
+		
+		$("#imgNm").text("");
+		$("#imgFile").val("");
+		alert("이미지 클릭시 이동할 URL 주소를 http:// 또는 https:// 포함하여 입력해 주세요.");
+		return false;
+		
+	}else{
+		
+		if(link.search("http://") == -1 && link.search("https://") == -1){
+			
+			$("#imgNm").text("");
+			$("#imgFile").val("");
+			alert("이미지 URL 주소에는 http:// 또는 https://를 포함하여 입력해야 합니다.");
+			return false;
+			
+		}
+		
+	}
+	
+	//첨부파일 선택 팝업 호출해주기
+	$("#imgFile").click();
+}
+
+//첨부 이미지 정보 확인 처리 해주기(이미지 리사이징 전단계)
+function imgResizeInfo(event){
+	
+	var fileValue = $("#imgFile").val().split("\\");
+	var fileName = fileValue[fileValue.length-1];
+	
+	var fileExt = fileName.split('.').pop().toLowerCase();
+
+	if(fileExt.length > 0){
+		if($.inArray(fileExt, ['jpg','jpeg','png']) == -1) {
+			alert('jpg, jpeg, png 파일만 업로드 할수 있습니다.');
+			return false;
+	   	}
+	   
+		$("#imgNm").text(fileName); 
+	}
+	
+	//알림톡 이미지 등록 요청하기
+	getTemplateImagUrl();
+}
+
+//카카오 친구톡 이미지 등록 요청 API 호출해주기
+function getTemplateImagUrl(){
+	
+	var bizForm = document.bizForm;
+	
+	var data = new FormData(bizForm);
+	var atchFileSts = true;	//첨부파일 갯수 상태 값1
+	
+	data.append("file0", $('#imgFile').prop('files')[0]);
+	
+	var fileValue = $("#imgFile").val().split("\\");
+	var fileName = fileValue[fileValue.length-1];
+	
+	var fileExt = fileName.split('.').pop().toLowerCase();
+	
+	if($.inArray(fileExt, ['jpg','jpeg', 'png']) == -1) {
+
+		alert('jpg, jpeg 파일만 업로드 할수 있습니다.');
+		return false;
+
+   	}
+	
+	var url = "";
+	
+	if(fileExt == "jpg" || fileExt == "jpeg" || fileExt == "png"){
+		
+		url = "/web/mjon/kakao/template/sendKakaoFriendsTemplateImageUploadAjax.do";
+		
+	}else{
+		
+		alert('jpg, jpeg 파일만 업로드 할수 있습니다.');
+		return;
+		
+	}
+	
+	$.ajax({
+		type : 'POST'
+		, enctype : 'multipart/form-data'
+		, url : url
+		, data : data
+		, dataType:'json'
+		, async:true
+		, processData: false
+		, contentType: false
+		, cache : false
+		, success : function(returnData, status){
+			if(returnData.result == "success") {
+				
+				var code = returnData.code;
+				var msg = returnData.msg;
+				var imgUrl = returnData.imgUrl;
+				
+				if(code == "200"){//등록 성공시 메시지 및 이미지 표시 처리
+					alert("이미지 등록이 완료 되었습니다.");
+					$("#templateImageName").val(fileName);
+					$("#templateImageUrl").val(imgUrl);
+				
+					$('.kakao_image').css("display", "block");
+					$("#kakaoImg").attr("src", imgUrl);
+					$("#imageFileName").val(fileName);
+					
+				}else if(code == "2002"){//일반 이미지 가로 세로 크기 오류 발생
+					
+					alert("권장 이미지 사이즈는  가로 720px 세로 720px(가로:세로 비율이 2:1 아닐 경우, 가로 500px, 세로 250px 이하일 경우 업로드 불가), 최대용량 500KB 입니다.");
+					
+					//첨부파일 정보를 지워준다.
+					$("#imgFile").val("");
+				    $("#imgNm").text("");  
+					return false;
+					
+				}else if(code == "405"){
+					
+					alert("권장 이미지 사이즈는  가로 800px 세로 600px(가로:세로 비율이 2:1 아닐 경우, 가로 500px, 세로 250px 이하일 경우 업로드 불가), 최대용량 2MB 입니다.");
+					
+					//첨부파일 정보를 지워준다.
+					$("#imgFile").val("");
+				    $("#imgNm").text("");  
+					return false;
+					
+				}else{
+					
+					alert(msg);
+					
+					//첨부파일 정보를 지워준다.
+					$("#imgFile").val("");
+				    $("#imgNm").text(""); 
+					return false;
+				}
+				
+			}else if(returnData.result == "loginFail"){
+				location.href="/web/user/login/login.do";
+			}else{
+				alert("이미지 등록 중 오류가 발생하였습니다.!!!");
+				return false;
+			}
+		},
+		beforeSend : function(xmlHttpRequest) {
+        	//로딩창 show
+        	$('.loading_layer').addClass('active');				
+		},	        	        
+        complete : function(xhr, textStatus) {
+        	//로딩창 hide
+        	$('.loading_layer').removeClass('active');
+		},
+		error : function(request , status, error){
+			alert("code:"+request.status+"\n"+"message:"+request.responseText+"\n"+"error:"+error);
+		}
+	});
+	
+	//첨부파일 이름 및 데이터 지워주기 - 완료 후 팝업이 자동으로 닫힘.
+	$("#imgFile").val("");
+}
+
+function fnImageDel(){
+	
+	var selectAgentCode = $("select[name='selectAgentCode']").val();	// 선택 채널ID
+	var imageUrl = $("#templateImageUrl").val();
+	var bizForm = document.bizForm;
+	
+	bizForm.senderKey.value = selectAgentCode;
+	
+	if(imageUrl == ''){
+		
+		alert("등록된 이미지가 없습니다.");
+		return false;
+		
+	}
+	
+	var tmpFriendId = $("#tmpFriendId").val();
+	
+	if(tmpFriendId != ''){
+		
+		if(!confirm("불러온 템플릿의 이미지도 함께 삭제 됩니다. 계속하시겠습니까?")){
+			return false;
+		}
+		
+	}
+	
+	var data = new FormData(bizForm);
+	
+	if(confirm("등록된 이미지를 삭제하시겠습니까?")){
+		
+		$.ajax({
+			type: "POST"
+			, url: "/web/mjon/kakao/template/deleteKakaoFriendsImageDataAjax.do"
+			, data: data
+			, dataType: 'json'
+			, async: false
+			, processData: false
+			, contentType: false
+			, cache: false
+			, success: function (returnData, status) {
+				
+				var result = returnData.result;
+				var code = returnData.code;
+				var msg = returnData.msg;
+				
+				if(result == "notLogin") {//차단 발신번호인 경우 등록 불가
+					alert("로그인후 진행이 가능합니다.");
+				}else if(returnData.result == "success") {
+					
+					if(code == "200"){
+						alert("등록된 이미지가 삭제되었습니다.");
+						$("#imgTitle").val("");
+						$("#imgLink").val("");
+						$("#imgFile").val("");
+						$("#imgNm").text("첨부파일 이미지");
+						$("#templateImageUrl").val("");
+						$('.kakao_image').css("display", "none");
+						$("#kakaoImg").attr("src", "");
+						$("#imageFileName").val("");	//이미지 파일명 삭제
+						$("#img_file_0").click();		//이미지 첨부 안함 선택
+						return false;
+						
+					}else{
+						alert(msg);
+						return false;
+					}
+				}else{
+					alert("이미지 삭제에 오류가 발생하였습니다.");
+					return false;
+				}
+			}
+			,error: function (e) {
+				console.log("ERROR : ", e);
+				alert("이미지 삭제에 오류가 발생하였습니다.");
+			}
+		});
+	}
+}
+
+//알림톡 내용 초기화 해주기
+function fnContentsReset(){
+	$("#inputTemplateContent").val("");
+	$(".template_text").text("내용 미리보기");
+	$('.nowChar').text("0 /");
+}
+
+//저장 템플릿 불러오기 선택시 
+function myTemplateSelect(ftId){
+	
+	var form = document.templateForm;
+	
+	form.friendId.value = ftId;
+	
+	form.action="/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgDataView.do";
+	form.submit();
+	
+	
+}
+
+//광고성 정보 포함 선택에 따른 텍스트 표시 부분 처리
+$(document).on('change', '.inputAdFlag', function(){
+	
+	var adFlagVal = $(this).val();
+	advTextChange(adFlagVal);
+	
+});
+
+function advTextChange(adFlagVal){
+	
+	if(typeof adFlagVal == "undefined" || adFlagVal == null || adFlagVal == ""){
+		
+		$("#ad_Y").prop('checked',true);
+		
+	}else if(adFlagVal == 'N'){//광고 표시 숨김처리
+		
+		$('.adFlag').hide();
+		$('.kakao_block_text').hide();
+		$('#adFlagFront').hide();
+		$('.adFlagAfter').hide();
+		
+	}else{
+		$('.adFlag').show();
+		$('.kakao_block_text').show();
+		$('#adFlagFront').show();
+		$('.adFlagAfter').show();
+	}
+	
+}
+
+
+//친구톡 내용 템플릿으로 저장하기 - 내용 저장하기
+function myTemplateSave(){
+
+	var selectAgentCode = $("select[name='selectAgentCode']").val();	// 선택 채널ID
+	var yellowId = $("select[name='selectAgentCode'] option:selected").text().replaceAll('@','');			// 선택 채널명
+	var inputTemplateName = $("#inputTemplateName").val();				// 입력 템플릿 이름
+	var imageType = $("input[name=img_file_add]:checked").val();		// 첨부 이미지 종류(없음, 일반, 와이드 이미지)
+	var imageTitle = $("#imgTitle").val();								//첨부이미지 제목
+	var imageLink = $("#imgLink").val();								//첨부이미지 클릭시 이동 링크 주소
+	var inputTemplateAd = $("input[name=ad_flag]:checked").val();	// 광고성메시지 선택 여부
+	var inputTemplateContent = $("#inputTemplateContent").val();		// 템플릿 내용
+	
+	var inputTemplateImageName = $("#templateImageName").val();	// 템플릿 이미지 파일명
+	var inputTemplateImageUrl = $("#templateImageUrl").val();		// 템플릿 이미지 링크
+	
+	// 공통 유효성 검사 조회
+	if(selectAgentCode == null || selectAgentCode == ""){
+		alert("채널ID를 선택해 주세요");
+		return;
+	};
+	
+	if(inputTemplateName == null || inputTemplateName == ""){
+		alert("템플릿 명을 입력해 주세요");
+		$("#inputTemplateName").focus();
+		return;
+	};
+	
+	if(imageType == 'I' || imageType == 'W'){
+		
+		if(imageTitle == ''){
+			
+			alert("이미지 제목을 입력해 주세요.");
+			return false;
+			
+		}
+		
+		if(imageLink == ''){
+			
+			alert("이미지 클릭시 이동할 URL을 입력해 주세요.");
+			return false;
+			
+		}else{
+			
+			if(imageLink.search("http://") == -1 && imageLink.search("https://") == -1){
+				
+				alert("이미지 URL 주소에는 http:// 또는 https://를 포함하여 입력해야 합니다.");
+				return false;
+				
+			}
+			
+		}
+		
+		if(inputTemplateImageUrl == ''){
+			
+			alert("이미지를 선택해 주세요.");
+			return false;
+			
+		}
+		
+	}
+	
+	
+	if(inputTemplateContent == null || inputTemplateContent == ""){
+		alert("템플릿 내용을 입력해 주세요.");
+		return;
+	};
+	
+	var bizForm = document.bizForm;
+	bizForm.senderKey.value = selectAgentCode;
+	//bizForm.yellowId.value = yellowId;
+	bizForm.templateName.value = inputTemplateName;
+	//bizForm.imageType.value = imageType;
+	//bizForm.imgTitle.value = imageTitle;
+	//bizForm.imgLink.value = imageLink;
+	bizForm.adFlag.value = inputTemplateAd;
+	bizForm.templateContent.value = inputTemplateContent;
+	
+	var data = new FormData(bizForm);
+	
+	if(confirm("템플릿을 저장하시겠습니까?")){
+		
+		$.ajax({
+			type: "POST"
+			, url: "/web/mjon/kakao/template/insertKakaoFriendsTemplateDataAjax.do"
+			, data: data
+			, dataType: 'json'
+			, async: false
+			, processData: false
+			, contentType: false
+			, cache: false
+			, success: function (returnData, status) {
+				
+				var result = returnData.result;
+				
+				if(result == "loginFail") {//차단 발신번호인 경우 등록 불가
+					alert("로그인후 진행이 가능합니다.");
+				}else if(returnData.result == "success") {
+					alert("템플릿 저장이 완료되었습니다.");
+					return false;
+				}else{
+					
+					alert("템플릿 저장에 오류가 발생하였습니다.");
+					return false;
+					
+				}
+			}
+			,error: function (e) {
+				console.log("ERROR : ", e);
+				alert("템플릿 저장에 오류가 발생하였습니다.");
+			}
+		});
+		
+	}
+	
+}
+
+//주소록 불러오기 버튼 클릭시
+$('.popupAddr').click(function(){
+	
+	if(loginVO == "" || loginVO == null){
+		alert("주소록 불러오기 서비스는 로그인 후 이용 가능합니다.");
+		location.href="<c:url value='/web/user/login/login.do'/>";
+		return false;
+	}
+	
+	$("#addrGroupLoad").load("/web/mjon/msgdata/selectAddrGroupListAjax.do", "" ,function(response, status, xhr){
+		//리스트 스크롤 처리해주기
+		$(".adr_pop_list").mCustomScrollbar({
+			axis: 'y',
+			scrollbarPosition: "outside",
+			theme: "dark",
+			autoHideScrollbar: false
+		});
+	});
+	
+});
+
+function fnEmptyId(){
+	
+	alert("로그인 후 이용이 가능합니다.");
+	return false;
+	
+}
+
+//분할전송 건수 데이터 체크
+function checkNumber(event) {
+	var divideCnt = $('#frmDivideCnt').val();
+	if(!(event.key >= 0 && event.key <= 9)) {
+		return false;
+	}
+	
+	var totCnt = divideCnt + "" + event.key;
+	if(Number(totCnt) > 5000){
+		alert("분할전송 건수는 5,000건을 초과할 수 없습니다.");
+		$('#frmDivideCnt').val("20");
+		return false;
+	}
+  
+  return true;
+}
+
+function fnSelectMinChk(){
+	
+	var resHour = $("#msgResHour option:selected").val();
+	var resMin = $("#msgResMin option:selected").val();
+	
+	if(resHour == '20'){
+		
+		if(resMin >= 50 && resMin <= 59){
+			
+			alert("친구톡은 20:50 ~ 익일 08:00까지 발송할 수 없습니다.");
+			$("#msgResMin").val("40").prop("selected", true);
+			
+		}
+		
+	}
+	
+}
+
+
+//친구톡 발송 금지 시간 처리
+function fn_noTimeSend(){
+	
+	var id = '${loginVO.id}';
+
+	if(id == ""){
+		
+		alert("로그인 후 이용이 가능합니다.");
+		return false;
+		
+	}
+	
+	alert("친구톡은 20:50 ~ 익일 08:00까지 발송할 수 없습니다.");
+	return false;
+	
+}
+
+//친구톡 데이터 전송 처리
+function fn_sendMsgData(){
+	
+	var senderKey = $("#selectAgentCode option:checked").val();
+	
+	if(senderKey == ''){
+		
+		alert("채널ID를 선택해 주세요.");
+		return false;
+		
+	}else{
+		
+		$("#inputSenderKey").val(senderKey);
+		
+	}
+	
+	//이미지 첨부 체크	
+	var imgChk = $("input[name=img_file_add]:checked").val();
+	if(!getFtImageType(imgChk)){
+		return false;
+	}
+	
+	//템플릿 버튼 갯수 체크 
+	var buttonCnt = $(".btn_kakao_type").length;
+
+	if(buttonCnt > 0){
+		//json 파일 필요 유무 셋팅
+		$("#bizJsonYn").val("Y");
+	}
+	
+	//친구톡 내용 체크
+	var tmpContents = $('#inputTemplateContent').val();
+
+	if(tmpContents == ""){
+		
+		alert("친구톡 내용을 입력해 주세요.");
+		return false;
+		
+	}
+	
+	var rtnStr = strChinJpnCheck(tmpContents);
+	
+	//문자내용에 이모지가 있는지 체크
+	if(!emojiCheck(tmpContents)) return false;
+	
+	if(rtnStr.length > 0){
+		
+		alert("입력하신 문구 중 \" " + rtnStr + " \" 는 일부 휴대폰에서 표기되지 않을 수 있습니다.");
+		
+	}
+	
+
+	//수신번호 리스트 체크하기
+	var numCnt = 0;
+	var nameList = [];		//치환문자 이름
+	var phoneNum = [];	//받는사람
+	var rep1List = [];		//치환문자1
+	var rep2List = [];		//치환문자2
+	var rep3List = [];		//치환문자3
+	var rep4List = [];		//치환문자4
+	
+	var varValList = [];		//치환문자 연결시킬 변수 셋팅
+	
+	var selectedData = tableL.getRows();
+	
+	var varValStatus = true; //치환분자 데이터 체크용
+	
+	if(selectedData == "" || selectedData == null){
+		
+		alert("받는사람 주소를 한 건 이상 입력해주세요.");
+		return false;
+	
+	}else{ // 선택한 Row '-' 문자 삭제하기
+		
+		var txtReplYn = $("#txtReplYn").val();
+		
+		if(txtReplYn == 'Y'){//치환문자가 있는 경우 변수 치환 처리
+		
+			for(var i=0; i < selectedData.length; i++){
+				
+				var nmStatus = false;
+				var rep1Status = false;
+				var rep2Status = false;
+				var rep3Status = false;
+				var rep4Status = false;
+				
+				
+				if(tmpContents.indexOf("\#{이름}")  > -1){
+					
+					nmStatus = true;
+				}
+				
+				if(tmpContents.indexOf("\#{1}")  > -1){
+					
+					rep1Status = true;
+				}
+				
+				if(tmpContents.indexOf("\#{2}")  > -1){
+					
+					rep2Status = true;
+				}
+				
+				if(tmpContents.indexOf("\#{3}")  > -1){
+					
+					rep3Status = true;
+				}
+				
+				if(tmpContents.indexOf("\#{4}")  > -1){
+					
+					rep4Status = true;
+				}
+				
+				//일괄변환 문자에 콤마(,)가 들어가있으면 배열로 넘길때 문제가 발생하여 특수문자(§)로 치환하여 넘겨주도록 한다.
+				var name = tableL.getRows()[i].getData().name;
+				var phone = removeDash(tableL.getRows()[i].getData().phone);
+				var rep1 = tableL.getRows()[i].getData().rep1;
+				var rep2 = tableL.getRows()[i].getData().rep2;
+				var rep3 = tableL.getRows()[i].getData().rep3;
+				var rep4 = tableL.getRows()[i].getData().rep4;
+				var varValStr = "";	//¶ 구분자
+				
+				if(phone == ""){
+					
+					alert("수신 목록에 핸드폰 번호가 없는 항목이 있습니다.");
+					return false;
+					
+				}else if(!checkHpNum(phone)){
+					
+					alert("수신 목록에 잘 못된 핸드폰 번호가 있습니다. 핸드폰 번호 : " + phone + " 입니다.");
+					return false;
+					
+				}else{
+					
+					if(typeof(name) != 'undefined' && name != null && name !=""){
+						if(!emojiCheck(name)){//이모지 체크 해주기
+							return false;
+						}
+						
+						//이름 치환변수가 있으면 저장
+						if(nmStatus){
+							//nameList[i] = name.replaceAll(",","§");
+							if(varValStr == ''){
+								
+								varValStr = name.replaceAll(",","§");
+							}else{
+								
+								varValStr = varValStr + "¶" +  name.replaceAll(",","§");
+								
+							}
+						}
+					}else{
+						if(nmStatus){
+							varValStatus = false;
+						}
+					}
+
+					
+					if(phone != '' && phone != null){
+						//연락처 변수 저장하기
+						phoneNum[i] = phone;
+						if(varValStr == ''){
+							varValStr = phone;	
+						}else{
+							varValStr = varValStr + "¶" +  phone;
+						}
+					}else{
+						varValStatus = false;
+					}
+					
+					if(typeof(rep1) != 'undefined' && rep1 != null && rep1 !=""){
+						if(!emojiCheck(rep1)){//이모지 체크 해주기
+							return false;
+						}
+						
+						if(rep1Status){
+							
+							if(varValStr == ''){
+								//rep1List[i] = rep1.replaceAll(",","§");
+								varValStr = rep1.replaceAll(",","§");
+							}else{
+								//rep1List[i] = rep1.replaceAll(",","§");
+								varValStr = varValStr + "¶" +  rep1.replaceAll(",","§");
+							}
+						}
+					}else{
+						if(rep1Status){
+							varValStatus = false;
+						}
+					}
+					
+					if(typeof(rep2) != 'undefined' && rep2 != null && rep2 !=""){
+						if(!emojiCheck(rep2)){//이모지 체크 해주기
+							return false;
+						}
+						
+						if(rep2Status){
+							
+							if(varValStr == ''){
+								varValStr = rep2.replaceAll(",","§");
+							}else{
+								//rep2List[i] = rep2.replaceAll(",","§");
+								varValStr = varValStr + "¶" +  rep2.replaceAll(",","§");
+							}
+						}
+					}else{
+						if(rep2Status){
+							varValStatus = false;
+						}
+					}
+					
+					
+					if(typeof(rep3) != 'undefined' && rep3 != null && rep3 !=""){
+						if(!emojiCheck(rep3)){//이모지 체크 해주기
+							return false;
+						}
+						
+						if(rep3Status){
+							
+							if(varValStr == ''){
+								varValStr = rep3.replaceAll(",","§");
+							}else{
+								//rep3List[i] = rep3.replaceAll(",","§");
+								varValStr = varValStr + "¶" +  rep3.replaceAll(",","§");
+							}
+						}
+					}else{
+						if(rep3Status){
+							varValStatus = false;
+						}
+					}
+					
+					
+					if(typeof(rep4) != 'undefined' && rep4 != null && rep4 !=""){
+						if(!emojiCheck(rep4)){//이모지 체크 해주기
+							return false;
+						}
+						
+						if(rep4Status){
+							
+							if(varValStr == ''){
+								varValStr = rep4.replaceAll(",","§");
+							}else{
+								//rep4List[i] = rep4.replaceAll(",","§");
+								varValStr = varValStr + "¶" +  rep4.replaceAll(",","§");
+							}
+						}
+					}else{
+						if(rep4Status){
+							varValStatus = false;
+						}
+					}
+					
+				}
+				
+				varValList[i] = varValStr;
+				
+			}
+		
+		}else{//치환문자가 없는 경우 휴대폰 번호만 입력
+			
+			for(var i=0; i < selectedData.length; i++){
+				
+				//일괄변환 문자에 콤마(,)가 들어가있으면 배열로 넘길때 문제가 발생하여 특수문자(§)로 치환하여 넘겨주도록 한다.
+				var phone = removeDash(tableL.getRows()[i].getData().phone);
+				
+				if(phone == ""){
+					
+					alert("수신 목록에 핸드폰 번호가 없는 항목이 있습니다.");
+					return false;
+					
+				}else if(!checkHpNum(phone)){
+					
+					alert("수신 목록에 잘 못된 핸드폰 번호가 있습니다. 핸드폰 번호 : " + phone + " 입니다.");
+					return false;
+					
+				}else{
+					
+					phoneNum[i] = phone;
+					
+				}
+				
+			}
+		
+		}
+	
+	}
+	
+	//예약문자 시간 체크
+	var reserYn = $("input[name=reserYn]:checked").val(); // 예약 발송 여부 확인
+	
+	if(reserYn == 'Y'){
+		
+		var date = $(".resDate").val();//form.msgResDate.value;
+		var hour = $("#msgResHour option:selected").val();
+		var min = $("#msgResMin option:selected").val();
+		
+		if(date == ""){
+			
+			alert("예약전송 날짜를 선택해 주세요.");
+			return false;
+			
+		}else{
+			
+			var now = new Date();
+			var reqDate = date + " " + hour + ":" + min + ":00";
+			var gapDate = getGapDayTime(date, hour, min);
+			
+			if(gapDate < 0){ // 음수이면 이전날짜, 크면 이후 날짜.
+				alert("예약 날짜는 현재 시간 이후의 날짜 및 시간을 선택해 주세요.");
+				return false;
+			}else{
+				$("#reqDate").val(reqDate);	//예약일자 파라미터 저장
+			}
+		}
+
+	}else{
+		
+		$("#reqDate").val("");	//예약일자 파라미터 저장
+		
+	}
+	
+	
+	//대체문자 선택 및 내용 체크
+	var subMsgSendYn = "N";
+	
+	if($("#send_fail_check").is(":checked")){
+		
+		subMsgSendYn = "Y";
+		$("#subMsgSendYn").val("Y");
+		
+		//대체문자 체크사항 함수 호출
+		if(!fn_subMsgCheck()){ 
+			
+			/* $("#send_fail_check").prop("checked", false);
+			$(".replace_send_wrap").slideUp(400);
+			// 초기화 버튼 클릭
+			$('#failCheckInit').click();
+			$('.send_top .send_right .phone').css({'top': '0','transition': 'top .4s linear'}); */
+			return false;
+		
+		}
+		
+	}else{
+		
+		subMsgSendYn = "N";
+		$("#subMsgSendYn").val("N");
+		
+	}
+	
+	//수신전화번호 목록
+	$("#callToList").val(phoneNum);
+	
+	//치환변수 데이터 정보 목록(수신번호 포함, tabulator 정보 모두 )
+	$("#varValList").val(varValList);
+
+	//템플릿 내용 입력
+	$("#templateContent").val(tmpContents);
+	
+	//광고포함 여부
+	var adFlag = $("input[name=ad_flag]:checked").val();
+	$("#adFlag").val(adFlag);
+	
+	//치환문자에 대한 데이터 누락 체크
+	if(!varValStatus){
+		
+		alert("특정문구 일괄변환에 대한 일부 데이터가 누락된 부분이 있습니다. 데이터를 확인해 주세요.");
+		return false;
+		
+	}
+	
+	if(!confirm("친구톡을 발송하시겠습니까?")){
+		
+		return false;
+		
+	}
+	
+	var spamChk = true;
+	
+	var spmData = new FormData(document.bizForm);
+	$.ajax({
+		type: "POST"
+		, url: "/web/mjon/kakao/friendstalk/selectSpamKakaoFriendsTalkMsgChkAjax.do"
+		, data: spmData
+		, dataType:'json'
+		, async: false
+		, processData: false
+		, contentType: false
+		, cache: false
+		, success: function (returnData, status) {
+			if(status == 'success'){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
+				
+				if("fail" == returnData.result){
+					alert(returnData.message);
+					spamChk = false;
+					return false;
+				}else if("loginFail" == returnData.result){
+					alert(returnData.message);
+					spamChk = false;
+					return false;
+				}else if("spams" == returnData.result){
+					//alert("전송 내용에  스팸문구가 포함되어 있습니다.")
+					$("#spamStatus").val("Y");
+					return false;
+				}else{
+					spamChk = true;
+					return false;
+				}
+				
+			} else if(status== 'fail'){
+				alert(returnData.message);
+				return false;
+			}
+		}
+		, error: function (e) {
+			alert("문자 발송에 실패하였습니다.");
+			console.log("ERROR : ", e);
+			return false;
+		}
+	});
+	
+	if(spamChk){
+		var data = new FormData(document.bizForm);
+		$.ajax({
+			type: "POST"
+			, url: "/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgSendAjax.do"
+			, data: data
+			, dataType: 'json'
+			, async: true
+			, processData: false
+			, contentType: false
+			, cache: false
+			, success: function (returnData, status) {
+				if(status == 'success'){
+					if("loginFail" == returnData.result){
+						
+						alert(returnData.message);
+						return false;
+						
+					}else  if('fail' == returnData.result){
+						
+						alert(returnData.message);
+						return false;
+						
+					}else if('authFail' == returnData.result){
+						
+						alert(returnData.message);
+						location.reload();
+						
+					} else if(status == 'success'){
+						
+						var kakaoSendCnt = returnData.resultSts;
+						
+						$('.pop_msg_success').css({'display':'block','opacity':'1','left':'50%','top':'50%','transform':'translate(-50%,-50%)'});
+						
+						//예약발송 건의 경우 결과 팝업 문구 변경
+						if(reserYn == 'Y'){
+							$('.pop_msg_success .msg_text').html("예약 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 예약 되었습니다.");
+						}else{
+							$('.pop_msg_success .msg_text').html("발송 성공 : <strong>"+ kakaoSendCnt + "</strong>건의<br>친구톡이 발송 되었습니다.");
+						}
+						
+						$('.mask').addClass('on');
+					}
+				}
+			}
+			,beforeSend : function(xmlHttpRequest) {
+				//로딩창 show
+				$('.loading_layer').addClass('active');
+			}
+			,complete : function(xhr, textStatus) {
+				//로딩창 hide
+				$('.loading_layer').removeClass('active');
+			}
+			,error: function (e) {
+				console.log("ERROR : ", e);
+				alert("카카오 친구톡 전송에 실패하였습니다.");
+			}
+		}); 
+	}
+	
+}
+
+
+/**
+ * @description 대체문자 오류체크 funciton
+ */
+function fn_errorChk(){
+
+	// 대체문자가 없을 시 return false;
+	if($('#txtReplYn').val() === 'N')
+	{
+		alert('오류가 없습니다.');
+		return false;
+	}
+
+	// 치환 부분 변수명만 추출 = 배열
+	var varList = $("#inputTemplateContent").val().match(/#\{([^}]+)\}/g);
+	
+	var smsTxt = $('#smsTxtArea').val();
+	for(var i=0; i < varList.length; i++){
+		if(smsTxt.indexOf(varList[i]) < 0){
+			if(confirm(varList[i] + '값이 없습니다. 치환문자 없이 진행하시겠습니까?')){
+				fn_insertErrorYN('Y');
+			};
+			return false;
+		}
+		smsTxt = smsTxt.replace(varList[i], '');
+	};
+	alert('오류가 없습니다.');
+	fn_insertErrorYN('Y');
+}
+
+
+function fn_insertErrorYN(val){
+	$('#errorChk').val(val);
+}
+
+/*
+ *대체문자 필수 항목 체크 
+ *
+ **/
+ 
+function fn_subMsgCheck(){
+
+	if($('#callFromList').val() === ''){
+		
+		if(confirm('대체문자 전송을 위한 발신번호가 등록되지 않았습니다. \n대체문자 발신번호를 지금 등록하시겠습니까?')){
+			window.location="<c:out value='/web/user/sendNumberManage.do' />";
+		}
+		$("#send_fail_check").prop("checked", false);
+		
+		return false;
+		
+	}else{
+	
+		//발신번호 입력 처리
+		$("#callFrom").val(removeDash($('#callFromList').val()));
+		
+	}
+	
+	if($("#smsTxtArea").val() === ''){
+		
+		alert("대체문자 내용을 입력해 주세요.");
+		return false;
+		
+	}else{//대체문자에 치환문자 여부 체크
+		
+		var smsTxtArea = $("#smsTxtArea").val();
+		var replStatus = false;
+	
+		if(smsTxtArea.indexOf("\#{이름}")  > -1){
+			replStatus = true;
+		}
+		
+		if(smsTxtArea.indexOf("\#{1}")  > -1){
+			replStatus = true;
+		}
+		
+		if(smsTxtArea.indexOf("\#{2}")  > -1){
+			replStatus = true;
+		}
+		
+		if(smsTxtArea.indexOf("\#{3}")  > -1){
+			replStatus = true;
+		}
+		
+		if(smsTxtArea.indexOf("\#{4}")  > -1){
+			replStatus = true;
+		}
+		
+		if(replStatus){
+			$("#subMsgTxtReplYn").val("Y");
+		}else{
+			$("#subMsgTxtReplYn").val("N");
+		}
+		
+		$("#subMsgTxt").val(smsTxtArea);
+		
+	}
+	
+	
+	
+	return true;
+	
+}
+
+//문자 바이트수 계산하기 함수
+function thisFnByteString(contents){
+	var totalByte = 0; 
+	//var content = contents;
+	var adverYn = $("input[name='send_adYn']:checked").val();
+	var adTxtLeng = 0;
+	var denyTxtLeng = 0;
+
+	$('#msgLeng').html("");
+	$('#limitLeng').html("");
+	var conLeng = conByteLeng(contents); // 내용 문자 입력 바이트 수 계산하기
+	
+		
+	$('#msgLeng').text(conLeng);
+	
+	//문자 길이 변수에 저장해주기
+	$('#smsLen').val(conLeng);
+	
+	
+	if(conLeng > 90){
+		
+		$('#msgLeng').html(conLeng + " / ");
+		$('#limitLeng').html("2000");
+		$('.msg_com').html("장문");
+		$('#msgType').val("6"); // 메세지 타입 설정
+		
+		$('.msg_com').removeClass("msg_short"); //단문 클래스 삭제하고
+		$('.put_left').removeClass("short"); //내용 입력 박스 클래스 삭제
+		$('.msg_com').addClass("msg_long"); // 장문 클래스 삽입
+		$("#subMsgType").val("LMS");
+		
+	}else{
+		
+		$('#msgLeng').html(conLeng + " / ");
+		$('#limitLeng').html("90");
+		$('.msg_com').html("단문");
+		$('#msgType').val("4"); // 메세지 타입 설정
+		$('.msg_com').removeClass("msg_long"); //단문 클래스 삭제하고
+		$('.put_left').removeClass("long"); //내용 입력 박스 클래스 삭제
+		$('.msg_com').addClass("msg_short"); // 장문 클래스 삽입
+		$("#subMsgType").val("SMS");
+		
+	}
+		
+}
+
+function msgSuccessClose(obj){
+	$(obj).closest('.pop_msg_success').attr('style','');
+	location.reload(true);
+	$('html').scrollTop(0);
+}
+
+function getFtImageType(imgChk){
+	
+	if(imgChk == 'I' || imgChk == 'W'){
+		
+		var imgTitle = $("#imgTitle").val();
+		var imgLink = $("#imgLink").val();
+		var imgFileName = $("#imageFileName").val();
+		
+		if(imgTitle == ''){
+			
+			alert("친구톡 이미지 제목정보를 입력해 주세요.");
+			return false;
+			
+		}
+		
+		if(imgLink == ''){
+			
+			alert("친구톡 이미지 선택시 이동할 링크 주소를 입력해 주세요.");
+			return false;
+			
+		}
+		
+		if(imgFileName == ''){
+			
+			alert("친구톡 이미지를 등록해 주세요.");
+			return false;
+			
+		}
+		
+		$("#imageType").val(imgChk);
+		
+		//json 파일 필요 유무 셋팅
+		$("#bizJsonYn").val("Y");
+	}
+	
+	return true;
+	
+}
+
+
+function goToKakaoTestPopUp(){
+	
+	if(loginVO == "" || loginVO == null){
+		alert("테스트 발송 서비스는 로그인 후 이용 가능합니다.");
+		location.href="<c:url value='/web/user/login/login.do'/>";
+		return false;
+	}
+	
+	//기업회원 체크
+	if(!usrDeptChk()){
+		return false;
+	}
+	
+	var form = document.bizForm;
+	var senderKey = $("#selectAgentCode option:checked").val();
+	var yellowId = $("#selectAgentCode option:checked").text();
+	
+	if(senderKey == ''){
+		alert("채널ID를 선택해 주세요.");
+		return false;
+	}else{
+		$("#inputSenderKey").val(senderKey);
+		$("#yellowId").val(yellowId);
+	}
+	
+	var imgChk = $("input[name=img_file_add]:checked").val();
+	if(!getFtImageType(imgChk)){
+		return false;
+	}
+	
+	//템플릿 버튼 갯수 체크 
+	var buttonCnt = $(".btn_kakao_type").length;
+
+	if(buttonCnt > 0){
+		//json 파일 필요 유무 셋팅
+		$("#bizJsonYn").val("Y");
+	}
+	
+	//친구톡 내용 체크
+	var tmpContents = $('#inputTemplateContent').val();
+
+	if(tmpContents == ""){
+		
+		alert("친구톡 내용을 입력해 주세요.");
+		return false;
+		
+	}else{
+		
+		//템플릿 내용 입력
+		$("#templateContent").val(tmpContents);
+		
+	}
+	
+	var rtnStr = strChinJpnCheck(tmpContents);
+	
+	//문자내용에 이모지가 있는지 체크
+	if(!emojiCheck(tmpContents)) return false;
+	
+	if(rtnStr.length > 0){
+		
+		alert("입력하신 문구 중 \" " + rtnStr + " \" 는 일부 휴대폰에서 표기되지 않을 수 있습니다.");
+		
+	}
+	
+	
+	var selectedData = tableL.getRows();
+	
+	if(selectedData == "" || selectedData == null){
+		
+		alert("받는사람 주소를 한 건 이상 입력해주세요.");
+		return false;
+	
+	}else{
+		
+		//치환문구 변환 
+		var txtReplYn = $("#txtReplYn").val();
+		
+		if(txtReplYn == 'Y'){
+			
+			var name = tableL.getRows()[0].getData().name;
+			var phone = removeDash(tableL.getRows()[0].getData().phone);
+			var rep1 = tableL.getRows()[0].getData().rep1;
+			var rep2 = tableL.getRows()[0].getData().rep2;
+			var rep3 = tableL.getRows()[0].getData().rep3;
+			var rep4 = tableL.getRows()[0].getData().rep4;
+			
+			var varValList = [];		//치환문자 연결시킬 변수 셋팅
+			
+			
+			var nmStatus = false;
+			var rep1Status = false;
+			var rep2Status = false;
+			var rep3Status = false;
+			var rep4Status = false;
+			
+			var varValStr = "";
+			var varValStatus = true;
+
+			
+			if(tmpContents.indexOf("\#{이름}")  > -1){
+				nmStatus = true;
+			}
+			
+			if(tmpContents.indexOf("\#{1}")  > -1){
+				rep1Status = true;
+			}
+			
+			if(tmpContents.indexOf("\#{2}")  > -1){
+				rep2Status = true;
+			}
+			
+			if(tmpContents.indexOf("\#{3}")  > -1){
+				rep3Status = true;
+			}
+			
+			if(tmpContents.indexOf("\#{4}")  > -1){
+				rep4Status = true;
+			}
+			
+	
+			if(nmStatus && (typeof(name) != 'undefined' && name != null && name !="")){
+				
+				if(varValStr == ''){
+					varValStr = name.replaceAll(",","§");
+				}else{
+					varValStr = varValStr + "¶" +  name.replaceAll(",","§");
+				}
+				
+			}else{
+				
+				if(nmStatus){
+					varValStatus = false;
+				}
+				
+			}
+			
+			if(varValStr == ''){
+				varValStr = phone;
+			}else{
+				varValStr = varValStr + "¶" +  phone;
+			}
+
+			if(rep1Status && (typeof(rep1) != 'undefined' && rep1 != null && rep1 !="")){
+				
+				if(varValStr == ''){
+					varValStr = rep1.replaceAll(",","§");
+				}else{
+					varValStr = varValStr + "¶" +  rep1.replaceAll(",","§");
+				}
+				
+			}else{
+				
+				if(rep1Status){
+					varValStatus = false;
+				}
+				
+			}
+			
+			
+			if(rep2Status && (typeof(rep2) != 'undefined' && rep2 != null && rep2 !="")){
+			
+				if(varValStr == ''){
+					varValStr = rep2.replaceAll(",","§");
+				}else{
+					varValStr = varValStr + "¶" +  rep2.replaceAll(",","§");
+				}
+				
+			}else{
+				
+				if(rep2Status){
+					varValStatus = false;
+				}
+				
+			}
+			
+			if(rep3Status && (typeof(rep3) != 'undefined' && rep3 != null && rep3 !="")){
+				
+				if(varValStr == ''){
+					varValStr = rep3.replaceAll(",","§");
+				}else{
+					varValStr = varValStr + "¶" +  rep3.replaceAll(",","§");
+				}
+				
+			}else{
+				
+				if(rep3Status){
+					varValStatus = false;
+				}
+				
+			}
+			
+			if(rep4Status && (typeof(rep4) != 'undefined' && rep4 != null && rep4 !="")){
+				
+				if(varValStr == ''){
+					varValStr = rep4.replaceAll(",","§");
+				}else{
+					varValStr = varValStr + "¶" +  rep4.replaceAll(",","§");
+				}
+			
+			}else{
+				
+				if(rep4Status){
+					varValStatus = false;
+				}
+				
+			}
+			
+			if(!varValStatus){
+				
+				alert("특정문구 일괄변환에 대한 일부 데이터가 누락된 부분이 있습니다. 데이터를 확인해 주세요.");
+				return false;
+				
+			}
+			
+			varValList[0] = varValStr;
+			
+			$("#varValList").val(varValList);
+		}
+		
+	}
+
+	form.method = "post"; 
+	window.open("about:blank", 'testSendPop', 'width=770, height=850, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=1');
+	form.target = "testSendPop";
+	form.action = "/web/mjon/kakao/friendstalk/selectKakaoFriendsTalkTestSendPopup.do";
+	form.submit();
+}
+
+
+function msgResultLink(){
+	var reserYn = $("input[name=reserYn]:checked").val(); // 예약 발송 여부 확인
+	if(reserYn == 'Y'){
+		
+		location.href="/web/mjon/reservmsg/selectReservKaKaoView.do";
+		
+	}else{
+		location.href="/web/kakao/sent/selectKakaoSentView.do";
+	}
+}
+
+</script>
+
+<!-- 로딩바 -->
+<div class="loading_layer">
+	<div class="loading_container">
+		<div class="bar"></div>
+		<div class="text">Loading</div>
+	</div>
+</div>
+
+<div class="inner">
+	<div class="send_top">
+		<!-- tab button -->
+		<%@include file="/WEB-INF/jsp/web/kakao/include/KaKaoAlimtalkTopMenuTap.jsp" %>
+		<!-- // tab button -->
+		<div class="top_content kakaotalksend_cont current pay_tab_wrap">
+			<div class="heading">
+				<h2>친구톡 전송</h2>
+				<button type="button" class="button info">친구톡 사용방법</button>
+			</div>
+			
+			<!-- 유저 보유잔액 -->
+			<input type="hidden" id="oriUserMoney" value="<c:out value='${userMoney}' />">
+			
+			<!-- 각 금액 단가  -->
+			<input type="hidden" id="kakaoFtPrice" value="<c:out value='${sendPrice.kakaoFtPrice}' />"> <!-- 친구톡 단가  -->
+			<input type="hidden" id="longPrice" value="<c:out value='${sendPrice.longPrice}' />"> <!-- mms 단가  -->
+			<input type="hidden" id="shortPrice" value="<c:out value='${sendPrice.shortPrice}' />"> <!-- sms 단가  -->
+			
+			<form id="bizForm" name="bizForm" method="multipart/form-data">
+				<input type="hidden" id="menuTopTab" name="menuTopTab" value="tabFriend">
+				<input type="hidden" id="imageType" name="imageType" value="<c:out value='${resultTemplateVO.imageType}'/>"/>
+				<input type="hidden" id="inputSenderKey" name="senderKey" value="<c:out value='${resultTemplateVO.senderKey}'/>"/>
+				<input type="hidden" id="imageFileName" name="imageFileName" value="<c:out value='${resultTemplateVO.imageFileName}'/>">
+				<input type="hidden" id="templateImageUrl" name="templateImageUrl" value="<c:out value='${resultTemplateVO.templateImageUrl}'/>">
+				<input type="hidden" id="tmpFriendId" name="friendId" value="<c:out value='${resultTemplateVO.friendId}'/>">
+				<input type="hidden" id="templateName" name="templateName" value=""/>
+				<input type="hidden" id="adFlag" name="adFlag" value="N"/>
+				<input type="hidden" id="templateContent" name="templateContent" value=""/>
+				
+				<!-- 예약발송 여부 -->
+				<input type="hidden" id="reserveYn" name="reserveYn" value="N"/>
+				<input type="hidden" id="reqDate" name="reqDate" value=""/>
+				
+				<!-- 변환문자 유무 -->
+				<input type="hidden" id="txtReplYn" name="txtReplYn" value="N"/>
+				<input type="hidden" id="varValList" name="varValList" value=""/>
+				
+				<!-- 수신목록 -->
+				<input type="hidden" id="callToList" name="callToList" value=""/>
+				
+				<!-- 스팸 문구 유무 -->
+				<input type="hidden" id="spamStatus" name="spamStatus" value="N"/>
+				<input type="hidden" id="atSmishingYn" name="atSmishingYn" value="<c:out value='${atSmishingYn}' />"/>
+				
+				<!-- 오류 체크 -->
+				<input type="hidden" id="errorChk" name="errorChk" value="N"/>
+				
+				<!-- 대체문자 여부 -->
+				<input type="hidden" id="subMsgSendYn" name="subMsgSendYn" value="N"/>
+				<input type="hidden" id="subMsgType" name="subMsgType" value="SMS"/>
+				<input type="hidden" id="subMsgTxtReplYn" name="subMsgTxtReplYn" value="N"/>
+				<input type="hidden" id="subMsgTxt" name="subMsgTxt" value=""/>
+				
+				<!-- 발신번호정보 -->
+				<input type="hidden" id="callFrom" name="callFrom" value=""/>
+				
+				<!-- 친구톡 발송 단가 정보 -->
+				<input type="hidden" id="kakaoFtPrice" name="kakaoFtPrice" value="<c:out value='${sendPrice.kakaoFtPrice}' />"/>
+				<input type="hidden" id="eachPrice" name="eachPrice" value="0"/>
+				<input type="hidden" id="totPrice" name="totalPrice" value="0"/>
+				
+				<!-- 친구톡 발송 json 파일 필요 유무 -->
+				<input type="hidden" id="bizJsonYn" name="bizJsonYn" value="N"/>
+				
+				<input type="hidden" id="yellowId" name="yellowId" value=""/>
+				
+				
+				<div class="send_general friend_talk_wrap kakao_wrap">
+					<div class="send_left">
+						<table class="tType1">
+							<colgroup>
+								<col style="width: 130px;">
+								<col style="width: auto;">
+							</colgroup>
+							<tbody>
+								<tr>
+									<th>채널ID</th>
+									<td>
+										<label for="selectAgentCode" class="채널ID 선택"></label>
+										<select class="select_gray_type" name="selectAgentCode" id="selectAgentCode" onchange="javascript:fnAgentCodeChg();">
+											<option value="">채널ID 선택</option>
+											<c:forEach var="kakaoProfileInfo" items="${resultProfileList}" varStatus="status">
+												<option value="${kakaoProfileInfo.senderKey}" <c:if test="${kakaoProfileInfo.senderKey eq resultTemplateVO.senderKey}">selected</c:if> ><c:out value='${kakaoProfileInfo.yellowId}'/></option>
+											</c:forEach>
+										</select>
+									</td>
+								</tr>
+								<tr>
+	                                <th>템플릿명</th>
+	                                <td>
+	                                    <input type="text" class="template_name" id="inputTemplateName" name="inputTemplateName" value="<c:out value='${resultTemplateVO.templateName}'/>" placeholder="템플릿명을 입력해주세요. (최대 50자)">
+	                                    <p class="template_name_cf">최대 50자, 템플릿 관리용</p>
+	                                </td>
+	                            </tr>
+								<tr>
+									<th>이미지 첨부</th>
+									<td>
+										<div class="img_sort_wrap">
+											<input type="radio" name="img_file_add" id="img_file_0" value="" <c:if test="${resultTemplateVO.imageType eq ''}">checked</c:if> > <label for="img_file_0">첨부안함</label>
+											<input type="radio" name="img_file_add" id="img_file_1" value="I" <c:if test="${resultTemplateVO.imageType eq 'I'}">checked</c:if> > <label for="img_file_1">이미지 첨부</label>
+											<input type="radio" name="img_file_add" id="img_file_2" value="W" <c:if test="${resultTemplateVO.imageType eq 'W'}">checked</c:if> > <label for="img_file_2">와이드 이미지 첨부</label>
+											<div class="img_add_info_wrap basic_img_add_wrap">
+												<p class="info_title_text"><span class="c_e40000">*</span> 이미지 첨부 안내</p>
+												<ul class="info_text">
+												<li>- 권장사이즈 : 720px * 720px</li>
+												<li>- 제한사이즈 : 가로 500px 미만, 가로:세로 비율이 2:1 미만 또는 3:4 초과시 업로드 불가</li>
+												<li>- 파일형식 : jpg, png (최대 500kb)</li>
+												<li>- 이미지 첨부시 메시지 내용은 최대 400자까지 입력할 수 있습니다.</li>
+												</ul>
+											</div>
+											<div class="img_add_info_wrap wide_img_add_wrap">
+												<p class="info_title_text"><span class="c_e40000">*</span> 와이드 이미지 첨부 안내</p>
+												<ul class="info_text">
+												<li>- 권장사이즈 : 800px * 600px</li>
+												<!-- <li>- 제한사이즈 : 가로 500px 미만, 가로:세로 비율이 2:1 미만 또는 3:4 초과시 업로드 불가</li> -->
+												<li>- 파일형식 : jpg, png (최대 2mb)</li>
+												<li>- 와이드 이미지 첨부시 메시지 내용은 최대 76자, 버튼 1개까지 입력할 수 있습니다.</li>
+												</ul>
+											</div>
+										</div>
+										<div class="img_file_add_wrap">
+											<ul class="img_file_info_wrap">
+												<li>
+													<p>이미지 제목</p>
+													<input type="text" id="imgTitle" name="imgTitle" value="<c:out value='${resultTemplateVO.imgTitle}'/>" class="img_url" maxLength="50">
+												</li>
+												<li>
+													<p>이미지 클릭시 이동할 URL</p>
+													<input type="text" id="imgLink" name="imgLink" value="<c:out value='${resultTemplateVO.imgLink}'/>" placeholder="http://" class="img_url" maxLength="1000">
+												</li>
+												<li>
+													<p class="file_name"  id="imgNm" name="imgNm">첨부파일 이미지</p>
+													<button type="button" class="btn_del btnImgDel" onclick="buttonTypeDel(this);">
+														<img src="/publish/images/btn_delete.png" id="imgDel" alt="이미지 삭제" onclick="fnImageDel();">
+													</button>
+												</li>
+											</ul>
+											<button type="button" class="btnType btnType8 btn_img_upload" title="이미지 불러오기" onclick="javascript:upImgClick(); return false;">이미지 불러오기</button>
+											<input type="file" id="imgFile" accept=".jpg, .jpeg, .png" onchange="imgResizeInfo(event); return false;" style="display:none"/>
+										</div>
+									</td>
+								</tr>
+								<tr>
+									<th>광고포함 여부</th>
+									<td>
+										<input type="radio" class="inputAdFlag" name="ad_flag" id="ad_Y" value="Y" <c:if test="${resultTemplateVO.adFlag eq 'Y'}">checked</c:if> ><label for="ad_Y">광고성 정보 포함</label>
+										<%-- <input type="radio" class="inputAdFlag" name="ad_flag" id="ad_N" value="N" <c:if test="${resultTemplateVO.adFlag eq 'N'}">checked</c:if> ><label for="ad_N">포함 안함</label> --%>
+									</td>
+								</tr>
+								<tr>
+									<th>내용</th>
+									<td class="kakao_template_text">
+										<div class="put_left short">
+											<div class="put_text_wrap">
+												<div class="ad_txt">
+													<p><span id="adFlagFront">(광고)</span> <span id="yellowIdCon" >채널ID</span></p>
+												</div>
+												<label for="inputTemplateContent" class="label"></label>
+												<textarea id="inputTemplateContent" name="inputTemplateContent" class="put_text" placeholder="내용을 입력해주세요."><c:out value="${resultTemplateVO.templateContent}"/></textarea>
+												<div class="text_length">
+													<div>
+														<p class="adFlagAfter">수신거부 : 홈 > 채널차단</p>
+													</div>
+													<div>
+														<p><span class="fwMd nowChar">0 /</span><span class="c_002c9a fwMd totChar">1000</span>자</p>
+													</div>
+												</div>
+											</div>
+										</div>
+										<div class="put_right">
+											<div class="btn_popup_wrap">
+												<button type="button" class="btnType btnType7" onclick="miniPopup(this)">특수문자</button>
+													<div class="send_miniPop spc_character">
+														<div class="tab_character">
+															<a href="#none" class="on">특수문자</a>
+															<a href="#none">웃음</a>
+															<a href="#none">슬픔</a>
+															<a href="#none">분노</a>
+															<a href="#none">사랑</a>
+															<a href="#none">그외</a>
+														</div>
+														<!-- 특수문자 -->
+														<div class="cnt_character on">
+															<div class="box_character">
+															<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'D'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}"><c:out value="${symbolList.symbol}"/></a>
+																</c:if>
+															</c:forEach>
+															</div>
+														</div>
+														<!-- //특수문자 -->
+														
+														<!-- 웃음 -->
+														<div class="cnt_character emt_character">
+															<div class="box_character">
+																<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'S'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}">${symbolList.symbol}</a>
+																</c:if>
+																</c:forEach>
+															</div>
+														</div>
+														<!-- //웃음 -->
+														
+														<!-- 슬픔 -->
+														<div class="cnt_character emt_character">
+															<div class="box_character">
+																<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'C'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}">${symbolList.symbol}</a>
+																</c:if>
+																</c:forEach>
+															</div>
+														</div>
+														<!-- //슬픔 -->
+														
+														<!-- 분노 -->
+														<div class="cnt_character emt_character">
+															<div class="box_character">
+																<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'A'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}">${symbolList.symbol}</a>
+																</c:if>
+																</c:forEach>
+															</div>
+														</div>
+														<!-- //분노 -->
+														
+														<!-- 사랑 -->
+														<div class="cnt_character emt_character">
+															<div class="box_character">
+																<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'L'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}">${symbolList.symbol}</a>
+																</c:if>
+																</c:forEach>
+															</div>
+														</div>
+														<!-- //사랑 -->
+														
+														<!-- 그외 -->
+														<div class="cnt_character emt_character emt_etc ">
+															<div class="box_character">
+																<c:forEach var="symbolList" items="${symbolList}" varStatus="status">
+																<c:if test="${symbolList.symbolType == 'E'}">
+																	<a href="javascript:void(0)" class="symbolButton" value="${symbolList.symbol}">${symbolList.symbol}</a>
+																</c:if>
+																</c:forEach>
+															</div>
+														</div>
+														<!-- //그외 -->
+													<button type="button" class="btn_close" onclick="miniPopup(this)">닫기</button>
+												</div>
+											</div>
+											<div class="btn_popup_wrap convers_wrap">
+												<button type="button" class="btnType btnType7" onclick="miniPopup(this)">특정문구 일괄변환<i class="qmMark"></i></button>
+												<div class="send_miniPop convers">
+													<div>
+														<div class="convers_top">
+															<span>특정문구 일괄변환이란?</span>
+															<p>주소록, 엑셀에 입력된 내용을 이용해 수신자마다 다른 내용의 메시지를 발송하는 기능</p>
+														</div>
+														<div class="convers_middle">
+															<a href="javascript:void(0)" class="changeWord" value="\#{이름}"><c:out value="\#{이름}"/></a>
+														</div>
+														<div class="convers_bottom">
+															<a href="javascript:void(0)" class="changeWord" value="\#{1}"><c:out value="\#{1}"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="\#{2}"><c:out value="\#{2}"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="\#{3}"><c:out value="\#{3}"/></a>
+															<a href="javascript:void(0)" class="changeWord" value="\#{4}"><c:out value="\#{4}"/></a>
+														</div>
+													</div>
+													<button type="button" class="btn_close" onclick="miniPopup(this)">닫기</button>
+												</div>
+											</div>
+											<div class="btn_popup_wrap spc_wrap">
+												<button type="button" class="btnType btnType7" onclick="miniPopup(this)">이모티콘</button>
+												<div class="send_miniPop spc_character kakao_emoticon">
+													<%@include file="/WEB-INF/jsp/web/kakao/include/KakaoSentImoticonInfo.jsp" %>
+													<button type="button" class="btn_close" onclick="miniPopup(this)">닫기</button>
+												</div>
+											</div>
+											<div class="btn_popup_wrap template_call_wrap">
+												<c:choose>
+													<c:when test="${not empty loginVO.id}">
+														<button type="button" class="btnType btnType8" onclick="window.open('/web/mjon/kakao/template/selectKakaoFriendsTemplateListPopupAjax.do','_blank','width=930, height=780, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">템플릿 불러오기</button>
+													</c:when>
+													<c:otherwise>
+														<button type="button" class="btnType btnType8" onclick="javascript:fnEmptyId();">템플릿 불러오기</button>
+													</c:otherwise>
+												</c:choose>
+											</div>
+											<div class="send_btnWrap">
+												<c:choose>
+													<c:when test="${not empty loginVO.id}">
+														<button type="button" class="btnType btn_text_save" onclick="javascript:myTemplateSave();">내용 저장하기</button>
+													</c:when>
+													<c:otherwise>
+														<button type="button" class="btnType btn_text_save" onclick="javascript:fnEmptyId();">내용 저장하기</button>
+													</c:otherwise>
+												</c:choose>
+												
+												<button type="button" class="btnType btnType9" onclick="javascript:fnContentsReset();">초기화</button>
+											</div>
+										</div>
+									</td>
+								</tr>
+								<tr>
+									<th>버튼</th>
+									<td class="template_button">
+										<select name="tmpBtnSelect" id="tmpBtnSelect" class="select_gray_type template_button_select">
+											<option value="button_type_0">버튼타입 선택</option>
+											<option value="button_type_1">배송조회</option>
+											<option value="button_type_2">웹링크</option>
+											<option value="button_type_3">앱링크</option>
+											<option value="button_type_5">메시지전달</option>
+										</select>
+										<button type="button" class="btnType btnType6" onclick="friendTemplateButtonAdd();">추가</button>
+										<p class="cf_text fwRg">
+											<span class="c_e40000 fwBold">*</span> 버튼 타입중 <span class="c_e40000">봇키워드, 메시지전달</span>은 <span class="c_222">카카오톡 채널 관리자센터(https://center-pf.kakao.com)</span>에서 설정을 직접 한 후 이용하셔야 동작합니다. (최대 5개까지 등록가능)
+										</p>
+										<div class="button_add_wrap">
+	                                    	<c:forEach var="buttonList" items="${resultTemplateVO.buttonVOList}" varStatus="status">
+	                                    	
+	                                    		<c:if test="${buttonList.linkType eq 'DS'}">
+	                                    			<!-- 배송조회 버튼  -->
+			                                    	<dl class="button_type_wrap type1">
+				                                    	<dt>배송조회</dt>
+				                                    	<dd class="button_type_input">
+				                                    		<input type="hidden" id="buttonLikeTypeDeliv" name="buttonVOList[${status.index}].linkType" value="DS"/>
+				                                    		<input type="text" id="btnNmDeliv" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+				                                    		<p class="cf_text">*이용가능 택배사 : KG로지스, 우체국택배,일양로지스, GTX로지스, FedEx, 경동택배, 합동택배, 롯데택배</p>
+				                                    	</dd>
+				                                    	<dd>
+				                                    		<button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button>
+				                                    	</dd>
+			                                    	</dl>
+	                                    		</c:if>
+		                                    	
+		                                    	<c:if test="${buttonList.linkType eq 'WL'}">
+			                                    	<!-- 웹링크 버튼 -->
+			                                    	<dl class="button_type_wrap type2">
+			                                    		<dt>웹링크</dt>
+			                                    		<dd class="button_type_input">
+			                                    			<ul>
+			                                    				<li>
+			                                    					<input type="hidden" id="buttonLikeTypeWeb" name="buttonVOList[${status.index}].linkType" value="WL"/>
+			                                    					<input type="text" id="btnNmWeb" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+			                                    				</li>
+			                                    				<li>
+			                                    					<input type="text" id="buttonLinkMo" name="buttonVOList[${status.index}].linkMo" value="<c:out value='${buttonList.linkMo}'/>" placeholder="모바일 링크 입력">
+			                                    				</li>
+			                                    				<li>
+			                                    					<input type="text" id="buttonLinkPc" name="buttonVOList[${status.index}].linkPc" value="<c:out value='${buttonList.linkPc}'/>" placeholder="PC 링크 입력">
+			                                    				</li>
+			                                    			</ul>
+			                                    		</dd>
+			                                    		<dd>
+			                                    			<button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button>
+			                                    		</dd>
+			                                    	</dl>
+		                                    	</c:if>
+		                                    	
+		                                    	<c:if test="${buttonList.linkType eq 'AL'}">
+			                                    	<!-- 앱링크 버튼 -->
+			                                    	<dl class="button_type_wrap type3">
+			                                    		<dt>앱링크</dt>
+			                                    		<dd class="button_type_input">
+			                                    			<ul>
+			                                    				<li>
+			                                    					<input type="hidden" id="buttonLikeTypeApp" name="buttonVOList[${status.index}].linkType" value="AL"/>
+			                                    					<input type="text" id="btnNmApp" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+			                                    				</li>
+			                                    				<li>
+			                                    					<input type="text" id="buttonLinkAnd" name="buttonVOList[${status.index}].linkAnd" value="<c:out value='${buttonList.linkAnd}'/>" placeholder="Android 링크 입력">
+			                                    				</li>
+			                                    				<li>
+			                                    					<input type="text" id="buttonLinkIos" name="buttonVOList[${status.index}].linkIos" value="<c:out value='${buttonList.linkIos}'/>" placeholder="IOS 링크 입력">
+			                                    				</li>
+			                                    			</ul>
+			                                    		</dd>
+			                                    		<dd>
+			                                    			<button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button>
+			                                    		</dd>
+			                                    	</dl>
+		                                    	</c:if>
+		                                    	
+		                                    	<c:if test="${buttonList.linkType eq 'BK'}">
+			                                    	<!-- 봇키워드 -->
+			                                    	<dl class="button_type_wrap type4">
+			                                    		<dt>봇키워드</dt>
+		                                    			<dd class="button_type_input">
+		                                    				<input type="hidden" id="buttonLikeTypeBot" name="buttonVOList[${status.index}].linkType" value="BK"/>
+		                                    				<input type="text" id="btnNmBot" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+		                                    			</dd>
+		                                    			<dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd>
+			                                    	</dl>
+		                                    	</c:if>
+		                                    	
+		                                    	<c:if test="${buttonList.linkType eq 'MD'}">
+			                                    	<!-- 메시지 전달 -->
+			                                    	<dl class="button_type_wrap type5">
+													   <dt>메시지전달</dt>
+													   <dd class="button_type_input">
+													      <input type="hidden" id="buttonLikeTypeMsg" name="buttonVOList[${status.index}].linkType" value="MD" />
+													      <input type="text" id="btnNmMsg" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+													   </dd>
+													   <dd>
+													      <button type="button" class="btn_del" onclick="buttonTypeDel(this);">
+													         <img src="/publish/images/btn_delete.png" alt="">
+													      </button>
+													   </dd>
+													</dl>
+												</c:if>	                                    	
+		                                    	
+		                                    	<c:if test="${buttonList.linkType eq 'BC'}">
+			                                    	<!-- 상담톡 전환 -->
+													<dl class="button_type_wrap type6">
+													   <dt>상담톡전환</dt>
+													   <dd class="button_type_input">
+													      <input type="hidden" id="buttonLikeTypeCons" name="buttonVOList[${status.index}].linkType" value="BC" />
+													      <input type="text" id="btnNmCons" name="buttonVOList[${status.index}].name" value="<c:out value='${buttonList.name}'/>" placeholder="버튼명 입력(최대 14자)">
+													   </dd>
+													   <dd>
+													      <button type="button" class="btn_del" onclick="buttonTypeDel(this);">
+													         <img src="/publish/images/btn_delete.png" alt="">
+													      </button>
+													   </dd>
+													</dl>
+												</c:if>
+												
+												<c:if test="${buttonList.linkType eq 'BT'}">
+													<!-- 챗봇전환 -->
+													<dl class="button_type_wrap type7">
+													   <dt>챗봇전환</dt>
+													   <dd class="button_type_input">
+													      <input type="hidden" id="buttonLikeTypeBotChg" name="buttonVOList[${status.index}].linkType" value="BT" />
+													      <input type="text" id="btnNmBotChg" name="buttonVOList[${status.index}].name" value="" placeholder="버튼명 입력(최대 14자)">
+													   </dd>
+													   <dd>
+													      <button type="button" class="btn_del" onclick="buttonTypeDel(this);">
+													         <img src="/publish/images/btn_delete.png" alt="">
+													      </button>
+													   </dd>
+													</dl>
+												</c:if>
+											</c:forEach>
+	                                    </div>
+									</td>
+								</tr>
+								<tr>
+									<th>받는사람</th>
+									<td class="putText">
+										<div class="clearfix receipt_num">
+											<div class="receipt_num_top">
+												<label for="callTo" class="label">받는 번호입력</label>
+												<input type="text" placeholder="번호를 입력하세요" id="callTo" name="callTo" onfocus="this.placeholder=''" onblur="this.placeholder='번호를 입력하세요'" style="width:340px;">
+												<button type="button" class="btnType btnType6 addCallToF">번호추가</button>
+												<span><span class="vMiddle">*</span> 중복번호는 한번만 발송됩니다.</span>
+											</div>
+											<div class="receipt_num_midde">
+												<div class="listType list01 callList_box"  id="callList_box">
+													<div class="list_table list_head">
+														<div class="cb_wrap">
+															<label for="select_all" class="label"></label>
+															<input type="checkbox" id="select_all">
+														</div>
+														<div class="list_table_num">
+															<p>휴대폰</p>
+															<img src="/publish/images/sortUp.png">
+															<img src="/publish/images/sortDown.png">
+														</div>
+														<div class="list_table_name">
+															<p>이름</p>
+															<img src="/publish/images/sortUp.png">
+															<img src="/publish/images/sortDown.png">
+														</div>
+													</div>
+													<div class="list_body_wrap" id="wrap01_body">
+													</div>
+												</div>
+												<div class="put_right">
+													<div class="btn_popup_wrap spc_wrap">
+														<button type="button" data-tooltip="popup06" class="btnType btnType7 popupAddr">주소록 불러오기</button>
+													</div>
+													<div class="btn_popup_wrap">
+														<button type="button" data-tooltip="popup02" class="btnType btnType7">엑셀 불러오기</button>
+													</div>
+													<div class="btn_popup_wrap">
+														<button type="button" data-tooltip="popup03" class="btnType btnType7 tab1">최근 전송내역</button>
+													</div>
+													<div class="btn_popup_wrap">
+														<button type="button" data-tooltip="popup03" class="btnType btnType7 tab2">자주보내는 번호</button>
+													</div>
+													<div class="btn_popup_wrap check_validity_wrap">
+														<button type="button" class="btnType btnType7 check_validity">오류 검사<i class="qmMark error_qm"></i></button>
+														<div class="error_hover_cont send_hover_cont">
+															<p>휴대폰 번호 입력 시 해당 휴대폰 번호에 대한 형식이 어긋나거나 휴대폰 번호에 오류가 있는지 등을 검사하는 기능</p>
+															<span>(예시) 010-1234-0001(O) / 010-12345-0001(X)</span>
+														</div>
+													</div>
+												</div>
+											</div>
+											<div class="list_bottom clearfix">
+												<div class="remove_btnWrap">
+													<button type="button" class="btnType15 all_del"><i class="remove_img"></i>전체삭제</button>
+													<button type="button" class="btnType15 select_del"><i class="remove_img"></i>선택삭제</button>
+												</div>
+												<div class="list_bottom_right">
+													<p>총 <span class="c_e40000" id="rowTotCnt">0</span>건 / 중복 <span class="c_002c9a" id="rowDupCnt">0</span>건</p>
+<!-- 														<button type="button" class="address_reg2">주소록에 등록</button> -->
+												</div>
+											</div>
+										</div>
+									</td>
+								</tr>
+								<tr>
+									<th colspan="2" class="billingAmount">
+										<div>
+											<div class="final_pay">
+												<div class="pay_info_list">
+													<p>발송금액 :</p>
+													<div class="info" id="repPriceTxt" style="display: none;">
+														단문 : <strong>20</strong>건<span>/</span>장문 :
+														<strong>150</strong>건<span>/</span>그림문자 :
+														<strong>30</strong>건
+													</div>
+												</div>
+												<p class="price"><span id="repPriceTxt"></span><span id="totalPriceTxt">0</span> 원<span></span></p>
+											</div>
+											<div class="pay_type clearfix">
+												<div>
+													<input type="radio" id="radio_bill_1" name="radio_bill" checked="checked">
+													<label for="radio_bill_1">보유잔액</label>
+													<label for="userMoney" class="label">보유잔액</label>
+													<fmt:formatNumber type="number" maxFractionDigits="3" value="${sendPrice.userMoney}" var="commaPrice" />
+													<input type="text" id="userMoney" name="userMoney" value="<c:out value='${commaPrice}'/>" readonly="">
+													<span class="won">원</span>
+													<button type="button" class="btnType btnType21" onclick="location.href='/web/member/pay/PayView.do'">충전</button>
+												</div>
+												<div></div>
+											</div>
+										</div>
+									</th>
+								</tr>
+								<tr>
+									<th colspan="2" class="replace_send_th">
+										<div class="title_th"><p>대체문자</p> 
+											<input type="checkbox" id="send_fail_check" name="send_fail_check" class="send_fail_check">
+											<label for="send_fail_check">친구톡 전송 실패 시 문자 전송</label>
+										</div>
+										<div class="replace_send_wrap">
+											<table class="tType1">
+												<colgroup>
+													<col style="width: 90px;">
+													<col style="width: auto;">
+												</colgroup>
+												<tbody>
+													<tr>
+														<th>발신번호</th>
+														<td>
+															<label for="callFromList" class="label"></label>
+															<select id="callFromList" name="callFromList" class="sel_number">
+																<c:choose>
+																	<c:when test="${not empty resultPhonList}">
+																		<c:forEach var="phonList" items="${resultPhonList}" varStatus="status">
+																			<option value="${phonList}">${phonList}</option>
+																		</c:forEach>
+																	</c:when>
+																	<c:otherwise>
+																		<option value="">등록된 발신 번호가 없습니다.</option>
+																	</c:otherwise>
+																</c:choose>
+															</select>
+															<button type="button" class="btnType btnType6" onclick="location.href='/web/user/sendNumberManage.do'">번호등록</button>
+														</td>
+													</tr>
+													<tr>
+														<th>내용</th>
+														<td class="putText">
+															<div class="clearfix">
+																<div class="put_left short">
+																	<!-- 업로드한 이미지의 썸네일 영역 -->
+																	<ul class="thumb_wrap liOnImg ui-sortable"></ul>
+																	<!-- //업로드한 이미지의 썸네일 영역 -->
+																	<label for="smsTxtArea" class="label"></label>
+																	<textarea id="smsTxtArea" name="smsTxtArea" class="put_text"></textarea>
+																	<div class="text_length">
+																		<div name="afterDeny">
+																			<p>
+																				<span class="fwMd" id="msgLeng">0 /</span>
+																				<span class="c_002c9a fwMd" id="limitLeng">90</span>byte
+																			</p>
+																			<span class="msg_com msg_short">단문</span>
+																		</div>
+																	</div>
+																</div>
+																<div class="put_right">
+																	<button type="button" class="btnType btnType9" id="failCheckInit">초기화</button>
+																	<button type="button" class="btnType btnType7" onclick="javascript:fn_errorChk(); return false;">오류검사<i class="qmMark"></i></button>
+																</div>
+															</div>
+														</td>
+													</tr>
+												</tbody>
+											</table>
+										</div>
+									</th>
+								</tr>
+							</tbody>
+						</table>
+					</div>
+					<div class="send_right">
+						<div class="phone" style="top: 201px;">
+							<div class="phoneIn">
+								<p class="prev_p">
+									<img src="/publish/images/content/kakao_prev_icon.png" alt="">
+									<span id="spnYellowid">@채널ID</span>
+								</p>
+								<!-- 텍스트 미리보기 -->
+								<div class="text_preview">
+									<p class="friend_talk_title">
+										<span class="adFlag">(광고)</span>
+										<span id="spnYellowidAt">채널ID</span>
+									</p>
+									<div class="allimtalk_content">
+										<div class="kakao_image"  style="display:none;">
+											<img src="" id="kakaoImg" alt="">
+										</div>
+										<p class="template_text">내용 미리보기</p>
+										<div class="btnViewArea">
+											<c:forEach var="buttonList" items="${resultTemplateVO.buttonVOList}" varStatus="status">
+	                                    		<button type="button" class="btn_kakao_type btnEmpty"  style="display:block;"><c:out value="${buttonList.name}"/></button>
+	                                    	</c:forEach>
+										</div>
+									</div>
+									<p class="kakao_block_text">수신거부 : 홈 > 채널차단</p>
+								</div>
+								<!-- //텍스트 미리보기 -->
+							</div>
+							<p class="addText">※ 단말기 설정에 따라 다르게 보일 수 있습니다<p>
+						</div>
+						<div class="phone_bottom">
+							<div class="send_rev">
+								<div class="send_content" style="padding-bottom: 0;">
+									<div class="rev_radio">
+										<ul>
+											<li>
+												<input type="radio" id="reserYnN" name="reserYn" value="N" checked="checked"><label for="reserYnN">즉시</label>
+											</li>
+											<li>
+												<input type="radio" id="reserYnY" name="reserYn" value="Y"><label for="reserYnY">예약</label>
+											</li>
+										</ul>
+									</div>
+									<div class="send_btn">
+									<%-- 친구톡 발송시간 체크 하기 - 20:50 ~ 익일 08:00 사이에는 발송 금지 --%>
+										<c:choose>
+											<c:when test="${sendStatus eq 'true'}">
+												<button type="button" class="btnType btnType11" onclick="javascript:fn_sendMsgData(); return false;">발송하기</button>
+											</c:when>
+											<c:otherwise>
+												<button type="button" class="btnType btnType11" onclick="javascript:fn_noTimeSend(); return false;">발송하기</button>
+											</c:otherwise>
+										</c:choose>
+										<button type="button" class="btnType btnType10"  onclick="javascript:goToKakaoTestPopUp(); return false;">테스트발송</button>
+									</div>
+								</div>
+								<div class="rev_selected" >
+									<div class="rev_top">
+										<span>날짜 :</span>
+										<div class="calendar_wrap">
+											<input type="text" class="startDate2 inp resDate calendar picker__input picker__input--active" title="검색 시작일" id="startDate2" name="startDate2" value="" data-datecontrol="true" readonly="" aria-haspopup="true" aria-expanded="true" aria-readonly="false" aria-owns="startDate2_root">
+										</div>
+										<div class="selBox">
+											<label for="msgResHour" class="label">시 선택</label>
+											<select class="selType1" id="msgResHour" name="msgResHour">
+												<c:forEach  var="hour"  begin="8" end="20" step="1" varStatus="status">
+													<c:choose>
+														<c:when test="${hour < 10}">
+															<option value="0${hour}">0${hour}시</option>
+														</c:when>
+														<c:otherwise>
+															<option value="${hour}">${hour}시</option>
+														</c:otherwise>
+													</c:choose>
+												</c:forEach>
+											</select>
+											<label for="msgResMin" class="label">분 선택</label>
+											<select class="selType1" id="msgResMin" name="msgResMin" onChange="fnSelectMinChk();">
+												<c:forEach var="min" begin="0" end="59" step="1">
+													<c:choose>
+														<c:when test="${min < 10}">
+															<option value="0${min}">0${min}분</option>
+														</c:when>
+														<c:otherwise>
+															<option value="${min}">${min}분</option>
+														</c:otherwise>
+													</c:choose>
+												</c:forEach>
+											</select>
+										</div>
+									</div>
+									<%-- <div class="rev_bottom">
+										<input type="checkbox" id="inputDivideChk" name="inputDivideChk">
+										<label for="inputDivideChk">분할전송</label>
+										<input type="text" class="dividType1" id="frmDivideCnt" name="frmDivideCnt" value="20" onkeypress='return checkNumber(event)' maxlength="4"/>
+										<label for="divideCnt">건</label>
+										<select class="selType1" id="divideTime" name="divideTime">
+					 						<option value="5">05분</option>
+					 						<option value="10">10분</option>
+					 						<option value="15">15분</option>
+					 						<option value="20">20분</option>
+					 						<option value="30">30분</option>
+					 					</select>
+					 					<label for="divideTime">간격</label>
+									</div> --%>
+								</div>
+							</div>
+						</div>
+					</div>
+				</div>
+			</form>
+		</div>
+	</div>
+</div>
+
+<!-- 문자발송 성공 레이어팝업 -->
+<div class="tooltip-wrap">
+	<div class="popup-com pop_msg_success">
+		<div class="popup_heading">
+			<p>친구톡 전송 결과</p>
+			<button type="button" class="tooltip-close" onclick="msgSuccessClose(this);"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+		</div>
+		<div class="layer_in">
+			<div class="msg_text">발송 성공 : <strong>1</strong> 건,수신거부 : <span>0</span>건의<br>문자가 발송 되었습니다.</div>
+		</div>
+		<div class="popup_btn">
+			<button type="button" onclick="msgResultLink(); return false;">친구톡 발송결과 바로가기</button>
+			<button type="button" class="tooltip-close" onclick="msgSuccessClose(this);">확인</button>
+		</div>
+	</div>
+</div>
+
+<!-- 주소록 불러오기 -->
+<div class="tooltip-wrap">
+	<div class="popup-com import_layer popup06" tabindex="0" data-tooltip-con="popup06" data-focus="popup06" data-focus-prev="popup06-close" style="width: 1000px">
+		<div class="popup_heading">
+			<p><span>주소록 불러오기</p>
+			<button type="button" onClick="javascript:addrClose(); return false;">
+			<img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+		</div>
+		<div class="layer_in">
+			<div class="titBox titBox_pad">
+				<p>- 주소록 수정 및 변경은 <span>[주소록 관리]</span>에서만 가능합니다.</p>
+				<button type="button" class="adr_admin" onClick="location.href='/web/mjon/addr/selectAddrList.do'">주소록 관리</button>
+			</div>
+			<div class="adr_wrap">
+				<form id="searchAddrGrpForm" name="searchAddrGrpForm" method="post" style="display: flex; justify-content: space-between;">
+					<input type="hidden" id="searchAddrGrpId" name="searchAddrGrpId" value=""/>
+					<input type="hidden" id="type" name="type" value="all"/>
+					<input type="hidden" id="searchKeyword" name="searchKeyword" value=""/>
+					<input type="hidden" name="searchCondition" id="searchCondition" value="0" />
+					<div class="adr_pop_left">
+						<div class="adr_left_search">
+							<label for="searchGrpKeyword" class="label">그룹명 검색</label>
+							<input type="text" name="searchGrpKeyword" id="searchGrpKeyword"  placeholder="그룹명 검색" onfocus="this.placeholder=''" onblur="this.placeholder='그룹명 검색'" class="inputLight">
+							<button type="button" onClick="javascrit:fnAddrGrpSearch(); return false;"><img src="/publish/images/popup/search.png" alt="검색"></button>
+						</div>
+						<div class="adr_pop_box">
+							<div id="addrGroupLoad">
+							</div>
+						</div>
+						<!-- <div class="popup_btn">
+							<button type="button" class="btnType" onClick="javascript:fnSelectAddrGrpList(); return false;">선택 그룹 추가</button>
+						</div> -->
+					</div>
+					<div class="adr_pop_right">
+						<div class="clearfix">
+							<div class="btnWrap_last">
+								<label for="searchAddrCondition" class="label">카테고리 선택</label>
+								<select id="searchAddrCondition" name="searchAddrCondition" class="selType2">
+									<option value='0'>전체</option>
+									<option value='1'>그룹명</option>
+									<option value='2'>이름</option>
+									<option value='3'>핸드폰번호</option>
+								</select>
+								<label for="searchAddrKeyword" class="label">검색어 입력</label>
+								<input type="text" id="searchAddrKeyword" name="searchAddrKeyword" placeholder="검색어를 입력하세요"  onfocus="this.placeholder=''" onblur="this.placeholder='검색어를 입력하세요'" >
+								<button type="button" class="btnType btnType17"  onClick="javascrit:fnAddrSearch(); return false;">검색</button>
+							</div>
+							<!-- table -->
+							<div class="adr_excel adr_pop_list2 callAddr_box">
+							</div>
+							<!--// table -->
+						</div>
+						<div class="popup_btn_wrap2">
+							<button type="button" onClick="javascript:addrToList(); return false;">추가</button>
+							<button type="button" onClick="javascript:addrClose(); return false;">닫기</button>
+						</div>
+						<%-- 주소록 레이어 팝업 닫기 실행 코드 --%>
+						<input type="hidden" name="btnAddrClose" id="btnAddrClose" class="tooltip-close closeAddr" data-focus="popup06-close" />         
+					</div>
+				</form>
+			</div>
+		</div>
+	</div>
+</div>
+<!--// 주소록 불러오기 -->
+
+<!-- 엑셀 불러오기 -->
+<form id="excelToolTipForm" name="excelToolTipForm" method="post">
+	<div class="tooltip-wrap">
+		<div class="popup-com import_layer popup02" tabindex="0" data-tooltip-con="popup02" data-focus="popup02" data-focus-prev="popup02-close">
+			<div class="popup_heading">
+				<p><span>엑셀</span> 불러오기</p>
+				<button type="button" class="tooltip-close" data-focus="popup02-close"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+			</div>
+			<div class="layer_in">
+				<!-- 엑셀파일 불러오기 -->
+				<div class="hascont">
+					<div class="titBox">
+						<p>- 최대 2만 건까지 등록할 수 있습니다.</p>
+						<p>- [엑셀 불러오기]시 문서의 A, B열을 불러옵니다.(지원하는 파일 형식 : xls, xlsx)</p>
+						<p>- 휴대폰 항목은 숫자, 하이픈(-)만 인식하며, 번호 앞에 0이 생략되어도 정상 등록됩니다.
+						</p>
+						<!-- <button type="button" class="excel_btn" onclick="location.href='/cmm/fms/FileDown.do?atchFileId=FILE_000000000011651&fileSn=1'"><i></i>샘플파일 다운로드</button> -->
+						<button type="button" class="excel_btn" onclick="location.href='/download/msg/엑셀주소록_등록양식.xlsx'"><i class="downroad"></i>샘플파일 다운로드</button>
+					</div>
+					<div class="attachedFile">
+						<label for="excelNm" class="attachedFile_label">첨부파일</label>
+						<input type="text" id="excelNm" value="" readonly>
+						<input type="file" id="excelFile" accept=".xls, .xlsx" onchange="excelExport(event); return false;" style="display:none"/>
+						<button type="button" class="btnType btnType6 c1">찾아보기</button>
+<!-- 						<p><span class="vMiddle">*</span> 첨부된 파일은 <span class="c_e40000">[추가]버튼을 클릭</span>하셔야 받는 사람에 등록됩니다.</p> -->
+						<p><span class="vMiddle">*</span> 첨부된 파일은 <span class="c_e40000">[추가]버튼을 클릭</span>하셔야 받는 사람에 등록됩니다.</p>
+					</div>
+				</div><!--// 엑셀파일 불러오기 -->
+				<div class="popup_btn_wrap2">
+					<button type="button" class="tooltip-close" data-focus="popup02-close"  data-focus-next="popup02" id="excelAdd">추가</button>
+					<button type="button" class="tooltip-close" data-focus="popup02-close"  data-focus-next="popup02">닫기</button>                      
+				</div>
+			</div>
+		</div>
+	</div><!--// 엑셀 불러오기 -->
+</form>
+
+<form id="templateForm" name="templateForm" method="post">
+	<input type="hidden" id="friendId" name="friendId" value="<c:out value='${resultTemplateVO.friendId}'/>"/>
+</form>(No newline at end of file)
 
src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/include/ftDataIncludeExcel.jsp (added)
+++ src/main/webapp/WEB-INF/jsp/web/kakao/msgdata/include/ftDataIncludeExcel.jsp
@@ -0,0 +1,959 @@
+<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
+<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
+<%@ taglib prefix="ui" uri="http://egovframework.gov/ctl/ui"%>
+<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %>
+<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
+
+<script type="text/javascript" src="<c:url value='/publish/js/content.js'/>"></script>
+
+<script type="text/javascript">
+
+var $tableExcel = null; //엑셀입력 탭
+var $tableError = null; //엑셀입력 탭
+$(document).ready(function(){
+	
+	//Tabulator AJAX Data Loading
+	$tableError = new Tabulator("#tabulator_error", {
+		height:"255px",
+		width:"100%",
+	    layout:"fitColumns",
+	    autoColumns:false,
+	    headerHozAlign:"center", 
+	    validationMode:"highlight",
+	    clipboard:false,
+	    clipboardCopySelector:"table",
+	    clipboardPasteAction:"insert", // insert, update, replace
+	    placeholder:"등록 팝업에서 휴대폰을 선택 후 확인해주세요.", //fit columns to width of table (optional)
+	 	columns:[ //Define Table Columns
+		 	{title:"이름", field:"name", hozAlign:"center", headerHozAlign: "center", width:125},
+		 	{title:"휴대폰", field:"phone", hozAlign:"center", headerHozAlign: "center"},
+		 	{title:"미등록 결과", field:"result", hozAlign:"center", headerHozAlign: "center"}
+	 	]
+	});
+	
+
+    //Tabulator AJAX Data Loading
+    $tableExcel = new Tabulator("#tabulator_excel", {
+        height:"255px",
+        width:"100%",
+        layout:"fitColumns",
+        autoColumns:false,
+        headerHozAlign:"center", 
+        validationMode:"highlight",
+        clipboard:false,
+        clipboardCopySelector:"table",
+        clipboardPasteAction:"insert", // insert, update, replace
+        placeholder:"Excel 파일을 업로드 해주세요.", //fit columns to width of table (optional)
+        columns:[ //Define Table Columns
+            {formatter:"rowSelection", titleFormatter:"rowSelection",clipboard:false, headerHozAlign:"center", hozAlign:"center", headerSort:false, cellClick:function(e, cell){
+                cell.getRow().toggleSelect();
+                }
+            }, 
+            {formatter:"rownum", align:"center" ,title:"No", hozAlign:"center", headerHozAlign:"center", width:50},
+            {title:"A", field:"A", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]},
+            {title:"B", field:"B", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]},
+            {title:"C", field:"C", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]},
+            {title:"D", field:"D", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]},
+            {title:"E", field:"E", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]},
+            {title:"F", field:"F", hozAlign:"center", headerHozAlign: "center", width:140, validator:["maxLength:100", "string"]}
+        ],
+        validationFailed:function(cell, value, parameters){ // 유효성 체크 함수 
+            var valid = cell.isValid();
+            if(!valid){
+                alert("양식에 맞지 않는 정보가 입력되었습니다.");
+                
+                //해당 셀 데이터 삭제
+                cell.setValue("");
+            }
+            return value % parameters.phone;
+        },
+    });
+	
+	
+    
+	
+	// 타뷸레이터 width값 변경 시 위에 select width 값 변경
+	var titleArray = ["No","A","B","C","D","E","F"];
+
+	$tableExcel.on("columnWidth",function(column){
+		var titleIndex = titleArray.indexOf(column._column.definition.title);
+		titleIndex += 1;
+		
+		if(titleIndex != 0){
+			$('.select_adr_hd>div').eq(titleIndex).css('width', column._column.width);
+		}else{
+			$('.select_adr_hd>div').eq(0).css('width', column._column.width);
+			
+		}
+	});
+
+	$tableExcel.on("scrollHorizontal",function(left){
+	      $(".adr_excel").scrollLeft(left);
+	})
+
+
+	$(".adr_excel").on("scroll",function(){
+		$(".tabulator-tableholder").scrollLeft($(this).scrollLeft());
+	});
+
+	
+	
+
+    $("#excelFileC4").on("change", function(event) {
+    	var fileInfo =  event.target.files;
+        if(fileInfo.length > 0){
+            excelFileChange(fileInfo[0]);
+        } else {
+            fn_loadRemoveActive(); // 파일이 선택되지 않은 경우 로딩 상태 제거
+            setTimeout(() => { $(this).val(''); }, 0);  // 파일 선택 초기화
+        }
+    });
+	
+
+
+    $(document).on('click', '#btnAddrMassClose', function() {
+
+    	$('.field-selector').each(function() { $(this).val(''); });
+    	setAddrMassClose();
+    });
+    
+
+    $(document).on('click', '#closeBtn', function() {
+    	// 대량등록 닫기
+    	setAddrMassClose();
+    });
+    
+
+    
+
+ // 엑셀등록 닫기
+	function setAddrMassClose() {
+		$tableExcel.clearData();
+		$("#excelRowTotCnt").text(0); //총건수 수정
+		$("#excelRowDupCnt").text(0); //중복건수 수정
+		$("#excelRowErrorCnt").text(0); //중복건수 수정
+// 		dupliPhoneDataRealList.length = 0;	// 중복 휴대폰번호 초기화
+		
+		
+		// 중복 카운트
+		$("#errorPopDupCnt").text(0);
+		// 에러 카운트
+		$("#errorPopErrorCnt").text(0);
+		// 
+		$("#errorPopTotCnt").text(0);
+
+        // popup 영역
+        $tableError.clearData();
+		
+	}
+    
+  //#############################################################################################
+  //파일업로드 드래그앤 드롭
+  //#############################################################################################
+  var objDragAndDrop = $(".upload_area");
+  $(document).on("dragenter",".upload_area",function(e){
+    e.stopPropagation();
+    e.preventDefault();
+    //$(this).css('border', '2px solid #0B85A1');
+  });
+  $(document).on("dragover",".upload_area",function(e){
+    e.stopPropagation();
+    e.preventDefault();
+  });
+  $(document).on("drop",".upload_area",function(e){
+  	fn_loadAddActive();
+  	e.preventDefault();
+  	var files = e.originalEvent.dataTransfer.files;
+  	excelFileChange(files[0]);
+  });
+
+  $(document).on('dragenter', function (e){
+    e.stopPropagation();
+    e.preventDefault();
+  });
+  $(document).on('dragover', function (e){
+  	e.stopPropagation();
+  	e.preventDefault();
+  //objDragAndDrop.css('border', '2px dotted #0B85A1');
+  });
+  $(document).on('drop', function (e){
+    e.stopPropagation();
+    e.preventDefault();
+  });
+  //파일 드래그앤드롭 종료
+	
+	
+	
+	
+  
+
+	// 받는사람 선택삭제 버튼 처리해주기
+	$('#in_select_del').click(function(){
+		
+		if($tableExcel == null || $tableExcel == ""){
+			
+			alert("받는사람을 추가해 주세요.");
+			return false;
+		
+		}
+		
+		var selectedData = $tableExcel.getSelectedRows();
+		
+		if(selectedData == "" || selectedData == null){
+			
+			alert("삭제할 연락처를 선택해주세요.");
+			return false;
+		
+		}else{ // 선택한 Row 데이터 삭제하기
+			
+			if(confirm("선택하신 받는 사람을 삭제하시겠습니까?")){
+				
+				// 선택 데이터 삭제
+			    selectedData.forEach(row => row.delete());
+			    
+			    
+				totRows = $tableExcel.getRows().length;
+			    $("#excelRowTotCnt").text(totRows);
+			    
+				
+			}
+
+		}
+		
+	});
+
+	// 추가버튼
+	$('#btnAddrMassReg').click(function(){
+		
+		if($tableExcel.getData().length < 1){
+			alert("한 개 이상의 연락처를 입력하세요");
+			return false;
+		}
+//	 	else if (selectedData.length > 20000) {
+//	 		alert("2만줄 이상의 업로드는 데이터 부하로 업로드 할수 없습니다.");
+//	 		return false;
+//	 	}
+		
+		
+		// tableExcel 그룹의 select 요소들을 확인
+		var columns = $tableExcel.getColumns();
+		var isAddrPhoneNoSelected = columns.some(column => column.getField() === 'addrPhoneNo');
+
+		if (!isAddrPhoneNoSelected) {
+			alert('휴대폰이 선택되지 않았습니다.');
+			return false;
+
+		} 
+
+		var addrData = $tableExcel.getData().map((row, index) => ({
+		    name: row.addrNm,
+		    phone: removeDash(row.addrPhoneNo),
+		    rep1: row.addrInfo1,
+		    rep2: row.addrInfo2,
+		    rep3: row.addrInfo3,
+		    rep4: row.addrInfo4,
+		}));
+
+
+		
+		// 기존 tableL의 데이터를 가져옵니다.
+		var existingData = tableL.getData();
+		// 기존 데이터와 새로운 데이터를 합칩니다.
+		var combinedData = existingData.concat(addrData);
+
+
+		/** 
+		* @ phone을 기준으로 중복 제거 및 갯수 계산
+		* @ 결과 반환
+		* 	return {
+		* 		uniqueArray,       // 중복 제거된 배열                    
+		* 		uniqueCount, // 중복 제거된 데이터 개수  
+		* 		duplicateArray,    // 중복된 데이터 배열                   
+		* 		duplicateCount: duplicateArray.length // 중복된 데이터 개수
+		* 	};
+		*/
+		const result = removeDuplicatesAndCount(combinedData, 'phone');
+
+		
+		// 총 30만건이 넘으면 false
+		if (!validateRowLimit(result.uniqueCount)) {
+			return false;
+		}
+		
+
+        // 6. 수량/가격 계산
+        setAllCntData(result);
+        // fn_priceClclt(result.uniqueCount);
+		//결제 금액 구하기
+	    totalFtPriceSum(tableL.getRows().length);
+		
+		
+		$('#closeBtn').click();
+	});
+		
+  
+
+	//받는사람 전체삭제 버튼 처리
+	$('#allDel').click(function(){
+		var data = $tableExcel.getRows();	
+		$tableExcel.clearData();
+		$("#excelRowTotCnt").text(0); //총건수 수정
+		$("#excelRowDupCnt").text(0); //중복건수 수정
+		$("#excelRowErrorCnt").text(0); //중복건수 수정
+		dupliPhoneDataRealList.length = 0;	// 중복 휴대폰번호 초기화
+        $tableError.clearData();
+		
+		// select box 초기화
+		$('.field-selector').each(function() { $(this).val(''); });
+		
+	});
+	
+  
+
+    //치환문자 있는 엑섹불러오기 버튼 클릭시 파일 첨부 실행
+    $('.c3').click(function(){ // 엑셀파일 불러오기 선택 시
+        
+        $("#excelFileC4").click();
+        
+    });
+    
+    
+
+    //타이틀 select 선택 이벤트
+     $('.field-selector').on('change', function() {
+        fn_loadAddActive();
+
+        setTimeout(() => { 
+            var selectedFields = [];
+            var isDuplicate = false;
+    
+            if($tableExcel.getData().length < 1){
+                alert('데이터 입력 후 선택해 주세요.');
+                $(this).val(""); 
+                fn_loadRemoveActive();
+                return false;
+            }
+            
+            // 중복체크
+            $('.field-selector').each(function() {
+                var selectedField = $(this).val();
+                if (selectedField) {
+                    if (selectedFields.includes(selectedField)) {
+                        alert("중복된 필드를 선택할 수 없습니다.");
+                        $(this).val(""); // 중복 필드를 선택한 경우 빈 값으로 초기화
+                        isDuplicate = true;
+                        return false; // 반복문 종료
+                    }
+                    selectedFields.push(selectedField);
+                }
+            });
+    
+    
+            // 
+            updateTableFields($tableExcel);
+            
+            // 필드가 휴대폰이면 열 중복체크
+            if($(this).val() == 'addrPhoneNo'){
+                fn_phoneDupl($tableExcel);
+            }
+            fn_loadRemoveActive();
+
+        }, 0); // 지연 없이 즉시 실행되도록 0ms 지연을 설정
+        
+        
+    });
+    
+    
+    
+});
+
+
+
+function excelFileChange(file) {
+	if (file) {
+
+		// 파일 크기 체크 (20MB)
+		const maxSize = 20 * 1024 * 1024; // 20MB in bytes
+		if (file.size > maxSize) {
+			alert('파일 크기는 20MB를 초과할 수 없습니다.');
+			return;
+		}
+		
+		fn_loadAddActive();
+		var reader = new FileReader();
+		var extension = file.name.split('.').pop().toLowerCase();
+		reader.onload = function(e) {
+			setTimeout(() => { // 파일 읽기 완료 후 실행되도록 함
+				if (extension === 'xlsx') {
+					var data = new Uint8Array(e.target.result);
+					var workbook = XLSX.read(data, {type: 'array'});
+					var firstSheet = workbook.Sheets[workbook.SheetNames[0]];
+					var jsonData = XLSX.utils.sheet_to_json(firstSheet, {header: 1});
+					// 문제 데이터를 확인하는 함수 호출
+					findInvalidDBCharacters(jsonData);
+					processExcelData(jsonData);
+				} else if (extension === 'xls') {
+					var data = e.target.result;
+					var workbook = XLSX.read(data, { type: 'binary' });
+					var firstSheet = workbook.Sheets[workbook.SheetNames[0]];
+					var jsonData = XLSX.utils.sheet_to_json(firstSheet, { header: 1 });
+
+					// 문제 데이터를 확인하는 함수 호출
+					findInvalidDBCharacters(jsonData);
+
+					
+					processExcelData(jsonData);
+				} else if (extension === 'txt') {
+					var textData = e.target.result;
+					processTextData(textData);
+				} else {
+					alert('지원되지 않는 파일 형식입니다.');
+				}
+				fn_loadRemoveActive();
+			}, 0); // 지연 없이 즉시 실행되도록 0ms 지연을 설정
+		};
+		if (extension === 'xlsx') {
+			reader.readAsArrayBuffer(file);
+		} else if (extension === 'xls') {
+			reader.readAsBinaryString(file); // xls 파일에 적절한 read 메서드 호출
+		} else if (extension === 'txt') {
+			reader.readAsText(file);
+		}
+	}
+}
+
+//문제 데이터를 확인하는 함수
+function findInvalidDBCharacters(jsonData) {
+	console.log('DB 입력 값 검사 중...');
+	const invalidCharPattern = /[\uD800-\uDBFF][\uDC00-\uDFFF]/; // 4바이트 유니코드 문자 (이모지 등)
+	for (let rowIndex = 0; rowIndex < jsonData.length; rowIndex++) {
+		const row = jsonData[rowIndex];
+		if (Array.isArray(row)) {
+			for (let colIndex = 0; colIndex < row.length; colIndex++) {
+				const cell = row[colIndex];
+				if (typeof cell === 'string' && invalidCharPattern.test(cell)) {
+					console.warn('허용되지 않는 문자: row', rowIndex + 1,', col ', colIndex + 1, ', value:', cell);
+					// 허용되지 않는 문자를 제거 (선택 사항)
+					row[colIndex] = cell.replace(invalidCharPattern, '');
+					console.log('수정된 값:', row[colIndex]);
+				}
+			}
+		}
+	}
+}
+
+
+//엑셀 데이터 처리 함수
+function processExcelData(data) {
+	var keys = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];
+	var tableData = [];
+	var totalRows = data.length - 2; // 전체 데이터 수 (1, 2행 제외)
+	
+	
+	// 3번째 행부터 입력 
+	data.slice(0).forEach((row, index) => {
+	  var rowData = {};
+	  keys.forEach((key, idx) => { // index 변수명 변경 (내부와 외부에서 사용되므로 충돌 방지)
+	//   console.log('row[idx] : ', row[idx]);
+	//       rowData[key] = row[idx] ? row[idx].trim() : ""; // 각 컬럼에 대해 기본값을 설정
+	      rowData[key] = (typeof row[idx] === 'string') ? row[idx].trim() : row[idx];
+	  });
+	  tableData.push(rowData);
+	
+	});
+	
+	updateTable(tableData);
+}
+
+
+
+/* function processExcelData(data) {
+	console.log('data : ', data);
+	var keys = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];
+	var tableData = [];
+	var totalRows = data.length - 2; // 전체 데이터 수 (1, 2행 제외)
+	console.log('data: ', data);
+	
+	// 3번째 행부터 입력 
+	data.slice(0).forEach((row, index) => {
+		var rowData = {};
+		keys.forEach((key, idx) => { // index 변수명 변경 (내부와 외부에서 사용되므로 충돌 방지)
+// 			console.log('row[idx] : ', row[idx]);
+			   rowData[key] = row[idx] ? row[idx].trim() : ""; // 각 컬럼에 대해 기본값을 설정
+			rowData[key] = (typeof row[idx] === 'string') ? row[idx].trim() : row[idx];
+		});
+		tableData.push(rowData);
+	
+	});
+	console.log('tableData :: ', tableData);
+	updateTable(tableData);
+} */
+
+
+// 텍스트 데이터 처리 함수
+function processTextData(text) {
+	var lines = text.split('\n'); // 각 줄을 배열로 분리
+	var keys = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];
+	var tableData = [];
+	
+	lines.forEach(line => {
+		var rowData = {};
+		var row = line.split(','); // 쉼표로 분리
+		keys.forEach((key, index) => {
+			rowData[key] = row[index] ? row[index].trim() : ""; // 각 컬럼에 대해 기본값을 설정
+		});
+		tableData.push(rowData);
+	});
+	
+	updateTable(tableData);
+}
+
+//공통 테이블 업데이트 함수
+function updateTable(tableData) {
+	$tableExcel.setColumns([ //Define Table Columns
+ 		{formatter:"rowSelection", titleFormatter:"rowSelection",clipboard:false, headerHozAlign:"center", hozAlign:"center", headerSort:false, cellClick:function(e, cell){
+ 	        cell.getRow().toggleSelect();
+	 		}
+	 	}, 
+		{formatter:"rownum", align:"center" ,title:"No", hozAlign:"center", headerHozAlign:"center", width:10},
+	 	{title:"A", field:"A", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]},
+	 	{title:"B", field:"B", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]},
+	 	{title:"C", field:"C", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]},
+	 	{title:"D", field:"D", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]},
+	 	{title:"E", field:"E", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]},
+	 	{title:"F", field:"F", hozAlign:"center", headerHozAlign: "center", width:150, validator:["maxLength:100", "string"]}
+ 	]);
+
+	$tableExcel.setData(tableData).then(() => {
+		// excelRowTotCnt 업데이트
+		document.getElementById("excelRowTotCnt").innerText = tableData.length;
+	});
+	
+	fn_loadRemoveActive();
+}
+
+
+
+/* 
+* 타이틀 select 선택할때마다 실행해서         
+* 데이터테이블 필드값 수정                  
+*/
+function updateTableFields($objTabul) {
+	var currentData = $objTabul.getData();
+	var columns = [
+		{formatter: "rowSelection", titleFormatter: "rowSelection", clipboard: false, hozAlign: "center", headerHozAlign: "center", headerSort: false, cellClick: function(e, cell) {
+			cell.getRow().toggleSelect();
+		}}
+		,{formatter:"rownum", align:"center", title:"No", hozAlign:"center", headerHozAlign:"center", width:60}
+	];
+
+	var fieldMapping = [];
+	$('.field-selector').each(function(index) {
+		var selectedField = $(this).val();
+		//  ASCII 문자 코드 사용 - 65=A, 66=B ...
+		var field = String.fromCharCode(65 + index);
+		if (selectedField) {
+			columns.push({
+				title: field
+				, field: selectedField
+				, hozAlign: "center"
+				, headerHozAlign: "center"
+// 				, editor: "input"
+				, editor: false
+				, width: 140
+				, validator: ["maxLength:100", "string"]
+			});
+			fieldMapping.push(selectedField);
+		} else {
+			columns.push({
+				title: field
+				, field: field
+				, hozAlign: "center"
+				, headerHozAlign: "center"
+				, editor: false
+// 				, editor: "input"
+				, width: 140
+				, validator: ["maxLength:100", "string"]
+			});
+			fieldMapping.push(field);
+		}
+	});
+
+	var updatedData = currentData.map(row => {
+		console.log('row : ', row);
+		var newRow = {};
+		fieldMapping.forEach((field, index) => {
+			newRow[field] = row[Object.keys(row)[index]] ?? "";
+		});
+		return newRow;
+	});
+
+	$objTabul.setColumns(columns);
+	$objTabul.setData(updatedData);
+}
+
+
+/**
+ * @ 핸드폰 중복 데이터
+ * */
+function fn_phoneDupl($objTabul) {
+     
+    $tableError.clearData();
+    
+    var data = $objTabul.getData();
+    var phoneNumberChk = false;
+    var existingNumbers = new Set(); // 배열에서 Set으로 변경
+    
+    let errorCount = 0; // 중복 번호 개수를 저장할 변수
+    let duplicateCount = 0; // 중복 번호 개수를 저장할 변수
+    
+    const errors = [];  // 오류 데이터를 저장할 배열
+    const newData = []; // 유효한 데이터만 저장할 새로운 배열
+    
+    data.forEach((row, index) => {
+        
+        const number = row.addrPhoneNo;
+        
+        // number가 null, undefined, 빈 문자열이거나 숫자인 경우 처리
+        if (!number || (typeof number === 'string' && !number.trim())){
+            console.log("number : ", number);
+             return;
+        }
+        
+        const formattedNumber = formatPhoneNumber(number); // 번호 표준화
+        const cleanedNumber = formattedNumber.replace(/[^0-9]/g, ''); // 숫자만 남김
+        
+        if (!existingNumbers.has(cleanedNumber)) { // 중복 번호 체크
+            if (isValidPhoneNumber(formattedNumber)) { // 유효성 검사
+                row.addrPhoneNo = formattedNumber;
+                existingNumbers.add(cleanedNumber); // 추가된 번호를 기존 목록에 추가
+                newData.push(row); // 유효한 데이터만 새로운 배열에 추가
+            } else {
+                // 오류: 유효성 통과 못함
+                errorCount++;
+                
+                errors.push({ 
+                    name: row.addrNm, // 이름
+                    phone: row.addrPhoneNo, // 폰번호
+                    result: "오류" // 결과 메시지 추가
+                });
+            }
+        } else {
+            // 중복
+            duplicateCount++;
+            
+            errors.push({ 
+                name: row.addrNm, // 이름
+                phone: row.addrPhoneNo, // 폰번호
+                result: "중복" // 결과 메시지 추가
+            });
+        }
+    });
+    
+    // data 배열을 newData 배열로 대체
+    data = newData;
+    
+
+    // 수정된 데이터로 테이블 업데이트
+    $objTabul.setData(data);
+    // 오류 총 카운트
+    $("#excelRowTotCnt").text($objTabul.getDataCount());
+    // 중복 카운트
+    $("#excelRowDupCnt").text(duplicateCount);
+    // 에러 카운트
+    $("#excelRowErrorCnt").text(errorCount);
+    
+    // popup 영역
+    $("#errorPopTotCnt").text($objTabul.getDataCount());
+    // 중복 카운트
+    $("#errorPopDupCnt").text(duplicateCount);
+    // 에러 카운트
+    $("#errorPopErrorCnt").text(errorCount);
+    
+    
+    $tableError.setData(errors);
+    
+    if(errorCount > 0){
+        alert('휴대폰 형식에 맞지 않는 데이터는 삭제 후 업로드 됩니다.\nex) 발송불가 특수문자, 자릿수 오류 등');
+    }
+    
+    
+    
+}
+
+function fn_dupliPopupShow(){
+
+	$("#tableExcelDupliBtn").show();
+}
+
+
+// 상단 설명 더보기 
+function popMore(e){
+	$(e).closest(".pop_more_cont").toggleClass("pop_more_click");
+	
+	if($(e).closest(".pop_more_cont").is(".pop_more_click")){
+		$(e).html('숨기기');
+		$(e).append('<i></i>');
+	}else {
+		$(e).html('더보기');
+		$(e).append('<i></i>');
+	}
+}
+
+// excel 오류정보 테스트
+$(document).on('click', '#errorExcelBtn', function() {
+	if($tableError.getDataCount()<1){
+		alert('오류 정보가 없습니다.');
+		return false;
+	}
+	$tableError.download("xlsx", "error_data.xlsx");
+});
+
+
+</script>
+
+<!-- 중복전화번호 data-tooltip:addrMassDupli_layer -->
+<div class="tooltip-wrap">
+	<div class="popup-com addrMassDupli_layer" tabindex="0" data-tooltip-con="addrMassDupli_layer" data-focus="addrMassDupli_layer" data-focus-prev="addrMassDupli_layer-close" style="width: 270px; height: 500px;">
+		<div class="popup_heading">
+			<p>중복 휴대폰번호</p>
+			<button type="button" class="tooltip-close" data-focus="addrMassDupli_layer-close" onclick="setAddrDupliClose();"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+		</div>
+		<div class="layer_in" style="padding:20px 0px;" id="addrMassDupli_layer">
+		</div>
+		
+		<div class="popup_btn_wrap2" style="margin-top: 0px;">
+			<button type="button" class="tooltip-close" data-focus="addrMassDupli_layer-close" data-focus-next="addrMassDupli_layer">닫기</button>
+		</div>		
+		
+	</div>		
+</div>
+
+
+<!-- 주소록 상세 결과 팝업 data-tooltip:adr_popup14 -->
+	<div class="tooltip-wrap">
+		<div class="popup-com adr_layer adr_detail_result adr_popup14" tabindex="0" data-tooltip-con="adr_popup14" data-focus="adr_popup14" data-focus-prev="adr_popu14-close" style="width: 525px;z-index:125;">
+			<div class="popup_heading">
+				<p>주소록 상세 결과</p>
+				<button type="button" class="tooltip-close" data-focus="adr_popup14-close"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+			</div>
+			<div class="layer_in" style="padding:30px 20px;">
+				<div class="table_top">
+					<p>
+						총 <span class="c_e40000" id="errorPopTotCnt">0</span>건 
+						/ 중복 <span class="c_002c9a" id="errorPopDupCnt">0</span>건
+						/ 오류 <span class="c_002c9a" id="errorPopErrorCnt">0</span>건</p>
+						<button type="button" class="excel_btn btnType" id="errorExcelBtn"><i class="downroad"></i>엑셀 다운로드</button>
+				</div>
+				<div class="tb_wrap adr_list" id="tabulator_error">
+				<!-- $tableError 참고  -->
+                </div>
+                <ul class="cf_text_ul">
+                    <li>*중복번호는 하나의 번호만 등록됩니다.</li>
+                    <li>*휴대폰 형식에 맞지 않는 데이터는 삭제 후 업로드 됩니다.</li>
+                    <li>ex) 발송불가 특수문자, 자릿수 오류 등</li>
+                </ul>
+                <div class="popup_btn_wrap2">
+<!--                     <button type="button">저장</button> -->
+                    <button type="button" class="tooltip-close" data-focus="adr_popup14-close" data-focus-next="adr_popup14">닫기</button>                      
+                </div>
+			</div>
+		</div>
+	</div>
+
+<!--// 중복전화번호 팝업 -->
+				<div class="popup_heading">
+					<p>엑셀 불러오기</p>
+					<button type="button" class="tooltip-close" id="closeBtn" data-focus="popup07-close"><img src="/publish/images/content/layerPopup_close.png" alt="팝업 닫기"></button>
+			   </div>
+				<div class="layer_in execl_upload_layer" style="padding: 25px 30px;">
+<!-- 					<div class="list_tab_wrap2"> -->
+						<!-- tab button -->
+<!-- 						<ul class="list_tab" id="tbTabl">  -->
+<!-- 							<li class="tab active" data-tabul="tableExcel"><button type="button" onclick="popupTab(this,'1'); fn_tabToggle('1');">엑셀입력</button></li> -->
+<!-- 							<li class="tab" data-tabul="tableClip"><button type="button" onclick="popupTab(this,'2'); fn_tabToggle('2');">붙여넣기</button></li> -->
+<!-- 							<li class="tab" data-tabul="tableSelf"><button type="button" onclick="popupTab(this,'3'); fn_tabToggle('3');">직접입력</button></li> -->
+<!-- 						</ul>// tab button -->
+<!-- 					</div> -->
+					<!-- 엑셀입력 -->
+					<div class="popCont current pop_more_cont" id="popCont_1">
+						<div class="titBox">
+							<p>- 주소록은 한 번에 최대 30만건까지 등록(EXCEL파일, 최대용량 20MB) 가능합니다. </p>
+							<p>- 엑셀 파일에 비밀번호 설정, 제한된 보기, 수식 등이 설정되어 있는 경우 업로드가 불가합니다.</p>
+							<p>- 구분선(|), 역슬래시(\, ₩), 큰따옴표("), 이모지(이모티콘) 등 발송불가 특수문자는 저장되지 않습니다.</p>
+							<p>- 이름 200byte, [*1*]~[*4*] 200byte, 메모 250byte까지 입력 가능합니다.</p>
+							<p>- 주소록 등록이 어려우신 경우에는 <a href="<c:url value='/web/mjon/addragency/selectAddrAgencyList.do'/>" style="font-weight: bold; color: blue;">주소록 입력대행</a> 메뉴를 이용하실 수 있습니다. </p>
+						</div>
+						<div class="pop_more_wrap">
+								<button type="button" class="pop_more" onclick="popMore(this);">더보기<i></i></button>
+						</div>
+					</div><!--// 엑셀입력 -->
+
+					<!-- 공통 -->
+					<div>
+		                <table class="layer_tType1">
+		                    <caption>엑셀입력 표</caption>
+		                    <colgroup>
+		                        <col style="width: 95px">
+		                        <col style="width: auto">
+		                    </colgroup>
+		                    <tbody>
+		                        <tr>
+		                            <!-- <th>그룹 선택</th>
+		                            <td>
+		                                <label for="" class="label">그룹 선택</label>
+		                                <select id="addrGrpIdInfo" name="addrGrpIdInfo">
+		                                </select>
+	                                    <label for="" class="label">그룹명 입력</label>
+	                                    <input type="text" id="addrGrpNm" name="addrGrpNm" placeholder="새 그룹명을 입력해주세요." onfocus="this.placeholder=''" onblur="this.placeholder='새 그룹명을 입력해주세요.'"class="inputLight" style="width: 300px;">
+		                                <input type="file" id="excelFile" accept=".xls, .xlsx, .txt" style="display:none"/>
+		                                <button type="button" class="excel_btn2 btnType c3"><i class="uproad"></i>엑셀, TXT파일 업로드</button>
+		                            </td> -->
+		                            <td colspan="2" style="padding:10px 0;">
+		                            	<div class="file_upload_wrap" style="width:100%;display:flex;">
+						                	<div class="file_add upload_area">
+												<p><img src="/publish/images/content/file_add.png" alt="파일 붙여넣기">마우스로 엑셀파일을 여기에 끌어다 놓으세요</p>
+											</div>
+											<input type="file" id="excelFileC4" accept=".xls, .xlsx, .txt" style="display:none"/>
+					                        <button type="button" class="excel_btn2 btnType c3"><i class="uproad"></i>엑셀파일 업로드</button>
+						                </div>
+		                            </td>
+		                        </tr>
+		                    </tbody>
+		                </table>
+	                </div>
+	                
+					
+					<div class="excel_middle2">
+						<p>
+							총 <span class="c_e40000 fwBold" id="excelRowTotCnt">0</span>건 
+							/ 중복 <span class="c_002c9a fwBold" id="excelRowDupCnt">0</span>건
+							/ 오류 <span class="c_002c9a fwBold" id="excelRowErrorCnt">0</span>건
+							<button type="button" class="btn_list_detail" data-tooltip="adr_popup14"><img src="/publish/images/search.png"></button>
+						</p>
+<!-- 							&nbsp;  -->
+<!-- 							<button type="button" class="btnType btnType6" data-tooltip="addrMassDupli_layer" id="tableExcelDupliBtn">중복번호</button> -->
+<!-- 							&nbsp; -->
+<!-- 							<button type="button" class="btnType btnType6" data-tooltip="addrMassSaveDupli_layer" onclick="GetAddrMassSaveDupli()" id="btnAddrMassSaveDupli">중복번호</button> -->
+						</p>
+<!-- 						<button type="button" class="btnType btnType6 addCallToF">번호추가</button> -->
+					</div>
+					
+					
+
+                    <div class="adr_excel" style="margin-top: 13px; overflow-x:auto;">
+<!--                     <div class="adr_excel" style="margin-top: 13px;"> -->
+                        <!-- thead -->
+                        <div class="adr_hd select_adr_hd msg" data-group="tableExcel">
+                            <div style="width: 100px;"></div>
+                            <div style="width: 100px;"></div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                            <div style="width: 140px;">
+                                <label for="" class="label"></label>
+                                <select class="field-selector">
+                                    <option value="">선택하기</option>
+                                    <option value="addrNm">이름</option>
+                                    <option value="addrPhoneNo">휴대폰</option>
+                                    <option value="addrInfo1">[*1*]</option>
+                                    <option value="addrInfo2">[*2*]</option>
+                                    <option value="addrInfo3">[*3*]</option>
+                                    <option value="addrInfo4">[*4*]</option>
+<!--                                    <option value="addrComment">메모</option> -->
+                                </select>
+                            </div>
+                        </div>
+                     </div>     			
+					
+					<div class="drag_drop_wrap callList_includ_box" id="tabulator_excel">
+<!-- 						<img src="/publish/images/content/excel.jpg" style="width: 100%;"> -->
+					</div>
+					<div class="excel_middle">
+						<div class="select_btnWrap clearfix">
+							<div>
+								<button type="button" id="allDel"><i class="remove_img"></i>전체삭제</button>
+								<button type="button" id="in_select_del"><i class="remove_img"></i>선택삭제</button>
+							</div>
+	
+						</div>
+					</div><!--// 공통 -->     
+	
+					<!-- 붙여놓기 설명 -->
+<!-- 	                <div class="req_area"> -->
+<!-- 	                  	<div class="text_box"> -->
+<!-- 	                    	- 휴대폰 번호가 입력된 txt 파일을 열어 복사(Ctrl+c) + 붙여넣기(Ctrl+v)로도 입력하실 수 있습니다.<br> -->
+<!-- 							- 휴대폰 번호는 필수입력 항목입니다.<br> -->
+<!-- 							- 이름,휴대폰 번호,[*1*],[*2*],[*3*],[*4*],메모 순서대로 입력해주세요.(예 : 010-1234-5678,홍길동,변수1…메모)<br> -->
+<!-- 							- 이름은 24byte, [*1*]~[*4*] 40byte, 메모는 250byte까지 입력 가능합니다.<br> -->
+<!-- 							- '오류 검사'를 통해 등록된 데이터에 전화번호 입력 오류를 확인하실 수 있습니다. -->
+<!-- 						</div> -->
+<!-- 	                </div> -->
+					<div class="popup_btn_wrap2" style="margin: 0 auto 30px auto;">
+						<button type="button" id="btnAddrMassReg">추가</button>
+						<button type="button" id="btnAddrMassClose" class="tooltip-close" data-focus="adr_popup07-close"  data-focus-next="popup07">닫기</button>                      
+					</div>
+				
+				</div>(No newline at end of file)
src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentAllListAjax.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentAllListAjax.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentAllListAjax.jsp
@@ -227,9 +227,15 @@
 											<span class="di">분할</span>
 										</c:if>
 									</c:if>
-                                    <a href="#none" onclick="fn_sentDetailView('${resultAllSentList.msgGroupId}');">
+                                   <%--  <a href="#none" onclick="fn_sentDetailView('${resultAllSentList.msgGroupId}');">
                                     	<c:out value="${resultAllSentList.smsTxt}"/>
+                                   	</a> --%>
+                                    <a href="#none" onclick="fn_sentDetailView('${resultAllSentList.msgGroupId}');">
+										<c:if test="${resultAllSentList.adFlag eq 'Y'}">(광고)</c:if>
+											<c:out value="${resultAllSentList.smsTxt}" />
+										<c:if test="${resultAllSentList.adFlag eq 'Y'}">수신거부 : 홈 > 채널차단</c:if>
                                    	</a>
+                                   	
                                 </div>
                             </td>
 							<td>
src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailPopAjax.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailPopAjax.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailPopAjax.jsp
@@ -18,19 +18,24 @@
     <!--// 탭스타일 -->
 
     <!-- 카카오 알림톡 미리보기 -->
+<!--     <div class="tab_phone current" id="tab_phone_1"> -->
+<!--         <div class="phone_kakako friendtalk"> -->
     <div class="tab_phone current kakao_wrap" id="tab_phone_1">
-        <div class="phone_kakako">
+        <div class="phone_kakako friendtalk">
             <div class="phoneIn">
-                <p class="prev_p"><img src="/publish/images/content/kakao_prev_icon.png" alt=""><c:out value="${resultMsgDetail.yellowId}"/></p>
+                <p class="prev_p"><img src="/publish/images/content/kakao_prev_icon.png" alt=""><c:out value="${searchVO.yellowId}"/></p>
                 <!-- 텍스트 미리보기 -->
                 <div class="text_preview">
-                    <div class="allimtalk_title">
+                    <%--< div class="allimtalk_title">
                         <img src="/publish/images/content/icon_allimtalk.png" alt="">
                         <c:choose>
                             <c:when test="${msgType eq '8'}">알림톡 도착</c:when>
                             <c:when test="${msgType eq '9'}">친구톡 도착</c:when>
                         </c:choose>
-                    </div>
+                    </div> --%>
+					<c:if test="${resultMsgDetail.adFlag eq 'Y'}">
+						<p class="friend_talk_title"><span>(광고)</span></p>
+					</c:if>
                     <div class="allimtalk_content">
                         <c:choose>
                             <c:when test="${kakaoTemplateInfo.templateEmphasizeType eq 'TEXT'}">
@@ -68,6 +73,11 @@
                             </c:choose>
                         </c:forEach>
                     </div>
+                    
+					<c:if test="${resultMsgDetail.adFlag eq 'Y'}">
+						<p class="kakao_block_text">수신거부 : 홈 > 채널차단</p>
+					</c:if>
+                    
                 </div>
             </div>
             <p class="addText">※ 단말기 설정에 따라 다르게 보일 수 있습니다</p>
@@ -77,14 +87,23 @@
 
 	<c:if test="${resultMsgDetail.bizKakaoResendYn eq 'Y'}">
     <!-- 문자 미리보기 -->
+	    <div class="tab_phone" id="tab_phone_2" style="display: none;position:relative;">
 <!-- 	    <div class="tab_phone" id="tab_phone_2" > -->
-	    <div class="tab_phone" id="tab_phone_2" style="display:none" >
+<!-- 	    <div class="tab_phone" id="tab_phone_2" style="display:none" > -->
 	        <div class="phoneIn">
 	            <div>
 	                <p class="prev_p"><img src="/publish/images/search.png"> 문자내용</p>
 	                <div class="text_length2 clearfix">
 	                	<c:if test="${resultMsgDetail.bizKakaoResendYn eq 'Y'}">
 		                	<c:choose>
+		                		<c:when test="${resultMsgDetail.bizKakaoResendType eq 'SMS'}">
+		                			<span class="msg_com msg_short">단문</span>
+		               			</c:when>
+		                		<c:otherwise>
+		                			<span class="msg_com msg_long">장문</span>
+		                		</c:otherwise>
+		                	</c:choose>
+<%-- 		                	<c:choose>
 		                		<c:when test="${resultMsgDetail.bizKakaoResendTypeCnt > 1}">
 									<span class="msg_com msg_short">단문</span>
 									<span class="msg_com msg_long">장문</span>
@@ -96,17 +115,34 @@
 		                		<c:otherwise>
 		                			<span class="msg_com msg_long">장문</span>
 		                		</c:otherwise>
-		                	</c:choose>
+		                	</c:choose> --%>
 	                	</c:if>
-						<div>
+						<!-- <div>
 							<span>글자크기</span>
 							<button type="button" onclick="changeFontSize('plus');"><img src="/publish/images/content/font_plus.png"></button>
 							<button type="button" onclick="changeFontSize('minus');"><img src="/publish/images/content/font_minus.png"></button>
-						</div>
+						</div> -->
 					</div>
 	                <div class="text_preview">
+		                <c:if test="${not empty fileInfos}">
+							<div class="preiew_img">
+								<c:forEach var="fileInfo" items="${fileInfos}">
+									<div class="img_box">
+											<img src="<c:url value='/cmm/fms/getImage2.do'/>?atchFileId=<c:out value="${fileInfo.atchFileId}"/>&fileSn=<c:out value="${fileInfo.fileSn}"/>" alt="발송된 그림문자 미리보기" style="width: 100%">
+									</div>
+								</c:forEach>
+							</div>
+						</c:if>
 	                    <div class="preview_auto">
+	                    
+							<c:if test="${resultMsgDetail.adFlag eq 'Y'}">
+								<p class="ad_tit">(광고)</p>
+							</c:if>
 	                        <p class="realtime">${fn:replace(resultMsgDetail.bizKakaoResendOrgnlTxt, newLineChar, "<br/>")}</p>
+	                        
+							<c:if test="${resultMsgDetail.adFlag eq 'Y'}">
+								<p class="deny_receipt">무료거부 0808800858</p>
+							</c:if>
 	                    </div>
 	                </div>
 	            </div>
src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailView.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailView.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentDetailView.jsp
@@ -394,20 +394,18 @@
 }
 
 /**
- * @알림톡 재전송
+ * @재전송
  */
-function fnMjMsgReSendAll() {
-	
+ function fnMjMsgReSendAll() {
+	var msgType = $('#msgType').val();
 	var form = document.reSendAllForm;
-
-	if (!confirm("알림톡 발송 화면으로 이동합니다.")) {
-		return;
-	}
-
-	// msgKind에 따른 action 설정
-	form.action = "/web/mjon/alimtalk/kakaoAlimtalkMsgDataView.do";
 	
-	form.submit();
+	if (confirm((msgType == '8' ? '알림톡' : '친구톡') + " 발송 화면으로 이동합니다.")) {
+		form.action = msgType == '8'
+			? '/web/mjon/alimtalk/kakaoAlimtalkMsgDataView.do'
+			: '/web/mjon/kakao/friendstalk/kakaoFriendsTalkMsgDataView.do';
+		form.submit();
+	}
 }
 
 
@@ -649,7 +647,11 @@
                               <div class="res_info_in">
                                   <div class="res_info_top clearfix">
 								<p>발송정보</p>
-                                      <p><button type="button" class="btnType btnType3" onclick="fnMjMsgReSendAll();">재전송</button></p>
+								<p>
+<%--                                       <c:if test="${result.msgType eq '8'}"> --%>
+                                            <button type="button" class="btnType btnType3" onclick="fnMjMsgReSendAll();">재전송</button>
+<%--                                       </c:if> --%>
+                               </p>
                                   </div>
                                   <div class="res_info_btm">
                                       <dl>
@@ -667,6 +669,7 @@
                                           <dd>
                                           	<c:if test="${result.msgType eq '8'}">알림톡</c:if>
 											<c:if test="${result.msgType eq '9'}">친구톡</c:if>
+											<input type="hidden" id="msgType" value="${result.msgType }"/>
 										  </dd>
                                       </dl>
                                       <dl>
@@ -750,41 +753,50 @@
                                       <p>상세결과</p>
                                       <p></p>
                                   </div>
-							<div class="res_num">
-								<div class="res_info_btm1">
-									<dl>
-										<dt>전체건수</dt>
-										<dd><a href="#" data-tooltip="rev_popup04"><span class="c_222_g"><fmt:formatNumber value="${result.msgGroupCnt}" type="number" groupingUsed="true" /></span>건</a></dd>
-									</dl>
-								</div>
-								<div class="res_info_btm1">
-									<dl>
-										<dt>성공건수</dt>
-										<dd><span class="c_002c9a_g"><fmt:formatNumber value="${result.successCount + result.kakaoResendSuccCount}" type="number" groupingUsed="true" /></span>건(${result.successPct})</dd>
-									</dl>
-								</div>
-							</div>
-							<div class="res_num">	
-								<div class="res_info_btm1">
-									<dl>
-										<dt>대기건수</dt>
-										<dd><span class="c_666_g"><fmt:formatNumber value="${result.waitCount}" type="number" groupingUsed="true" /></span>건(${result.waitingPct})</dd>
-									</dl>
-								</div>
-								<div class="res_info_btm1">
-									<dl>
-										<dt>실패건수</dt>
-										<%-- 대체문자 기능 on
-										카카오톡 실패 + 대체문자 실패 = 실패
-										대체문자 기능 off
-										카카오톡 실패 = 실패
-										*카카오톡 실패임에도 대체문자로 발송이 성공한 경우 실패로 체크하지 않음 --%>
-										<dd><span class="c_e40000_g"><fmt:formatNumber value="${result.failCount - result.kakaoResendSuccCount}" type="number" groupingUsed="true" /></span>건(${result.failedPct})</dd>
-									</dl>
-								</div>
-							</div>
-							<p class="table_bottom_txt">* 전체건수를 클릭하면 받는 사람 상세정보를 확인하실 수 있습니다.</p>
-							<p class="table_bottom_txt">* 대체문자 발송 성공 시, 성공건수로 집계됩니다.</p>
+									<div class="res_num">
+										<div class="res_info_btm1">
+											<dl>
+												<dt>전체건수</dt>
+												<dd><a href="#" data-tooltip="rev_popup04"><span class="c_222_g"><fmt:formatNumber value="${result.msgGroupCnt}" type="number" groupingUsed="true" /></span>건</a></dd>
+											</dl>
+										</div>
+										<div class="res_info_btm1">
+											<dl>
+												<dt>성공건수</dt>
+												<dd><span class="c_002c9a_g"><fmt:formatNumber value="${result.successCount + result.kakaoResendSuccCount}" type="number" groupingUsed="true" /></span>건(${result.successPct})</dd>
+											</dl>
+										</div>
+									</div>
+									<div class="res_num">	
+										<div class="res_info_btm1">
+											<dl>
+												<dt>대기건수</dt>
+												<dd><span class="c_666_g"><fmt:formatNumber value="${result.waitCount}" type="number" groupingUsed="true" /></span>건(${result.waitingPct})</dd>
+											</dl>
+										</div>
+										<div class="res_info_btm1">
+											<dl>
+												<dt>실패건수</dt>
+												<%-- 대체문자 기능 on
+												카카오톡 실패 + 대체문자 실패 = 실패
+												대체문자 기능 off
+												카카오톡 실패 = 실패
+												*카카오톡 실패임에도 대체문자로 발송이 성공한 경우 실패로 체크하지 않음 --%>
+												<dd><span class="c_e40000_g"><fmt:formatNumber value="${result.failCount - result.kakaoResendSuccCount}" type="number" groupingUsed="true" /></span>건(${result.failedPct})</dd>
+											</dl>
+										</div>
+									</div>
+									<div class="table_btn clearfix">
+										<div class="table_btn_left">
+											<!-- <button type="button" data-tooltip="rev_popup02" class="btnType btnType14"><i class="add_img"></i>주소록 등록</button>
+											<button type="button" class="excel_btn btnType"><i class="downroad"></i>엑셀 다운로드</button> -->
+											<p class="table_bottom_txt">* 전체건수를 클릭하면 받는 사람 상세정보를 확인하실 수 있습니다.</p>
+											<p class="table_bottom_txt" style="padding:5px 0 0 0;">* 대체문자 발송 성공 시, 성공건수로 집계됩니다.</p>
+										</div>
+										<div class="table_btn_right">
+											
+										</div>
+									</div>
                               </div>
                           </div>
                           <!--// 상세결과-->
@@ -925,6 +937,7 @@
 	
 	<form id="resPopForm" name="resPopForm" method="post">
 		<input type="hidden" id="msgGroupId" name="msgGroupId" value="${result.msgGroupId}"/>
+		<input type="hidden" name="yellowId" value="${result.yellowId}"/>
 <!-- 		<input type="hidden" id="msgId" name="msgId" value=""/> -->
 	</form>
 
src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentView.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentView.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/sent/KakaoSentView.jsp
@@ -364,6 +364,7 @@
 			
 			cntView('allCnt', response.allCnt);
 			cntView('atCnt', response.atCnt);
+			cntView('ftCnt', response.ftCnt);
 			//cntView('ftCnt', response.ftCnt);
         },
         error: function(error) {
@@ -564,6 +565,15 @@
 							<table class="tType4"><tbody><tr><td>LOADING...</td></tr></tbody></table>
 						</div>
 					</div>
+					<div class="rev_admin_in">
+						<div class="rev_admin_top clearfix">
+							<p>친구톡</p>
+							<p></p>
+						</div>
+						<div class="rev_admin_btm admin_btm">
+							<table class="tType4"><tbody><tr><td>LOADING...</td></tr></tbody></table>
+						</div>
+					</div>
 				</div>
 				
 				<div class="rev_admin" id="lodingAfter" style="display:none;">
@@ -607,37 +617,39 @@
 							</dl>
 						</div>
 					</div>
-					<%-- <div class="rev_admin_in" id="ftCnt">
+					<div class="rev_admin_in" id="ftCnt">
 						<div class="rev_admin_top clearfix">
 							<p>친구톡</p>
-							<p><span><fmt:formatNumber value="${totFtCnt}" pattern="#,###"/></span> 건</p>
+							<p><span name="allCnt">0</span> 건</p>
 						</div>
 						<div class="rev_admin_btm">
 							<dl>
 								<dt>대기</dt>
-								<dd><span class="c_002c9a" name="waitCnt"><fmt:formatNumber value="${waitFtCnt}" pattern="#,###"/></span>건</dd>
+								<dd><span class="c_002c9a" name="waitCnt">0</span>건</dd>
 							</dl>
 							<dl>
 								<dt>성공</dt>
-								<dd><span class="c_002c9a" name="succCnt"><fmt:formatNumber value="${succFtCnt}" pattern="#,###"/></span>건</dd>
+								<dd><span class="c_002c9a" name="succCnt">0</span>건</dd>
 							</dl>
 							<dl>
 								<dt>실패</dt>
-								<dd><span class="c_e40000" name="failCnt"><fmt:formatNumber value="${failFtCnt}" pattern="#,###"/></span>건</dd>
+								<dd><span class="c_e40000" name="failCnt">0</span>건</dd>
 							</dl>
 						</div>
-					</div> --%>
+					</div>
 				</div>
 				
 				<div class="list_tab_wrap2 type4">
 					<!-- tab button -->
 					<ul class="list_tab">
-						<li class="tab <c:if test="${searchVO.tabType eq '' || searchVO.tabType eq 'all' || empty searchVO.tabType}">active</c:if>"><button type="button" onclick="fnTabLoad('',0); return false;">전체</button></li>
-						<li class="tab <c:if test="${searchVO.tabType eq 'at'}">active</c:if>"><button type="button" onclick="fnTabLoad('at', 1); return false;">알림톡</button></li>
-						<%-- <c:if test="${pageContext.request.serverName == 'localhost' 
-						            || pageContext.request.serverName == '119.193.215.98'}">
-							<li class="tab <c:if test="${searchVO.tabType eq 'ft'}">active</c:if>"><button type="button" onclick="fnTabLoad('ft', 2); return false;">친구톡</button></li>
-				        </c:if> --%>
+						<li class="tab active"><button type="button" onclick="fnTabLoad('',0); return false;">전체</button></li>
+						<li class="tab"><button type="button" onclick="fnTabLoad('at', 1); return false;">알림톡</button></li>
+						<c:if test="${pageContext.request.serverName == 'localhost' 
+						            || pageContext.request.serverName == '119.193.215.98'
+                                    || pageContext.request.serverName == '192.168.0.176'
+						            }">
+							<li class="tab"><button type="button" onclick="fnTabLoad('ft', 2); return false;">친구톡</button></li>
+				        </c:if>
 					</ul><!--// tab button -->
 				</div>
 				<!-- 예약관리 > 전체 -->
src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateList.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateList.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateList.jsp
@@ -400,7 +400,8 @@
         <div class="top_content kakaotalkset_cont current pay_tab_wrap">
             <div class="heading">
                 <h2>카카오톡 설정</h2>
-                <button type="button" class="button info" onclick="window.open('popup_kakaoset_template.html','_blank','width=790, height=300, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">사용안내</button>
+                <button type="button" class="button info" onclick="window.open('/web/pop/ft/kakaoTemplatePop.do','_blank','width=850, height=240, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">사용안내</button>
+<!--                 <button type="button" class="button info" onclick="window.open('popup_kakaoset_template.html','_blank','width=790, height=300, top=100, left=100, fullscreen=no, menubar=no, status=no, toolbar=no, titlebar=yes, location=no, scrollbars=yes')">사용안내</button> -->
             </div>
             <div class="list_tab_wrap2 type2 ">
                 <!-- tab button -->
@@ -436,7 +437,7 @@
 	            <!-- //검색조건 -->
 	            <div class="list_info">
 	                <div class="btn_wrap left">
-	                    <button type="button" class="btnType btnType2 template_add" onclick="location.href='/web/mjon/kakao/template/selectKakaoFriendsTemplateDataRegist.do'">템플릿 등록</button>
+<!-- 	                    <button type="button" class="btnType btnType2 template_add" onclick="location.href='/web/mjon/kakao/template/selectKakaoFriendsTemplateDataRegist.do'">템플릿 등록</button> -->
 	                    <button type="button" class="btnType btnType3 btn_list"><i></i>리스트</button>
 	                    <button type="button" class="btnType btnType3 btn_thumbnail"><i></i>썸네일</button>
 	                </div>
src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateListAjax.jsp
--- src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateListAjax.jsp
+++ src/main/webapp/WEB-INF/jsp/web/kakao/template/ft/KakaoFriendstalkTemplateListAjax.jsp
@@ -132,7 +132,7 @@
 			                        <dt>등록일</dt>
 			                        <dd><c:out value="${templatInfoList.frstRegistPnttm}"/>
 			                            <div class="btn_wrap">
-			                                <button type="button" class="btn_template_edit" title="템플릿 수정"  onclick="javascript:fnTemplateDetail('<c:out value="${templatInfoList.friendId}"/>'); return false;"><img src="/publish/images/content/btn_template_edit.png" alt=""></button>
+<%-- 			                                <button type="button" class="btn_template_edit" title="템플릿 수정"  onclick="javascript:fnTemplateDetail('<c:out value="${templatInfoList.friendId}"/>'); return false;"><img src="/publish/images/content/btn_template_edit.png" alt=""></button> --%>
 			                                <button type="button" class="btn_template_delete" title="템플릿 삭제"  onclick="javascript:fnTemplateDelBtn('<c:out value="${templatInfoList.friendId}"/>'); return false;"><img src="/publish/images/content/btn_template_delete.png" alt=""></button>
 			                            </div>
 			                        </dd>
src/main/webapp/WEB-INF/jsp/web/msgdata/include/msgDataIncludeExcel.jsp
--- src/main/webapp/WEB-INF/jsp/web/msgdata/include/msgDataIncludeExcel.jsp
+++ src/main/webapp/WEB-INF/jsp/web/msgdata/include/msgDataIncludeExcel.jsp
@@ -943,19 +943,6 @@
 <!-- 									<option value="addrComment">메모</option> -->
 								</select>
                             </div>
-                            <!-- <div style="width: 125px;">
-                                <label for="" class="label"></label>
-								<select class="field-selector">
-									<option value="">선택하기</option>
-									<option value="addrNm">이름</option>
-									<option value="addrPhoneNo">휴대폰</option>
-									<option value="addrInfo1">[*1*]</option>
-									<option value="addrInfo2">[*2*]</option>
-									<option value="addrInfo3">[*3*]</option>
-									<option value="addrInfo4">[*4*]</option>
-									<option value="addrComment">메모</option>
-								</select>
-                            </div> -->
                         </div>
                      </div>					
 					
 
src/main/webapp/WEB-INF/jsp/web/pop/kakaoFtPop.jsp (added)
+++ src/main/webapp/WEB-INF/jsp/web/pop/kakaoFtPop.jsp
@@ -0,0 +1,30 @@
+
+<%@ page contentType="text/html; charset=utf-8"%>
+<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
+<%@ taglib prefix="ui" uri="http://egovframework.gov/ctl/ui"%>
+<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"%>
+<%@ taglib prefix="spring" uri="http://www.springframework.org/tags"%>
+<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt"%>
+<%@ taglib prefix="ec" uri="/WEB-INF/tld/ecnet_tld.tld"%>
+
+
+  <!-- 카톡발송 > 친구톡 -->
+  <div class="info_popup" id="friendTalkSend">
+    <div class="popup_heading">
+      <p>사용안내</p>
+    </div>
+    <div class="layer_in layer_info_wrap">
+      <ul class="info_list">
+        <!-- <li>친구톡 텍스트형은 최대 1,000자까지 작성 가능 (이미지형 400자, 와이드 이미지형 76자)</li>
+        <li>변수는 [*이름*], [*1*]~[*4*]까지 입력하실 수 있습니다.</li>
+        <li>광고성 정보 포함 시, 자동으로 (광고) 표시와 080무료수신거부번호가 메시지 내용에 표시됩니다.</li>
+        <li>(광고) 표기 사용 여부는 선택 가능하나, (광고) 표기 해제에 따른 법규 의무사항 미준수 시, 메시지 발송이 중단될 수 있습니다.</li>
+        <li>예약 메시지는 수정 가능하며, 예약 발송시간 5분전까지만 취소·삭제 가능합니다.</li>
+         -->
+        <li>친구톡은 알림톡과 달리 별도의 템플릿 심사가 필요하지 않습니다.</li> 
+		<li>등록된 템플릿은 자유롭게 삭제가 가능합니다.</li>
+		<li>문자온이 제공하지 않는 이모티콘, 이모지를 포함하여 템플릿을 등록할 경우, 친구톡 발송이 불가합니다.</li>
+      </ul>
+    </div>
+  </div>
+  <!-- // 카톡발송 > 친구톡 -->(No newline at end of file)
src/main/webapp/js/kakao/at/addr.js
--- src/main/webapp/js/kakao/at/addr.js
+++ src/main/webapp/js/kakao/at/addr.js
@@ -202,55 +202,63 @@
 }
 
 function loadAddrList(){
-	console.log(' loadAddrList() ');
 
-	/*
-		serialize 를 사용할때는 processData, contentType 옵션 제가할것
-	*/
-	var data = $("#searchAddrGrpForm").serialize();	 
+    /*
+        serialize 를 사용할때는 processDa    ta, contentType 옵션 제가할것
+    */
+    var data = $("#searchAddrGrpForm").serialize();  
 
-	// var url = "/web/mjon/msgdata/selectMsgAddrListAjax.do";
     var url = "/web/mjon/msgdata/selectMsgAddrListAjax_advc.do";
-	
-	$.ajax({
+    
+    $.ajax({
         type: "POST",
         url: url,
         data: data,
         dataType:'json',
-        async: false,
+        async: true,
         cache: false,
         success: function (data) {
-        console.log('data : ', data);
-        if(data.status == "OK"){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
-            
-                var addrList = data.object;
+            console.log('data : ', data);
+            if(data.status == "OK"){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
                 
-                if(addrList.length == 0){
+                    var addrList = data.object;
                     
-                    alert("주소록 정보가 없습니다.");
-//                    tableAddr.setData([]);
-                    return false;
-                }
-                tableAddr.setData(addrList);
-        } 
-        else
-        {
-            alert("주소록 불러오기에 실패하였습니다. !!");
-        }
+                    if(addrList.length == 0){
+                        
+                        alert("주소록 정보가 없습니다.");
+//                      tableAddr.setData([]);
+                        return false;
+                    }
+                    
+//                  console.log('data : ', data);
+                    tableAddr.setData(addrList);
+            } 
+            else
+            {
+                alert("주소록 불러오기에 실패하였습니다. !!");
+            }
         },
-        error: function (e) {
-             alert("주소록 불러오기에 실패하였습니다."); console.log("ERROR : ", e); 
-         }
-        , beforeSend : function(xmlHttpRequest) {
-                    //로딩창 show
+        error: function (jqXHR, textStatus, errorThrown) {
+            alert("주소록 불러오기에 실패하였습니다.");
+            console.error("Error:", jqXHR.status, textStatus, errorThrown);
+            console.error("Response Text:", jqXHR.responseText);
+            try {
+                let json = JSON.parse(jqXHR.responseText);
+                console.log("Parsed JSON Response:", json);
+            } catch (e) {
+                console.error("JSON Parse Error:", e);
+            }
+        },
+        beforeSend : function(xmlHttpRequest) {
+            //로딩창 show
             $('.loading_layer').addClass('active');             
-        }
-        , complete : function(xhr, textStatus) {
+        },                      
+        complete : function(xhr, textStatus) {
             //로딩창 hide
             $('.loading_layer').removeClass('active');
         }
     });
-	
+    
 }
 
 function fnAddrSearch(){
 
src/main/webapp/js/kakao/ft/addr.js (added)
+++ src/main/webapp/js/kakao/ft/addr.js
@@ -0,0 +1,538 @@
+$(document).on('click', '.addressregi_btn', function() {
+	var tableData = tableL.getRows();
+	var dataLen = tableL.getRows().length;
+	if(dataLen == 0){
+		alert("연락처 정보를 등록해 주세요.");
+		return false;
+	}else{
+		$('.addressregi_layer').css({'width':'680px','display':'block','left':'50%','top':'50%','transform':'translate(-50%,-50%)'});
+		setTimeout(function(){
+			$('.addressregi_layer').css({'opacity':'1'});
+		},150);
+		$('.mask').addClass('on');
+		getAddrGroupList();
+	}
+});
+
+$(document).on('click', '.addressregi_layer .tooltip-close', function() {
+	$('.addressregi_layer').attr('style','');
+	$("#addrGrpNm").val("");
+});
+
+$(document).on('change', '#addrGrpIdInfo', function() {
+	if ($("#addrGrpIdInfo option:selected").val() != "NEW") {
+		$("#addrGrpNm").val(""); //	새그룹명 Clear;
+	}
+});
+
+//주소록 그룹정보 불러오기
+function getAddrGroupList() {
+ $.ajax({
+     type : "POST",
+     async : false,
+     url : "/web/mjon/addr/addrGroupListAjax.do",
+     data : {},
+     dataType:'json',
+     success : function(data) {
+		//alert(JSON.stringify(data.addrGroupList));
+
+		// Show Html
+		getAddrGroupListShow(data.addrGroupList);
+     },
+     error : function(xhr, status, error) {
+         alert(error);
+         return false;
+     }
+ });	    	
+}
+
+//Show Html
+function getAddrGroupListShow(jsonList) {
+	var sHtml = "";
+	sHtml += "<option value='NEW'>그룹추가</option>";
+	sHtml += "<option value='0'>그룹미지정</option>";
+	sHtml += "<option value='bookmark'>자주보내는 번호</option>";
+	for (var j = 0; j < jsonList.length; j++) {
+		sHtml += "	<option value='" + $.trim(jsonList[j].addrGrpId) + "' />" + $.trim(jsonList[j].addrGrpNm) + "</option>";
+	}
+	    	
+	$("#addrGrpIdInfo").html(sHtml);
+}
+
+//주소록 그룹 중복체크
+function getAddrGroupDuplCheckAjax(addrGrpNm) {
+	console.log('addrGrpNm : ', addrGrpNm)
+	console.log('getAddrGroupDuplCheckAjax()')
+	var isReturn = true;
+	
+	$.ajax({
+		url : "<c:url value='/web/addr/getAddrGroupDuplCheckAjax.do' />", 
+		type : 'POST', 
+		data : {"addrGrpNm" : addrGrpNm},
+		dataType:'json',
+		async: false,			// 동기
+		success : function(data, status){
+			if(data.isSuccess == true) {
+				if(data.isDupl == true) {
+					//alert("중복된 그룹명입니다.");
+					isReturn = false;
+				}
+			} 
+			else {
+				//alert("Message : " + msg);
+			}			
+		},
+		error: function (e) {
+			//alert("주소록 중복체크에 실패했습니다.");
+		}
+	});
+	
+	return isReturn;	
+}
+
+//주소록 팝업 닫기 기능
+function addrClose(){
+	
+	$(".closeAddr").trigger("click");
+	
+	//주소록 레이어 팝업의 Tabulator 데이터 지워주기
+	tableAddr.clearData();
+	
+}
+
+//주소록 불러오기에서 수신자 리스트 추가해 주기
+/*
+function addrToList(){
+	
+	var selectedData = tableAddr.getSelectedRows();
+	var tableData = [];
+	
+	if(selectedData == "" || selectedData == null){
+		
+		alert("주소록을 선택해 주세요.");
+		return false;
+	
+	}else{ // 선택한 Row 데이터 저장해주기
+		
+		// 선택한 Row 데이터 저장해주기
+//		if(selectedData.length > 500){
+//			alert("최대 발송 건수는 500 입니다.");
+//			return false;
+//		}else{
+			for(var i=0; i < selectedData.length; i++){
+				
+				//좌측 받는사람 리스트를 담아둔 배열에 데이터를 추가해 준다.
+				tableData.push({
+					phone: removeDash(selectedData[i].getData().addrPhone)
+					, name: selectedData[i].getData().addrName
+					// , rep1: selectedData[i].getData().addrRep1
+					// , rep2: selectedData[i].getData().addrRep2
+					// , rep3: selectedData[i].getData().addrRep3
+					// , rep4: selectedData[i].getData().addrRep4
+				});
+			
+			}
+			//선택한 데이터 받는사람 리스트에 추가해 주기
+			addPhoneInfo(tableData);
+			$(".closeAddr").trigger("click");
+			
+			//주소록 레이어 팝업의 Tabulator 데이터 지워주기
+			tableAddr.clearData();
+//		}
+	}
+	
+}
+*/
+
+//주소록 불러오기에서 수신자 리스트 추가해 주기
+function addrToList_advc(type){
+
+
+    // 선택된 데이터 또는 전체 데이터 변수 초기화
+    let selectedData = type === 'select' ? tableAddr.getSelectedRows() : tableAddr.getData();
+    
+    // 데이터가 비어있으면 경고 후 종료
+    if (!selectedData || selectedData.length < 1) {
+        
+        if(tableAddr.getDataCount() < 1){
+            alert("주소록을 선택해 주세요.");
+        }else{
+            alert("전화번호를 선택해 주세요.");
+        }
+        
+        return false;
+    }
+
+
+    // 데이터 변환 로직
+    const addrData = selectedData.map(row => {
+        const rowData = type === 'select' ? row.getData() : row; // 'select'는 행 객체에서 데이터 추출
+        return {
+            name: rowData.name,
+            phone: removeDash(rowData.phone),
+            rep1: rowData.rep1,
+            rep2: rowData.rep2,
+            rep3: rowData.rep3,
+            rep4: rowData.rep4
+        };
+    })
+    .filter(row => checkHpNum(row.phone)); // 유효한 번호만 필터링;
+
+
+    // 기존 tableL의 데이터를 가져옵니다.
+    var existingData = tableL.getData();
+    // 기존 데이터와 새로운 데이터를 합칩니다.
+    var combinedData = existingData.concat(addrData);
+
+    // @ phone을 기준으로 중복 제거 및 갯수 계산
+    const result = removeDuplicatesAndCount(combinedData, 'phone');
+
+
+
+    // 총 30만건이 넘으면 false
+    if (!validateRowLimit(result.uniqueCount)) {
+        return false;
+    }
+
+    setAllCntData(result);
+
+
+    // 미리보기 버튼 활성화
+
+
+    $(".closeAddr").trigger("click");
+    //주소록 레이어 팝업의 Tabulator 데이터 지워주기
+    tableAddr.clearData();
+	//결제 금액 구하기
+	totalFtPriceSum(tableL.getRows().length);
+    // fn_priceClclt(result.uniqueCount);
+}
+
+function loadAddrList(){
+
+    /*
+        serialize 를 사용할때는 processDa    ta, contentType 옵션 제가할것
+    */
+    var data = $("#searchAddrGrpForm").serialize();  
+
+    var url = "/web/mjon/msgdata/selectMsgAddrListAjax_advc.do";
+    
+    $.ajax({
+        type: "POST",
+        url: url,
+        data: data,
+        dataType:'json',
+        async: true,
+        cache: false,
+        success: function (data) {
+            console.log('data : ', data);
+            if(data.status == "OK"){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
+                
+                    var addrList = data.object;
+                    
+                    if(addrList.length == 0){
+                        
+                        alert("주소록 정보가 없습니다.");
+//                      tableAddr.setData([]);
+                        return false;
+                    }
+                    
+//                  console.log('data : ', data);
+                    tableAddr.setData(addrList);
+            } 
+            else
+            {
+                alert("주소록 불러오기에 실패하였습니다. !!");
+            }
+        },
+        error: function (jqXHR, textStatus, errorThrown) {
+            alert("주소록 불러오기에 실패하였습니다.");
+            console.error("Error:", jqXHR.status, textStatus, errorThrown);
+            console.error("Response Text:", jqXHR.responseText);
+            try {
+                let json = JSON.parse(jqXHR.responseText);
+                console.log("Parsed JSON Response:", json);
+            } catch (e) {
+                console.error("JSON Parse Error:", e);
+            }
+        },
+        beforeSend : function(xmlHttpRequest) {
+            //로딩창 show
+            $('.loading_layer').addClass('active');             
+        },                      
+        complete : function(xhr, textStatus) {
+            //로딩창 hide
+            $('.loading_layer').removeClass('active');
+        }
+    });
+    
+}
+
+function fnAddrSearch(){
+	
+	var form = document.searchAddrGrpForm;
+	form.searchKeyword.value = form.searchAddrKeyword.value;
+	form.searchCondition.value = form.searchAddrCondition.value;
+	
+	loadAddrList();
+	
+}
+
+//주소록 불러오기 팝업의 그룹 선택시 우측에 주소록 불러오기
+function fnSelectAddrList(type,addrGrpId,item){
+	
+	// 주소록 그룹 열림, 닫힘 폴더이미지 초기화
+	$(".adr_pop_list div p").each(function (index, item) {
+		$(item).find("img").attr("src", "/publish/images/content/close_folder2.png");
+		$(item).find("img").attr("alt", "폴더 닫힘");
+	});
+	
+	var form = document.searchAddrGrpForm;
+	form.type.value = type;
+	form.searchAddrGrpId.value = addrGrpId;
+	form.searchKeyword.value = "";
+	form.searchCondition.value = form.searchAddrCondition.value;
+	
+	//왼쪽 그룹리스트의 그룸명을 선택시 검색어를 초기화해준다.
+	form.searchAddrKeyword.value="";
+	
+	/*
+	$(item).toggleClass("open");
+	if ($(item).hasClass("open") === true) {
+		$(item).find("img").attr("src", "/publish/images/content/open_folder2.png");
+		$(item).find("img").attr("alt", "폴더 열림");
+	}
+	else {
+		$(item).find("img").attr("src", "/publish/images/content/close_folder2.png");
+		$(item).find("img").attr("alt", "폴더 닫힘");
+	}
+	*/
+	   
+	$(item).find("img").attr("src", "/publish/images/content/open_folder2.png");
+	$(item).find("img").attr("alt", "폴더 열림");
+	
+	loadAddrList();
+	
+}
+
+//주소록 불러오기 그룹명 검색 기능처리
+function fnAddrGrpSearch(){
+	
+	var form = document.searchAddrGrpForm;
+	form.searchKeyword.value = form.searchGrpKeyword.value;
+	form.searchCondition.value = "";
+	
+	var sendData = $(document.searchAddrGrpForm).serializeArray();
+	
+	$("#addrGroupLoad").load("/web/mjon/msgdata/selectAddrGroupListAjax.do", sendData ,function(response, status, xhr){
+		//리스트 스크롤 처리해주기
+		$(".adr_pop_list").mCustomScrollbar({
+			axis: 'y',
+			scrollbarPosition: "outside",
+			theme: "dark",
+			autoHideScrollbar: false
+		});
+	});
+	
+}
+/*
+//문자발송 받는사람 목록 주소록에 등록하기 기능 처리
+$('.registAddr').click(function(){
+	
+	var tableData = tableL.getRows();
+	var dataLen = tableL.getRows().length;
+	
+	var nameList = [];		//치환문자 이름
+	var phoneNum = [];		//받는사람
+	var rep1List = [];		//치환문자1
+	var rep2List = [];		//치환문자2
+	var rep3List = [];		//치환문자3
+	var rep4List = [];		//치환문자4
+	
+	if(dataLen > 0){
+		
+		for(var i=0; i < dataLen; i++){
+			
+			var name = tableData[i].getData().name;
+			var phone = removeDash(tableData[i].getData().phone);
+			var rep1 = tableData[i].getData().rep1;
+			var rep2 = tableData[i].getData().rep2;
+			var rep3 = tableData[i].getData().rep3;
+			var rep4 = tableData[i].getData().rep4;
+			
+			if(phone == "" || phone == null){
+				
+				alert("수신 목록에 핸드폰 번호가 없는 항목이 있습니다.");
+				return false;
+				
+			}else if(!checkHpNum(phone)){
+				
+				alert("수신 목록에 잘 못된 핸드폰 번호가 있습니다. 핸드폰 번호 : " + phone + " 입니다.");
+				return false;
+				
+			}
+			
+			nameList[i] = name;
+			phoneNum[i] = phone;
+			
+			if(rep1 == "" || rep1 == null){
+				rep1List[i] = "-";
+			}else{
+				rep1List[i] = rep1;
+			}
+			
+			if(rep2 == "" || rep2 == null){
+				rep2List[i] = "-";
+			}else{
+				rep2List[i] = rep2;
+			}
+			
+			if(rep3 == "" || rep3 == null){
+				rep3List[i] = "-";
+			}else{
+				rep3List[i] = rep3;
+			}
+			
+			if(rep4 == "" || rep4 == null){
+				rep4List[i] = "-";
+			}else{
+				rep4List[i] = rep4;
+			}
+			
+			 if(name == "" || name == null){
+				
+				alert("수신 목록에 이름이 없는 항목이 있습니다. 이름을 입력해 주세요");
+				return false;
+				
+			}else{
+				
+				nameList[i] = name;
+				phoneNum[i] = phone;
+				
+				if(rep1 == "" || rep1 == null){
+					rep1List[i] = "-";
+				}else{
+					rep1List[i] = rep1;
+				}
+				
+				if(rep2 == "" || rep2 == null){
+					rep2List[i] = "-";
+				}else{
+					rep2List[i] = rep2;
+				}
+				
+				if(rep3 == "" || rep3 == null){
+					rep3List[i] = "-";
+				}else{
+					rep3List[i] = rep3;
+				}
+				
+				if(rep4 == "" || rep4 == null){
+					rep4List[i] = "-";
+				}else{
+					rep4List[i] = rep4;
+				}
+				
+			} 
+			
+		}
+		
+		//주소록 정보를 Form에 넣어준다.
+		
+		var form = document.msgForm;
+		form.callToList.value = phoneNum;
+		form.nameList.value = nameList;
+		form.rep1List.value = rep1List;
+		form.rep2List.value = rep2List;
+		form.rep3List.value = rep3List;
+		form.rep4List.value = rep4List;
+		form.addrGrpId.value = $("#addrGrpIdInfo").val();
+		form.addrGrpNm.value = $("#addrGrpNm").val();
+		
+		if ($("#addrGrpIdInfo option:selected").val() == "NEW" && ($("#addrGrpNm").val() == "" || $("#addrGrpNm").val() == null || $("#addrGrpNm").val() == undefined)) {
+			alert("저장할 그룹을 선택하거나 새 그룹명을 입력해주세요.");
+			return false;		
+		}else if ($("#addrGrpIdInfo option:selected").val() == "NEW" && $("#addrGrpNm").val() != "") {
+			//주소록 중복체크
+			if (getAddrGroupDuplCheckAjax() == false) {
+				alert("중복된 그룹명입니다. 새 그룹명을 입력해주세요.");
+				return false;			
+			}
+		}
+		if(confirm("연락처 정보를 주소록에 등록 하시겠습니까?")){
+			
+			var data = new FormData(form);
+			var url = "/web/mjon/msgdata/insertMsgAddrListAjax.do";
+			
+			$.ajax({
+		        type: "POST",
+		        enctype: 'multipart/form-data',
+		        url: url,
+		        data: data,
+		        dataType:'json',
+		        async: true,
+		        processData: false,
+		        contentType: false,
+		        cache: false,
+		        //timeout: 600000,
+		        success: function (returnData, status) {
+					if(status == 'success'){ // status 확인 필요한가. 석세스 안뜨면 에러 가지 않나
+						
+						if(returnData.result == "success"){
+							var alertMsg = "총 " + returnData.resultCnt + "건의 " +returnData.message;
+							
+							if(returnData.dupliCnt > 0) {
+								alertMsg += "\n"+numberWithCommas(returnData.dupliCnt) + "건은 이미 등록되어있는 번호입니다.";
+							}
+
+							alert(alertMsg);
+							$("#addrGrpNm").val("");
+							$('.addressregi_layer').attr('style','');
+							$('.mask').removeClass('on');
+							return false;
+							
+						}else if(returnData.result == "allDupl") {
+							alert("요청하신 "+returnData.dupliCnt+"건의 번호가 주소록에 이미 등록되어있습니다.");
+							$("#addrGrpNm").val("");
+							$('.addressregi_layer').attr('style','');
+							$('.mask').removeClass('on');
+							return false;
+						}else{
+							alert(returnData.message);
+							$("#addrGrpNm").val("");
+							$('.addressregi_layer').attr('style','');
+							$('.mask').removeClass('on');
+							return false;
+							
+						}
+						
+					} else if(status== 'fail'){
+						alert("주소록 등록에 실패하였습니다. !!");
+					}
+				},
+		        error: function (e) { 
+		        	alert("주소록 등록에 실패하였습니다."); 
+		        	console.log("ERROR : ", e); 
+		        },
+				beforeSend : function(xmlHttpRequest) {
+		        	//로딩창 show
+		        	$('.loading_layer').addClass('active');				
+				},	        	        
+		        complete : function(xhr, textStatus) {
+		        	//로딩창 hide
+		        	$('.loading_layer').removeClass('active');
+				}
+		    });
+			
+		}
+		
+	}else{
+		
+		alert("연락처 정보를 등록해 주세요.");
+		return false;
+		
+	}
+	
+	
+});
+*/
src/main/webapp/js/kakao/ft/ftPriceClclt.js
--- src/main/webapp/js/kakao/ft/ftPriceClclt.js
+++ src/main/webapp/js/kakao/ft/ftPriceClclt.js
@@ -13,6 +13,18 @@
  * 
  */
 
+//카카오 mms sms 단가 셋팅
+var KAKAO_FT_PRICE  = '';
+var SHORT_PRICE     = '';
+var LONG_PRICE      = '';
+$('document').ready(function(){
+
+    KAKAO_FT_PRICE  = Number($('#kakaoFtPrice').val());
+    SHORT_PRICE     = Number($('#shortPrice').val());
+    LONG_PRICE      = Number($('#longPrice').val());
+    console.log('SHORT_PRICE: ', SHORT_PRICE);
+    console.log('LONG_PRICE: ', LONG_PRICE); 
+});
 
 /**
  * 
@@ -21,67 +33,59 @@
  * 				대체 문자 X
  * 				카카오 금액 * 수신자 수 계산
  */
-function totalFtPriceSum(totRows){
-	
-	var collNumCnt = parseInt(totRows); 	//받는사람 건수
-	var price = $("#kakaoFtPrice").val();							//개별 건수 금액
-	var totalPrice = 0;						//전체 금액
-	var totalStr = "0";							//전체 합계 금액
-	var userMoney = $('#hdUserMoney').text(); 	//헤더 영역 보유 금액 불러오기
-	
-	if(!userMoney > 0){
-		
-		userMoney = 0;
-		
-	}
-	//헤더 영역 보유 금액 콤마 문자 제거
-	if(userMoney != ''){
-		
-		userMoney = userMoney.replaceAll("," , "");
-		
-	}
-	
-	//대체문자가 있는 경우 대체문자의 단/장문에 따른 금액 계산
-	var subMsgSts = $("#send_fail_check").is(":checked");
-	
-	if(subMsgSts){
-		
-		var conLeng = conByteLeng($('#smsTxtArea').val()); // 내용 문자 입력 바이트 수 계산하기
-		if(conLeng > 90){
-			price = $("#longPrice").val();
-		}else{
-			price = $("#shortPrice").val();
+function totalFtPriceSum(totRows) {
+	console.log(' :totalFtPriceSum: ');
+	// 수신자 수 계산
+	var count = parseInt(totRows);
+
+	// 기본 단가는 일반 친구톡 이미지 없는 가격
+	var price = KAKAO_FT_PRICE;
+
+	// 선택된 이미지 유형에 따라 단가 변경
+	var imgTypeId = $("input[name='img_file_add']:checked").attr("id");
+	if (imgTypeId === "img_file_1") price = KAKAO_FT_IMG_PRICE;
+	else if (imgTypeId === "img_file_2") price = KAKAO_FT_WIDE_IMG_PRICE;
+
+	// 대체문자 발송 옵션이 체크된 경우, 메시지 종류별로 단가 재계산
+	if ($("#send_fail_check").is(":checked")) {
+		if (imgTypeId === "img_file_0") {
+			// 장문/단문 여부에 따라 가격 비교 후 더 큰 값으로 설정
+
+			var contents = adYChkAndMakeContents($('#smsTxtArea').val());
+			
+			var len = conByteLeng(contents);
+			var long = parseFloat($("#longPrice").val());
+			var short = parseFloat($("#shortPrice").val());
+			price = Math.max(price, len > 90 ? long : short);
+		} else {
+			// 이미지 대체문자의 경우 별도 이미지 가격과 비교
+			var picture = parseFloat($('#picturePrice').val());
+			price = Math.max(price, picture);
 		}
-		
 	}
-	
-	totalPrice = price * collNumCnt;
-	
-	// 소수점 첫째자리 까지 표시
-	totalPrice = totalPrice.toFixed(1);
-	
-	if(totalPrice > 0){
-		
-		//totalStr = totalPrice.toFixed(2);
-		totalStr = totalPrice;
-		
-	}
-	
-	//개별 문자 단가 파라미터에 입력
+
+	// 총 금액 계산 (단가 * 수신자 수)
+	var totalPrice = (price * count).toFixed(1);
+	var totalStr = totalPrice > 0 ? totalPrice : "0";
+
+	// 사용자 보유 금액 텍스트에서 콤마 제거 (표시 용도일 뿐 실제 계산엔 안 씀)
+	var userMoney = $('#hdUserMoney').text().replaceAll(",", "") || 0;
+
+	// 개별 단가, 총 금액을 input/화면에 반영
 	$('#eachPrice').val(numberWithCommas(price));
-	
-	//결제금액 합계 파라이터에 입력
 	$('#totPrice').val(numberWithCommas(totalStr));
-	
-	//결제금액 합계 화면에 표시
 	$('#totalPriceTxt').text(numberWithCommas(totalStr));
-	
-	$('#repPriceTxt').hide();
-	
-	return totalStr;
-	
+	$('#repPriceTxt').hide(); // 기존 합계 표시 숨김
+
+	return totalStr; // 계산된 총 금액 반환
 }
 
+function adYChkAndMakeContents(contents){
+	if ($("#ad_Y").is(":checked")) {
+		contents = "(광고)"+contents+"\n무료거부 0808800858"
+	}
+	return contents;
+}
 
 
 /*
@@ -89,10 +93,6 @@
  * 
  * */
 
-//카카오 mms sms 단가 셋팅
-var KAKAO_FT_PRICE	= '';
-var SHORT_PRICE		= '';
-var LONG_PRICE		= '';
 
 /*$(document).ready(function(){
 
@@ -153,14 +153,17 @@
 /**
  * @description 금액 계산 function
  */
-function fn_priceClclt(){
+function fn_priceClclt(phoneSu = 0){
+    console.log('fn_priceClclt : ', phoneSu);
 
 	// 미리보기 텍스트
 	var templateHtml = $('#smsTxtArea').val();
 	// var templateHtml = $('.template_text').html();
 	
 	// 수신 번호 개수
-	var phoneSu = $('.phoneArea').length;
+    if(phoneSu == 0){
+        phoneSu = $('.phoneArea').length;
+    }
 	// 대체문자 있는지 확인
 	var isSendFailChecked = $("#send_fail_check").is(":checked");
 	// 치환문자 여부 확인
@@ -183,12 +186,13 @@
  * 				카카오 금액 * 수신자 수 계산
  */
 function fn_sendFailUnChecked(phoneSu){
+    console.log('phoneSu; ', phoneSu);
 	
 	// 카카오 금액 * 수신자 수
 	// fn_writePriceText() 첫 파라미터가 null이면 카카오 전송으로 인식
 	fn_writePriceText('', phoneSu);
 	// $('#totalPriceTxt').text((KAKAO_AT_PRICE * phoneSu).toFixed(1));
-	fn_priceText('알림톡', phoneSu);
+	fn_priceText('친구톡', phoneSu);
 }
 
 /**
@@ -279,9 +283,9 @@
 	msgTypeText = msgTypeText.trim();
 	if(msgTypeText == '단문') price = SHORT_PRICE * phoneSu;
 	else if(msgTypeText == '장문') price = LONG_PRICE * phoneSu;
-	else  price = KAKAO_AT_PRICE * phoneSu; // 카카오
+	else  price = KAKAO_FT_PRICE * phoneSu; // 카카오
 
-	$('#totalPriceTxt').text((price).toFixed(1));
+	$('#totalPriceTxt').text(numberWithCommas((price).toFixed(1)));
 }
 
 
src/main/webapp/js/kakao/ft/ftTabulator.js
--- src/main/webapp/js/kakao/ft/ftTabulator.js
+++ src/main/webapp/js/kakao/ft/ftTabulator.js
@@ -20,40 +20,41 @@
 	tableL = new Tabulator(".callList_box", {
 		height:"255px",
 	    layout:"fitColumns",
-	    //data:tabledata,
-	    //autoColumns:true,
 	    headerHozAlign:"center",
 	    validationMode:"highlight",
-	    //clipboard:false,
-	    //clipboardCopySelector:"table",
-	    //clipboardPasteAction:"insert", // insert, update, replace
 	    placeholder:"복사(Ctrl+C)한 내용을 여기에 붙여넣기(Ctrl+V) 해주세요.", //fit columns to width of table (optional)
 	    resizableColumns:false,
+        columnDefaults:{ // 공통설정
+            hozAlign: "center",
+            headerHozAlign: "center",
+            editor: false,
+            editor: false
+        },
 	 	columns:[ //Define Table Columns
-	 		{formatter:"rowSelection", titleFormatter:"rowSelection",clipboard:false, hozAlign:"center", headerSort:false, cellClick:function(e, cell){
-	 	        cell.getRow().toggleSelect();
-		 	}}, 
-		 	{title:"이름", hozAlign:"center", field:"name", editor:"input", validator:["maxLength:12"], cellEdited:function(cell){
+            {formatter:"rowSelection", headerHozAlign:"center", titleFormatter:"rowSelection",clipboard:false, hozAlign:"center", width:5, headerSort:false, cellClick:function(e, cell){
+                cell.getRow().toggleSelect();
+            }}, 
+		 	{title:"이름", hozAlign:"center", field:"name", validator:["maxLength:12"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 	    fnReplCell();
 		    }},
-		 	{title:"휴대폰", hozAlign:"center", field:"phone", editor:"input", width:100, validator:["required","minLength:10", "maxLength:12"], cellEdited:function(cell){
+		 	{title:"휴대폰", hozAlign:"center", field:"phone", width:100, validator:["required","minLength:10", "maxLength:12"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 	    fnDuplPhone();
 		    }},
-		 	{title:"[*1*]", hozAlign:"center", field:"rep1", editor:"input", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
+		 	{title:"[*1*]", hozAlign:"center", field:"rep1", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 	    fnReplCell();
 		    }},
-		 	{title:"[*2*]", hozAlign:"center", field:"rep2", editor:"input", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
+		 	{title:"[*2*]", hozAlign:"center", field:"rep2", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 		fnReplCell();
 		    }},
-		 	{title:"[*3*]", hozAlign:"center", field:"rep3", editor:"input", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
+		 	{title:"[*3*]", hozAlign:"center", field:"rep3", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 		fnReplCell();
 		    }},
-		 	{title:"[*4*]", hozAlign:"center", field:"rep4", editor:"input", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
+		 	{title:"[*4*]", hozAlign:"center", field:"rep4", minWidth:60, validator:["maxLength:40"], cellEdited:function(cell){
 		 	    //cell - cell component
 		 		fnReplCell();
 		    }},
@@ -88,7 +89,7 @@
 
 	});
 	
-	 
+	 /*
 	//주소록 불러오기 팝업 내용
 	//Tabulator AJAX Data Loading
 	tableAddr = new Tabulator(".callAddr_box", {
@@ -126,7 +127,61 @@
 	 		return value % parameters.addrPhone;
 	    },
 
-	});
+	});*/
+    
+
+    // Tabulator 초기화
+    tableAddr = new Tabulator(".callAddr_box", {
+        height: "255px",
+        layout: "fitData",
+        headerHozAlign: "center",
+        validationMode: "highlight",
+        resizableColumns: true,
+        progressiveLoad:"scroll",
+        progressiveRender: true,   // 렌더링 최적화 활성화
+        progressiveRenderSize: 100, // 한 번에 렌더링할 행 수를 줄임
+        placeholder: "주소록 그룹을 선택해 주세요.", 
+        columns: [
+//            {formatter: "rowSelection", clipboard: false, hozAlign: "center", headerSort: false}, 
+            {formatter: "rowSelection", titleFormatter: "rowSelection", hozAlign: "center", headerSort: false, 
+                cellClick: function(e, cell) {
+                    fn_loadAddActive(); // 로딩 활성화
+                    setTimeout(() => {
+                        tableAddr.blockRedraw(); // 렌더링 차단
+                        cell.getRow().toggleSelect(); // 선택 상태 토글
+                        tableAddr.restoreRedraw(); // 렌더링 재개
+                        fn_loadRemoveActive(); // 로딩 비활성화
+                    }, 0); // 비동기적으로 실행
+                }
+            },
+            {title: "No", hozAlign: "center", formatter: "rownum", align: "center", headerHozAlign: "center", width: 60},
+            {title: "그룹명", hozAlign: "center", field: "addrGroupNm", editor: false, width: 100},
+            {title: "이름", hozAlign: "center", field: "name", editor: false, width: 100},
+            {title: "휴대폰번호", hozAlign: "center", field: "phone", editor: false, width: 100},
+            {title: "[*1*]", hozAlign: "center", field: "rep1", editor: false, width: 84},
+            {title: "[*2*]", hozAlign: "center", field: "rep2", editor: false, width: 84},
+            {title: "[*3*]", hozAlign: "center", field: "rep3", editor: false, width: 84},
+            {title: "[*4*]", hozAlign: "center", field: "rep4", editor: false, width: 84},
+        ],
+        validationFailed: function(cell, value, parameters) { 
+            var valid = cell.isValid();
+            var fieldNm = cell.getField();
+            if (!valid) {
+                if (fieldNm == "addrName") {
+                    alert("받는사람 이름은 최대 12글자까지만 입력 가능합니다.");
+                } else if (fieldNm == "addrPhone") {
+                    alert("휴대폰번호는 하이픈(-)을 제외한 숫자만 정확히 입력해 주세요.");
+                } else if (fieldNm == "addrGroupNm") {
+                    alert("그룹명을 정확히 입력해 주세요. 2 ~ 40글자 이내로 입력 가능합니다.");
+                } else {
+                    alert("치환문자를 정확히 입력해 주세요. 100글자 이내로 입력 가능합니다.");
+                }
+                cell.setValue("");
+            }
+            console.log('parameters : ', parameters);
+            return value % parameters.addrPhone;
+        },
+    });
 	
 		
 	//받는사람 번호 버튼 클릭시 Tabulator에 데이터 넣어주기
@@ -186,6 +241,9 @@
 			addPhoneInfo(tabledata);
 	    	
 		    $('#callTo').val("");
+			
+
+			fn_priceText('친구톡', tableL.getDataCount());
 	    
 	    }
 		
@@ -496,6 +554,210 @@
 		
 	});
 	
+	
+	
+
+
+	//받는사람 목록에 복사/붙여넣기 기능 처리
+	$('.callList_box').on('paste', function (e) {
+		console.log(' :: callList _box :: ');
+		
+		var element = e.originalEvent.clipboardData.getData('text'); // 클립보드에 복사한 데이터 가져오기
+		
+		var elmSplit= [];
+		  
+		elmSplit = element.split("\n");
+		
+		var elmLen = elmSplit.length;
+		
+		  
+		if(elmLen < 0){
+			  
+			alert("붙여넣을 연락처를 복사해주세요.");
+			return false;
+			  
+		}else{
+			tableErrorData.length = 0;	// 오류 번호 배열 초기화
+			
+			var splitData = [];
+			var realPhone = [];
+			for(var i=0; i < elmLen; i++){
+				  
+				var splitStr = elmSplit[i]; 
+				var tabData = [];
+				var comData = [];
+				
+				if(splitStr.indexOf('\t') != -1){
+					splitData = splitStr.split('\t'); //탭 구분으로 데이터 분할
+				}else if(splitStr.indexOf(',') != -1){
+					splitData = splitStr.split(','); //콤마 구분으로 데이터 분할
+				}else{
+					splitData = splitStr.split(' ');
+				}
+				
+				
+				if(splitData.length == 0){// 데이터가 없는경우
+					
+					alert("탭으로 구분하여 데이터를 복사해 주세요.");
+				  	return false;
+				  
+				}else if(splitData.length == 1){// 데이터가 탭으로 구분이 없는 경우
+					
+					for(var j=0; j < splitData.length; j++){
+						if(checkHpNum(splitData[j].trim())){//핸드폰 양식이 맞는지 확인
+							//배열 끝에 데이터 추가해 주기
+							realPhone.push({phone: removeDash(splitData[j].trim()), name: ""});
+						}
+						else {
+							if (splitData[j].trim() != '' && splitData[j].trim() != null && splitData[j].trim() != undefined) {
+								tableErrorData.push(splitData[j].trim());
+							}
+						}
+					}
+				
+				}else{//데이터가 탭으로 구분되어 이름, 연락처로 구분된 경우
+					
+					var phone = "";	//핸드폰번호
+					var name = "";	//이름
+					var rep1 = "";	//치환문자1
+					var rep2 = "";	//치환문자2
+					var rep3 = "";	//치환문자3
+					var rep4 = "";	//치환문자4
+					var isPhoneValid = false;
+					for(var j=0; j < splitData.length; j++){
+
+						if(j == 0){
+							
+							name = splitData[j].trim();
+							
+						}
+						
+						if(j == 1){
+							
+							if(checkHpNum(splitData[j].trim())){//핸드폰 양식이 맞는지 확인
+								phone = removeDash(splitData[j].trim());
+								isPhoneValid = true;
+							}else{
+								if (splitData[j].trim() != '' && splitData[j].trim() != null && splitData[j].trim() != undefined) {
+									tableErrorData.push(splitData[j].trim());
+								}
+							}
+							
+						}
+						
+						if(j == 2){	//치환문자1
+							
+							rep1 = splitData[j].trim();
+							
+						}
+						
+						if(j == 3){	//치환문자2
+							
+							rep2 = splitData[j].trim();
+							
+						}
+						
+						if(j == 4){	//치환문자3
+							
+							rep3 = splitData[j].trim();
+							
+						}
+						
+						if(j == 5){	//치환문자4
+							
+							rep4 = splitData[j].trim();
+							
+						}
+						
+					}
+					
+					if (isPhoneValid == true) {
+						//배열 끝에 데이터 추가해 주기
+						realPhone.push({phone: phone, name: name, rep1 : rep1, rep2 : rep2, rep3 : rep3, rep4 : rep4 });
+					}
+					
+				}//else end
+				  
+			}
+			
+			var recTableData = tableL.getRows();		 // 받는사람 리스트의 전체 목록을 저장
+			var tableData = [];
+			
+			//기존 받는사람 리스트를 배열에 미리 담아둔다.
+			if(recTableData.length > 0){
+				
+				for(var j=0; j < recTableData.length; j++){
+					
+					tableData.push({phone: removeDash(recTableData[j].getData().phone.trim()) , name: recTableData[j].getData().name, rep1: recTableData[j].getData().rep1, rep2: recTableData[j].getData().rep2, rep3: recTableData[j].getData().rep3, rep4: recTableData[j].getData().rep4,  });
+				
+				}
+			
+			}
+			
+			if(realPhone.length > 0){
+				
+				for(var j=0; j < realPhone.length; j++){
+					
+					tableData.push({phone: removeDash(realPhone[j].phone.trim()) , name: realPhone[j].name, rep1: realPhone[j].rep1, rep2: realPhone[j].rep2, rep3: realPhone[j].rep3, rep4: realPhone[j].rep4});
+				
+				}
+			
+			}
+			
+			//tableData.push(realPhone);
+				
+			//중복 연락처 1개만 남기고 삭제하기
+			var removeDuplPhone = dupliPhoneData(tableData);
+			  
+			//수신자 리스트에 전화번호 추가해주기
+			//tableL.addData(removeDuplPhone);
+			tableL.setData(removeDuplPhone);
+			
+			totRows = tableL.getRows().length; 
+			updateTotCnt(totRows); //전체 데이터 갯수 구하기
+			
+			var smsTxtArea = $('#smsTxtArea').val();
+			
+			//일괄변환 문구 결제금액 처리
+			if(smsTxtArea.indexOf("[*이름*]") > -1
+					|| smsTxtArea.indexOf("[*1*]") > -1
+					|| smsTxtArea.indexOf("[*2*]") > -1
+					|| smsTxtArea.indexOf("[*3*]") > -1
+					|| smsTxtArea.indexOf("[*4*]") > -1){
+				
+				fnReplCell_advc();
+			}else{
+				//결제 금액 구하기
+//			    totalPriceSum(totRows);
+				// 붙여넣기 했을때 금액 구하는 함수
+				totalFtPriceSum(totRows);
+			}
+
+			if (tableErrorData.length > 0) {
+				alert("올바르지 않은 휴대폰 번호가 "+ tableErrorData.length +" 건 있습니다.");
+			}			
+			
+
+			// 미리보기 버튼 활성화
+//			updateButtons(0);
+
+			
+		}
+
+	});
+
+	
+	
+	
+	
+	
+	
+	
+	
+	
+	
+	
+	
 });
 
 var tableErrorData = [];
@@ -506,9 +768,9 @@
 * 토탈 카운트 화면에 노출
 * 변수 없는 리스트만 체크
 */ 
-function updateTotCnt(){
+function updateTotCnt(cnt){
 	
-	$("#rowTotCnt").text($('#wrap01_body .list_body').length);
+	$("#rowTotCnt").text(cnt);
 
 }
 
@@ -867,3 +1129,8 @@
 	}
 	
 });
+
+
+
+
+
 
src/main/webapp/js/kakao/ft/templateComm.js (added)
+++ src/main/webapp/js/kakao/ft/templateComm.js
@@ -0,0 +1,127 @@
+/**
+ * 알림톡 템플릿 내용 글자수 체크 및 표시, 미리보기 내용 표시
+ * 
+ * 
+ */
+
+function setContentsLeng(contents){
+	
+
+	if ($("#ad_Y").is(":checked")) {
+		contents = "(광고)"+contents+"\n무료거부 0808800858"
+	}
+	
+	var conLeng = strMaxCharacterCnt(contents);
+
+	if ($("#ad_Y").is(":checked")) {
+		// 광고문구 제거
+		contents = contents
+			.replace(/^\(광고\)/, '')                          // 맨 앞의 (광고) 제거
+			.replace(/\n무료거부\s*0808800858\s*$/, '');      // 맨 뒤의 '무료거부 0808800858' 제거
+
+	}
+	
+	
+	if(conLeng > 1000){
+		
+		alert("알림톡 내용은 1000자를 넘을 수 없습니다.");
+		var splicecon = strMaxLengthSubstring(contents, 999);
+		 $('#inputTemplateContent').val(splicecon);
+		return false;
+		
+	}else{//현재 입력한 글자수 우측 하단에 표시해 주기
+		
+		var repContent = "";
+		repContent = contents.replace(/(?:\r\n|\r|\n)/g, '<br/>');
+		
+		$('.nowChar').text(conLeng + " /");
+		$('.template_text').show();
+		if(repContent != ''){
+			$('.template_text').html(repContent);
+		}else{
+			$('.template_text').html("내용 미리보기");
+		}
+		
+	}
+	
+}
+
+/**
+ * 친구톡 템플릿 내용 글자수 체크 및 표시, 미리보기 내용 표시
+ * 
+ * 
+ */
+
+function setContentsLengForFriends(contents){
+	
+
+//	if ($("#ad_Y").is(":checked")) {
+//		contents = "(광고)"+contents+"\n무료거부 0808800858"
+//	}
+	
+	var conLeng = strMaxCharacterCnt(contents);
+
+//	if ($("#ad_Y").is(":checked")) {
+//		// 광고문구 제거
+//		contents = contents
+//			.replace(/^\(광고\)/, '')                          // 맨 앞의 (광고) 제거
+//			.replace(/\n무료거부\s*0808800858\s*$/, '');      // 맨 뒤의 '무료거부 0808800858' 제거
+//
+//	}
+	
+	
+	var imageType = $("input[name=img_file_add]:checked").val();
+	var limitLeng = 1000;
+	
+	if(imageType == 'I'){
+		
+		limitLeng = 400;
+		
+		if(conLeng > limitLeng){
+			
+			alert("일반 이미지 첨부시 친구톡 내용은 400자를 넘을 수 없습니다.");
+			
+		}
+		
+	}else if(imageType == 'W'){
+		
+		limitLeng = 76;
+		
+		if(conLeng > limitLeng){
+			
+			alert("와이드 이미지 첨부시 친구톡 내용은 76자를 넘을 수 없습니다.");
+			
+		}
+		
+	}else if(conLeng > 1000){
+		
+		alert("친구톡 내용은 1000자를 넘을 수 없습니다.");
+		
+	}
+	
+	//제한글자수를 넘겼을 경우 최대 글자수까지 잘라주고 미리보기 및 글자수 표시처리
+	if(conLeng > limitLeng){
+		
+		var splicecon = strMaxLengthSubstring(contents, limitLeng-1);
+		 $('#inputTemplateContent').val(splicecon);
+		contents = splicecon;
+		conLeng = strMaxCharacterCnt(contents);
+		
+	}
+	
+	var repContent = "";
+	repContent = contents.replace(/(?:\r\n|\r|\n)/g, '<br/>');
+	
+	$('.nowChar').text(conLeng + " /");
+	$('.totChar').text(" "+limitLeng);
+	$('.template_text').show();
+	if(repContent != ''){
+		$('.template_text').html(repContent);
+		$('#smsTxtArea').val(
+				$('.template_text').html().trim().replace(/(<br>|<br\/>|<br \/>)/g, '\r\n')
+			);
+	}else{
+		$('.template_text').html("내용 미리보기");
+	}
+	
+}(No newline at end of file)
src/main/webapp/publish/css/content.css
--- src/main/webapp/publish/css/content.css
+++ src/main/webapp/publish/css/content.css
@@ -1652,18 +1652,18 @@
 .kakao_wrap .img_add_info_wrap .info_text{font-size: 15px; color: #555;}
 .kakao_wrap .img_add_info_wrap .info_text li{margin: 3px 0 0 0;}
 .kakao_wrap .img_file_add_wrap .info_title_text{font-weight: 500;margin:0 0 5px 0;}
-.kakao_wrap .img_file_add_wrap .btn_img_upload{width: 251px;}
-.kakao_wrap .img_file_add_wrap .img_file_info_wrap{width: 100%; max-width: 623px; border: 1px solid #e5e5e5; border-radius: 5px;}
-.kakao_wrap .img_file_add_wrap .img_file_info_wrap li{display: flex;min-height: 60px;padding: 10px 20px; border-bottom: 1px solid #e5e5e5;justify-content: space-between; align-items: center; box-sizing: border-box;}
-.kakao_wrap .img_file_add_wrap .img_file_info_wrap li:last-child{border-bottom: 0;}
-.kakao_wrap .img_file_add_wrap .file_name{font-size: 16px; font-weight: 500;}
-.kakao_wrap .img_file_add_wrap .btn_del{width: 30px;height: 30px;border: 1px solid #002c9a;border-radius: 5px;}
-.kakao_wrap .img_file_add_wrap .img_url{width: calc(100% - 190px);}
-.kakao_wrap .img_file_add_wrap .info_text{margin:0 0 10px 0;}
-.kakao_wrap .img_file_add_wrap .info_text li{list-style:1.4;}
-.kakao_wrap .img_file_add_wrap .img_file_wrap{display:inline-flex;width:calc(100% - 200px);}
-.kakao_wrap .img_file_add_wrap .img_file_wrap .file_name{width:calc(100% - 28px);}
-.kakao_wrap .img_file_add_wrap .img_file_wrap .file_name.file_none{color:#666;font-weight:300;line-height:1.8;}
+.kakao_wrap .img_file_info_wrap .btn_img_upload{width: 251px;}
+.kakao_wrap .img_file_info_wrap{display:none;width: 100%; max-width: 623px; border: 1px solid #e5e5e5; border-radius: 5px;margin:15px 0 0 0;}
+.kakao_wrap .img_file_info_wrap li{display: flex;min-height: 60px;padding: 10px 20px; border-bottom: 1px solid #e5e5e5;justify-content: space-between; align-items: center; box-sizing: border-box;}
+.kakao_wrap .img_file_info_wrap li:last-child{border-bottom: 0;}
+.kakao_wrap .img_file_info_wrap .file_name{font-size: 16px; font-weight: 500;}
+.kakao_wrap .img_file_info_wrap .btn_del{width: 30px;height: 30px;border: 1px solid #002c9a;border-radius: 5px;}
+.kakao_wrap .img_file_info_wrap .img_url{width: calc(100% - 190px);}
+.kakao_wrap .img_file_info_wrap .info_text{margin:0 0 10px 0;}
+.kakao_wrap .img_file_info_wrap .info_text li{list-style:1.4;}
+.kakao_wrap .img_file_info_wrap .img_file_wrap{display:inline-flex;width:calc(100% - 200px);}
+.kakao_wrap .img_file_info_wrap .img_file_wrap .file_name{width:calc(100% - 28px);}
+.kakao_wrap .img_file_info_wrap .img_file_wrap .file_name.file_none{color:#666;font-weight:300;line-height:1.8;}
 
 .kakaotalkset_cont .kakao_wrap .template_category{width: 270px;}
 .kakaotalkset_cont .kakao_wrap .emphasis_select{width: 250px;}
src/main/webapp/publish/js/content.js
--- src/main/webapp/publish/js/content.js
+++ src/main/webapp/publish/js/content.js
@@ -1,3 +1,4 @@
+
 $(document).ready(function () {
 
 	// 문자작성 영역 클릭시 textarea 활성화
@@ -522,15 +523,15 @@
 	$("input[name='img_file_add']").change(function () {
 		if ($("input[name='img_file_add']:checked").attr("id") == "img_file_1") {
 			$(".basic_img_add_wrap").show().siblings(".wide_img_add_wrap").hide();
-			//$(".img_file_add_wrap").show();
+			$(".img_file_info_wrap").show();
 			$(".kakao_image img").show().attr("src", "/publish/images/content/kakao_img_basic.jpg");
 		} else if ($("input[name='img_file_add']:checked").attr("id") == "img_file_2") {
 			$(".wide_img_add_wrap").show().siblings(".basic_img_add_wrap").hide();
 			$(".kakao_image img").show().attr("src", "/publish/images/content/kakao_img_wide.jpg");
-			//$(".img_file_add_wrap").show();
+			$(".img_file_info_wrap").show();
 		} else {
 			$(".img_add_info_wrap, .img_file_add_wrap").hide();
-			$(".kakao_image img").hide();
+			$(".kakao_image img, .img_file_info_wrap").hide();
 		}
 	});
 
@@ -1661,31 +1662,45 @@
 	}
 	var buttonText;
 	var buttonView;
-	//템플릿 버튼은 buttonVO의 buttonVOList 변수에 셋팅해서 넘겨주게 만들었음
+	//템플릿 버튼은 buttonVO의 buttonVOList 변수에 셋팅해서 넘겨주게 만들었음 
 	if (buttonType == "button_type_1") {
-		buttonText = '<dl class="button_type_wrap type1"><dt>배송조회</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeDeliv" name="buttonVOList[' + buttonCnt + '].linkType" value="DS"/><input type="text" id="btnNmDeliv" name="buttonVOList[' + buttonCnt + '].name" value="배송조회" placeholder="버튼명 입력(최대 14자)" readonly><p class="cf_text">*이용가능 택배사 : KG로지스, 우체국택배,일양로지스, GTX로지스, FedEx, 경동택배, 합동택배, 롯데택배</p></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type1"><dt>배송조회</dt><dd class="button_type_input">'
+				+'<input type="hidden" id="buttonLikeTypeDeliv" name="buttonVOList[' + buttonCnt + '].linkType" value="DS"/>'
+				+'<input type="text" id="btnNmDeliv" name="buttonVOList[' + buttonCnt + '].name" value="배송조회" placeholder="버튼명 입력(최대 14자)" readonly><p class="cf_text">*이용가능 택배사 : KG로지스, 우체국택배,일양로지스, GTX로지스, FedEx, 경동택배, 합동택배, 롯데택배</p></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewDS">배송조회</button>';
 	} else if (buttonType == "button_type_2") {
-		buttonText = '<dl class="button_type_wrap type2"><dt>웹링크</dt><dd class="button_type_input"><ul><li><input type="hidden" id="buttonLikeTypeWeb" name="buttonVOList[' + buttonCnt + '].linkType" value="WL"/><input type="text" id="btnNmWeb" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></li><li><input type="text" id="buttonLinkMo" name="buttonVOList[' + buttonCnt + '].linkMo" value="" placeholder="모바일 링크 입력"></li><li><input type="text" id="buttonLinkPc" name="buttonVOList[' + buttonCnt + '].linkPc" value="" placeholder="PC 링크 입력"></li></ul></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type2"><dt>웹링크</dt><dd class="button_type_input">		'
+				+'<ul><li><input type="hidden" id="buttonLikeTypeWeb" name="buttonVOList[' + buttonCnt + '].linkType" value="WL"/>'
+				+'<input type="text" class="kakaoBtnNmList" id="btnNmWeb" name="buttonVOList[' + buttonCnt + '].name" value="" maxlength="14" placeholder="버튼명 입력(최대 14자)"></li>'
+				+'<li><input type="text" class="kakaoBtnList" id="buttonLinkMo" name="buttonVOList[' + buttonCnt + '].linkMo" value="" placeholder="모바일 링크 입력"></li>'
+				+'<li><input type="text" class="kakaoBtnList" id="buttonLinkPc" name="buttonVOList[' + buttonCnt + '].linkPc" value="" placeholder="PC 링크 입력"></li></ul></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewWL">웹링크</button>';
 	} else if (buttonType == "button_type_3") {
-		buttonText = '<dl class="button_type_wrap type3"><dt>앱링크</dt><dd class="button_type_input"><ul><li><input type="hidden" id="buttonLikeTypeApp" name="buttonVOList[' + buttonCnt + '].linkType" value="AL"/><input type="text" id="btnNmApp" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></li><li><input type="text" id="buttonLinkAnd" name="buttonVOList[' + buttonCnt + '].linkAnd" value="" placeholder="Android 링크 입력"></li><li><input type="text" id="buttonLinkIos" name="buttonVOList[' + buttonCnt + '].linkIos" value="" placeholder="IOS 링크 입력"></li></ul></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type3"><dt>앱링크</dt><dd class="button_type_input">'
+				+'<ul><li><input type="hidden" id="buttonLikeTypeApp" name="buttonVOList[' + buttonCnt + '].linkType" value="AL"/>'
+				+'<input type="text" class="kakaoBtnNmList" id="btnNmApp" name="buttonVOList[' + buttonCnt + '].name" value="" maxlength="14" placeholder="버튼명 입력(최대 14자)"></li>'
+				+'<li><input type="text" class="kakaoBtnList" id="buttonLinkAnd" name="buttonVOList[' + buttonCnt + '].linkAnd" value="" placeholder="Android 링크 입력"></li>	'
+				+'<li><input type="text" class="kakaoBtnList" id="buttonLinkIos" name="buttonVOList[' + buttonCnt + '].linkIos" value="" placeholder="IOS 링크 입력"></li></ul></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewAL">앱링크</button>';
 	} else if (buttonType == "button_type_4") {
-		buttonText = '<dl class="button_type_wrap type4"><dt>봇키워드</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeBot" name="buttonVOList[' + buttonCnt + '].linkType" value="BK"/><input type="text" id="btnNmBot" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type4"><dt>봇키워드</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeBot" name="buttonVOList[' + buttonCnt + '].linkType" value="BK"/>'
+				+'<input type="text" id="btnNmBot" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewBK">봇키워드</button>';
 	} else if (buttonType == "button_type_5") {
-		buttonText = '<dl class="button_type_wrap type5"><dt>메시지전달</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeMsg" name="buttonVOList[' + buttonCnt + '].linkType" value="MD"/><input type="text" id="btnNmMsg" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type5"><dt>메시지전달</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeMsg" name="buttonVOList[' + buttonCnt + '].linkType" value="MD"/>'
+				+'<input type="text" id="btnNmMsg" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewMD">메시지전달</button>';
 	} else if (buttonType == "button_type_6") {
-		buttonText = '<dl class="button_type_wrap type6"><dt>상담톡전환</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeCons" name="buttonVOList[' + buttonCnt + '].linkType" value="BC"/><input type="text" id="btnNmCons" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type6"><dt>상담톡전환</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeCons" name="buttonVOList[' + buttonCnt + '].linkType" value="BC"/>'
+				+'<input type="text" id="btnNmCons" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewBC">상담톡전환</button>';
 	} else if (buttonType == "button_type_7") {
-		buttonText = '<dl class="button_type_wrap type7"><dt>챗봇전환</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeBotChg" name="buttonVOList[' + buttonCnt + '].linkType" value="BT"/><input type="text" id="btnNmBotChg" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
+		buttonText = '<dl class="button_type_wrap type7"><dt>챗봇전환</dt><dd class="button_type_input"><input type="hidden" id="buttonLikeTypeBotChg" name="buttonVOList[' + buttonCnt + '].linkType" value="BT"/>'
+				+'<input type="text" id="btnNmBotChg" name="buttonVOList[' + buttonCnt + '].name" value="" placeholder="버튼명 입력(최대 14자)"></dd><dd><button type="button" class="btn_del" onclick="buttonTypeDel(this);"><img src="/publish/images/btn_delete.png" alt=""></button></dd></dl>';
 		buttonView = '<button type="button" class="btn_kakao_type btnEmpty" id="btnViewBT">챗봇전환</button>';
 	} else {}
-	$(".button_add_wrap").prepend(buttonText);
-	$(".btnViewArea").prepend(buttonView);
+	$(".button_add_wrap").append(buttonText);
+	$(".btnViewArea").append(buttonView);
 }
 
 //관리자 카카오톡 알림톡 샘플 템플릿 등록 - 버튼 추가
src/main/webapp/publish/kakao_friendtalk_send.html
--- src/main/webapp/publish/kakao_friendtalk_send.html
+++ src/main/webapp/publish/kakao_friendtalk_send.html
@@ -101,6 +101,7 @@
                                             <label for="img_file_1">이미지 첨부</label>
                                             <input type="radio" name="img_file_add" id="img_file_2">
                                             <label for="img_file_2">와이드 이미지 첨부</label>
+                                            
                                             <div class="img_file_add_wrap basic_img_add_wrap">
                                                 <p class="info_title_text"><span class="c_e40000">*</span> 이미지 첨부 안내</p>
                                                 <ul class="info_text">
src/main/webapp/publish/textingmsg_detail_friendtalk.html
--- src/main/webapp/publish/textingmsg_detail_friendtalk.html
+++ src/main/webapp/publish/textingmsg_detail_friendtalk.html
@@ -732,11 +732,13 @@
 											<div class="text_preview">
 		                                        <p class="friend_talk_title"><span>(광고)</span></p>
 		                                        <div class="allimtalk_content">
-		                                            <div class="kakao_image"><img src="/publish/images/content/kakao_img_basic.jpg" alt="" style="display:none;"></div>
+		                                        
+		                                            <div class="kakao_image"><img src="/publish/images/content/kakao_img_basic.jpg" alt=""></div>
 		                                            <p class="template_text">홍길동 고객님 안녕하세요.
 		                                                문자온입니다.
 		                                                문자온에서는 재결제 고객님들께 다양한 이벤트를 준비하였으니, 방문하셔서 혜택 놓치지 마세요.</p>
 		                                            <button type="button" class="btn_kakao_type">문자온 바로가기</button>
+		                                        
 		                                        </div>
 		                                        <p class="kakao_block_text">수신거부 : 홈 > 채널차단</p>
 		                                    </div>
Add a comment
List