on
[Python] 파이썬 HTML로 카카오맵 해상도 높은 큰 지도 만들기
[Python] 파이썬 HTML로 카카오맵 해상도 높은 큰 지도 만들기
반응형
부동산과 관련하여 지도를 확대해도, 축소해도 해상도가 높은 큰지도를 필요할 때가 있습니다.
특히, 축소해도 여전히 해상도가 높은 10,000px * 10,000px 의 큰 이미지가 필요할때 사용할 수 있겠습니다.
<예시>
1. 카카오맵 api KEY 발급
https://apis.map.kakao.com/web/
먼저, 카카오맵 api의 KEY를 발급을 받습니다.
우리가 사용할 키는 "JavaScript 키" 입니다.
플랫폼-Web
사이트 도메인에 http://localhost:8080 을 입력해줍니다.
(나중에 혹시 홈페이지에 넣을 필요가 있을 때, 본인의 사이트 도메인을 추가합니다.)
2. HTML 코드 작성
var mapContainer = document.getElementById('map'), // 지도를 표시할 div mapOption = { center: new kakao.maps.LatLng(37.512855, 126.940243), // 지도의 중심좌표 level: 1, // 지도의 확대 레벨 mapTypeId : kakao.maps.MapTypeId.SKYVIEW // 지도종류 }; // 지도를 생성한다 var map = new kakao.maps.Map(mapContainer, mapOption);
메모장에 위 코드를 입력하고
5줄: 원하는 지도 크기를 수정하고
8줄: 발급받은 Javascrip키를 수정하고
13줄: 큰지도의 위치를 위도와 경도로 입력해줍니다.
14줄: 확대레벨은 해상도를 가장 높게 하기 위하여 1로 설정합니다.
kakaomap.html 확장자로 저장해줍니다.
https://apis.map.kakao.com/web/sample/
더욱 다양한 코드와 설명은 위 카카오 공식 홈페이지를 확인해봅시다.
3. 로컬 서버 실행
처음 카카오api 플랫폼에서 사이트 도메인을 http://localhost:8080 으로 설정하였습니다.
https://sjblog1.tistory.com/39
이곳에서 파일을 다운로드 받거나 로컬 서버를 실행시켜줍니다.
4. 카카오맵 큰지도 실행
크롬 주소창에 http://localhost:8080 을 입력해줍니다.
여기서, 작성했던 코드를 저장한, kakaomap.html을 찾아 클릭합니다.
https://chrome.google.com/webstore/detail/gofullpage-full-page-scre/fdpohaocaechififmbbbbbknoalclacl?hl=ko
전체 웹페이지를 캡쳐할 수 있는 크롬 확장프로그램을 설치하여 이미지를 캡쳐합니다.
카카오맵 큰지도를 얻게 되었습니다.
5. 매번 지도 크기, 위도경도 찾아서 수정하기 귀찮다?
내가 원하는 위치의 위도, 경도를 검색하기 귀찮고,
큰지도 크기도 매번 메모장으로 수정하기 귀찮다!
파이썬으로 EXE 실행파일을 만들어 실행만 하면 옵션을 선택할 수 있게 사용해보자!
이후 개인홈페이지를 만들 때 참고할 수 있겠습니다!
javascript_key 변수 2곳만 입력해주면 됩니다.
이후 pyinstaller 을 사용하여 EXE 실행파일을 만들어 줍니다.
샘플 EXE 실행파일을 제공해 드리고 싶었으나 javascript key 문제로 부득이하게 제공을 못해드립니다.
import requests from flask import Flask, render_template import os import getpass import threading import webbrowser import sys #1.html 생성 def getKakaoMap1HTML(): #키입력 javascript_key = "" result = "" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + " " + "
" result = result + " " + "
" result = result + " 키워드로 장소검색하고 목록으로 표출하기" + "
" result = result + "
" result = result + ".map_wrap, .map_wrap * {margin:0;padding:0;font-family:'Malgun Gothic',dotum,'돋움',sans-serif;font-size:12px;}" + "
" result = result + ".map_wrap a, .map_wrap a:hover, .map_wrap a:active{color:#000;text-decoration: none;}" + "
" result = result + ".map_wrap {position:relative;width:100%;height:500px;}" + "
" result = result + "#menu_wrap {position:absolute;top:0;left:0;bottom:0;width:250px;margin:10px 0 30px 10px;padding:5px;overflow-y:auto;background:rgba(255, 255, 255, 0.7);z-index: 1;font-size:12px;border-radius: 10px;}" + "
" result = result + ".bg_white {background:#fff;}" + "
" result = result + "#menu_wrap hr {display: block; height: 1px;border: 0; border-top: 2px solid #5F5F5F;margin:3px 0;}" + "
" result = result + "#menu_wrap .option{text-align: center;}" + "
" result = result + "#menu_wrap .option p {margin:10px 0;} " + "
" result = result + "#menu_wrap .option button {margin-left:5px;}" + "
" result = result + "#placesList li {list-style: none;}" + "
" result = result + "#placesList .item {position:relative;border-bottom:1px solid #888;overflow: hidden;cursor: pointer;min-height: 65px;}" + "
" result = result + "#placesList .item span {display: block;margin-top:4px;}" + "
" result = result + "#placesList .item h5, #placesList .item .info {text-overflow: ellipsis;overflow: hidden;white-space: nowrap;}" + "
" result = result + "#placesList .item .info{padding:10px 0 10px 55px;}" + "
" result = result + "#placesList .info .gray {color:#8a8a8a;}" + "
" result = result + "#placesList .info .jibun {padding-left:26px;background:url(https://t1.daumcdn.net/localimg/localimages/07/mapapidoc/places_jibun.png) no-repeat;}" + "
" result = result + "#placesList .info .tel {color:#009900;}" + "
" result = result + "#placesList .item .markerbg {float:left;position:absolute;width:36px; height:37px;margin:10px 0 0 10px;background:url(https://t1.daumcdn.net/localimg/localimages/07/mapapidoc/marker_number_blue.png) no-repeat;}" + "
" result = result + "#placesList .item .marker_1 {background-position: 0 -10px;}" + "
" result = result + "#placesList .item .marker_2 {background-position: 0 -56px;}" + "
" result = result + "#placesList .item .marker_3 {background-position: 0 -102px}" + "
" result = result + "#placesList .item .marker_4 {background-position: 0 -148px;}" + "
" result = result + "#placesList .item .marker_5 {background-position: 0 -194px;}" + "
" result = result + "#placesList .item .marker_6 {background-position: 0 -240px;}" + "
" result = result + "#placesList .item .marker_7 {background-position: 0 -286px;}" + "
" result = result + "#placesList .item .marker_8 {background-position: 0 -332px;}" + "
" result = result + "#placesList .item .marker_9 {background-position: 0 -378px;}" + "
" result = result + "#placesList .item .marker_10 {background-position: 0 -423px;}" + "
" result = result + "#placesList .item .marker_11 {background-position: 0 -470px;}" + "
" result = result + "#placesList .item .marker_12 {background-position: 0 -516px;}" + "
" result = result + "#placesList .item .marker_13 {background-position: 0 -562px;}" + "
" result = result + "#placesList .item .marker_14 {background-position: 0 -608px;}" + "
" result = result + "#placesList .item .marker_15 {background-position: 0 -654px;}" + "
" result = result + "#pagination {margin:10px auto;text-align: center;}" + "
" result = result + "#pagination a {display:inline-block;margin-right:10px;}" + "
" result = result + "#pagination .on {font-weight: bold; cursor: default;color:#777;}" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " 키워드 : " + "
" result = result + " 검색하기 " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + " " + "
" result = result + "사이즈: " + "
" result = result + " X " + "
" result = result + "줌레벨: 1234567891011121314" + "
" result = result + "확인" + "
" result = result + "" + "
" result = result + "
" result = result + "// 마커를 담을 배열입니다" + "
" result = result + "var markers = [];" + "
" result = result + "var mapContainer = document.getElementById('map'), // 지도를 표시할 div " + "
" result = result + " mapOption = {" + "
" result = result + " center: new kakao.maps.LatLng(37.566826, 126.9786567), // 지도의 중심좌표" + "
" result = result + " level: 3 // 지도의 확대 레벨" + "
" result = result + " }; " + "
" result = result + "// 지도를 생성합니다" + "
" result = result + "var map = new kakao.maps.Map(mapContainer, mapOption); " + "
" result = result + "// 장소 검색 객체를 생성합니다" + "
" result = result + "var ps = new kakao.maps.services.Places(); " + "
" result = result + "// 검색 결과 목록이나 마커를 클릭했을 때 장소명을 표출할 인포윈도우를 생성합니다" + "
" result = result + "var infowindow = new kakao.maps.InfoWindow({zIndex:1});" + "
" result = result + "// 키워드로 장소를 검색합니다" + "
" result = result + "searchPlaces();" + "
" result = result + "// 키워드 검색을 요청하는 함수입니다" + "
" result = result + "function searchPlaces() {" + "
" result = result + " var keyword = document.getElementById('keyword').value;" + "
" result = result + " if (!keyword.replace(/^\s+|\s+$/g, '')) {" + "
" result = result + " alert('키워드를 입력해주세요!');" + "
" result = result + " return false;" + "
" result = result + " }" + "
" result = result + " // 장소검색 객체를 통해 키워드로 장소검색을 요청합니다" + "
" result = result + " ps.keywordSearch( keyword, placesSearchCB); " + "
" result = result + "}" + "
" result = result + "// 장소검색이 완료됐을 때 호출되는 콜백함수 입니다" + "
" result = result + "function placesSearchCB(data, status, pagination) {" + "
" result = result + " if (status === kakao.maps.services.Status.OK) {" + "
" result = result + " // 정상적으로 검색이 완료됐으면" + "
" result = result + " // 검색 목록과 마커를 표출합니다" + "
" result = result + " displayPlaces(data);" + "
" result = result + " // 페이지 번호를 표출합니다" + "
" result = result + " displayPagination(pagination);" + "
" result = result + " } else if (status === kakao.maps.services.Status.ZERO_RESULT) {" + "
" result = result + " alert('검색 결과가 존재하지 않습니다.');" + "
" result = result + " return;" + "
" result = result + " } else if (status === kakao.maps.services.Status.ERROR) {" + "
" result = result + " alert('검색 결과 중 오류가 발생했습니다.');" + "
" result = result + " return;" + "
" result = result + " }" + "
" result = result + "}" + "
" result = result + "// 검색 결과 목록과 마커를 표출하는 함수입니다" + "
" result = result + "function displayPlaces(places) {" + "
" result = result + " var listEl = document.getElementById('placesList'), " + "
" result = result + " menuEl = document.getElementById('menu_wrap')," + "
" result = result + " fragment = document.createDocumentFragment(), " + "
" result = result + " bounds = new kakao.maps.LatLngBounds(), " + "
" result = result + " listStr = '';" + "
" result = result + " // 검색 결과 목록에 추가된 항목들을 제거합니다" + "
" result = result + " removeAllChildNods(listEl);" + "
" result = result + " // 지도에 표시되고 있는 마커를 제거합니다" + "
" result = result + " removeMarker();" + "
" result = result + " for ( var i=0; i
" result = result + " // 마커를 생성하고 지도에 표시합니다" + "
" result = result + " var placePosition = new kakao.maps.LatLng(places[i].y, places[i].x)," + "
" result = result + " marker = addMarker(placePosition, i), " + "
" result = result + " itemEl = getListItem(i, places[i]); // 검색 결과 항목 Element를 생성합니다" + "
" result = result + " // 검색된 장소 위치를 기준으로 지도 범위를 재설정하기위해" + "
" result = result + " // LatLngBounds 객체에 좌표를 추가합니다" + "
" result = result + " bounds.extend(placePosition);" + "
" result = result + " // 마커와 검색결과 항목에 mouseover 했을때" + "
" result = result + " // 해당 장소에 인포윈도우에 장소명을 표시합니다" + "
" result = result + " // mouseout 했을 때는 인포윈도우를 닫습니다" + "
" result = result + " (function(marker, title) {" + "
" result = result + " kakao.maps.event.addListener(marker, 'mouseover', function() {" + "
" result = result + " displayInfowindow(marker, title);" + "
" result = result + " });" + "
" result = result + " kakao.maps.event.addListener(marker, 'mouseout', function() {" + "
" result = result + " infowindow.close();" + "
" result = result + " });" + "
" result = result + " itemEl.onmouseover = function () {" + "
" result = result + " displayInfowindow(marker, title);" + "
" result = result + " };" + "
" result = result + " itemEl.onmouseout = function () {" + "
" result = result + " infowindow.close();" + "
" result = result + " };" + "
" result = result + " })(marker, places[i].place_name);" + "
" result = result + " fragment.appendChild(itemEl);" + "
" result = result + " }" + "
" result = result + " // 검색결과 항목들을 검색결과 목록 Elemnet에 추가합니다" + "
" result = result + " listEl.appendChild(fragment);" + "
" result = result + " menuEl.scrollTop = 0;" + "
" result = result + " // 검색된 장소 위치를 기준으로 지도 범위를 재설정합니다" + "
" result = result + " map.setBounds(bounds);" + "
" result = result + "}" + "
" result = result + "// 검색결과 항목을 Element로 반환하는 함수입니다" + "
" result = result + "function getListItem(index, places) {" + "
" result = result + " var el = document.createElement('li')," + "
" result = result + " itemStr = '' +" + "
" result = result + " '' +" + "
" result = result + " ' ' + places.place_name + '';" + "
" result = result + " if (places.road_address_name) {" + "
" result = result + " itemStr += ' ' + places.road_address_name + '' +" + "
" result = result + " ' ' + places.address_name + '';" + "
" result = result + " } else {" + "
" result = result + " itemStr += ' ' + places.address_name + ''; " + "
" result = result + " }" + "
" result = result + " itemStr += ' ' + places.phone + '' +" + "
" result = result + " '';" + "
" result = result + " el.innerHTML = itemStr;" + "
" result = result + " el.className = 'item';" + "
" result = result + " return el;" + "
" result = result + "}" + "
" result = result + "// 마커를 생성하고 지도 위에 마커를 표시하는 함수입니다" + "
" result = result + "function addMarker(position, idx, title) {" + "
" result = result + " var imageSrc = 'https://t1.daumcdn.net/localimg/localimages/07/mapapidoc/marker_number_blue.png', // 마커 이미지 url, 스프라이트 이미지를 씁니다" + "
" result = result + " imageSize = new kakao.maps.Size(36, 37), // 마커 이미지의 크기" + "
" result = result + " imgOptions = {" + "
" result = result + " spriteSize : new kakao.maps.Size(36, 691), // 스프라이트 이미지의 크기" + "
" result = result + " spriteOrigin : new kakao.maps.Point(0, (idx*46)+10), // 스프라이트 이미지 중 사용할 영역의 좌상단 좌표" + "
" result = result + " offset: new kakao.maps.Point(13, 37) // 마커 좌표에 일치시킬 이미지 내에서의 좌표" + "
" result = result + " }," + "
" result = result + " markerImage = new kakao.maps.MarkerImage(imageSrc, imageSize, imgOptions)," + "
" result = result + " marker = new kakao.maps.Marker({" + "
" result = result + " position: position, // 마커의 위치" + "
" result = result + " image: markerImage " + "
" result = result + " });" + "
" result = result + " marker.setMap(map); // 지도 위에 마커를 표출합니다" + "
" result = result + " markers.push(marker); // 배열에 생성된 마커를 추가합니다" + "
" result = result + " return marker;" + "
" result = result + "}" + "
" result = result + "// 지도 위에 표시되고 있는 마커를 모두 제거합니다" + "
" result = result + "function removeMarker() {" + "
" result = result + " for ( var i = 0; i < markers.length; i++ ) {" + "
" result = result + " markers[i].setMap(null);" + "
" result = result + " } " + "
" result = result + " markers = [];" + "
" result = result + "}" + "
" result = result + "// 검색결과 목록 하단에 페이지번호를 표시는 함수입니다" + "
" result = result + "function displayPagination(pagination) {" + "
" result = result + " var paginationEl = document.getElementById('pagination')," + "
" result = result + " fragment = document.createDocumentFragment()," + "
" result = result + " i; " + "
" result = result + " // 기존에 추가된 페이지번호를 삭제합니다" + "
" result = result + " while (paginationEl.hasChildNodes()) {" + "
" result = result + " paginationEl.removeChild (paginationEl.lastChild);" + "
" result = result + " }" + "
" result = result + " for (i=1; i<=pagination.last; i++) {" + "
" result = result + " var el = document.createElement('a');" + "
" result = result + " el.href = "#";" + "
" result = result + " el.innerHTML = i;" + "
" result = result + " if (i===pagination.current) {" + "
" result = result + " el.className = 'on';" + "
" result = result + " } else {" + "
" result = result + " el.onclick = (function(i) {" + "
" result = result + " return function() {" + "
" result = result + " pagination.gotoPage(i);" + "
" result = result + " }" + "
" result = result + " })(i);" + "
" result = result + " }" + "
" result = result + " fragment.appendChild(el);" + "
" result = result + " }" + "
" result = result + " paginationEl.appendChild(fragment);" + "
" result = result + "}" + "
" result = result + "// 검색결과 목록 또는 마커를 클릭했을 때 호출되는 함수입니다" + "
" result = result + "// 인포윈도우에 장소명을 표시합니다" + "
" result = result + "function displayInfowindow(marker, title) {" + "
" result = result + " var content = '' + title + '';" + "
" result = result + " infowindow.setContent(content);" + "
" result = result + "}" + "
" result = result + "// 검색결과 목록의 자식 Element를 제거하는 함수입니다" + "
" result = result + "function removeAllChildNods(el) { " + "
" result = result + " while (el.hasChildNodes()) {" + "
" result = result + " el.removeChild (el.lastChild);" + "
" result = result + " }" + "
" result = result + "}" + "
" result = result + "//지도 현재 중심좌표" + "
" result = result + "kakao.maps.event.addListener(map, 'center_changed', function() {" + "
" result = result + "// 지도의 레벨을 얻어옵니다" + "
" result = result + "var level = map.getLevel();" + "
" result = result + "// 지도의 중심좌표를 얻어옵니다 " + "
" result = result + "center = map.getCenter(); " + "
" result = result + "var message = '지도 레벨 ' + level + ' ';" + "
" result = result + "message += '위도 ' + center.getLat() + ', 경도 ' + center.getLng() + '';" + "
" result = result + "var resultDiv = document.getElementById('result');" + "
" result = result + "resultDiv.innerHTML = message;" + "
" result = result + "});" + "
" result = result + "var center = map.getCenter();" + "
" result = result + "//버튼클릭시" + "
" result = result + "function button1_click(){" + "
" result = result + " position= [center.getLng(),center.getLat()]" + "
" result = result + " var width = $('#width').val();" + "
" result = result + " var height = $('#width').val();" + "
" result = result + " var zoomlevel = $('#zoomlevel').val();" + "
" result = result + " location.href='/about?index=' + position + '&width;=' + width + '&height;=' + height + '&zoomlevel;=' + zoomlevel;" + "
" result = result + "}" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" return result #2.html 생성 def getKakaoMap2HTML(): #키입력 javascript_key = "" result = "" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "" + "
" result = result + "
" result = result + "//좌표전달받기" + "
" result = result + "var para = document.location.href.split('=');" + "
" result = result + "var posi = ((para[1]).split('&'))[0].split(',').map(Number); //[y,x]" + "
" result = result + "var width = Number((para[2].split('&'))[0]);" + "
" result = result + "var height = Number((para[3].split('&'))[0]);" + "
" result = result + "var zoomlevel = Number(para[4]);" + "
" result = result + "var style_1 = document.getElementById('map');" + "
" result = result + "style_1.style.width = width + 'px';" + "
" result = result + "style_1.style.height = height + 'px';" + "
" result = result + "var mapContainer = document.getElementById('map'), // 지도를 표시할 div " + "
" result = result + " mapOption = {" + "
" result = result + " center: new kakao.maps.LatLng(posi[1], posi[0]), // 지도의 중심좌표" + "
" result = result + " level: zoomlevel, // 지도의 확대 레벨" + "
" result = result + " mapTypeId : kakao.maps.MapTypeId.SKYVIEW // 지도종류" + "
" result = result + " }; " + "
" result = result + "// 지도를 생성한다 " + "
" result = result + "var map = new kakao.maps.Map(mapContainer, mapOption);" + "
" result = result + "" + "
" result = result + "" + "
" return result #pyinstaller 파일경로 찾기 def resource_path(relative_path): if hasattr(sys, '_MEIPASS'): return os.path.join(sys._MEIPASS, relative_path) else: return os.path.join(os.path.abspath("."), relative_path) #main map_1html = getKakaoMap1HTML() #1.html 생성 map_2html = getKakaoMap2HTML() #2.html 생성 #1.html 저장 work_path = resource_path('') #현재경로확인 print(work_path) #현재경로확인 os.makedirs(work_path + '\\templates', exist_ok = True) #현재경로에 폴더생성 with open(work_path + "\\templates\\1.html", 'w', encoding = 'UTF8') as html_1file: #HTML생성 html_1file.write(map_1html) #2.html 저장 with open(work_path + "\\templates\\2.html", 'w', encoding = 'UTF8') as html_2file: #HTML생성 html_2file.write(map_2html) #Flask 로컬호스트 app = Flask(__name__) @app.route('/') def index(): return render_template('1.html') @app.route('/about') def about(): return render_template('2.html') if __name__== "__main__": threading.Timer(0.50, lambda: webbrowser.open("http://localhost:{}".format(8080))).start() #시간지연실행 print() print("!!!!!!!!!!!!!!!!!Ctrl + C 를 눌러 종료하세요!!!!!!!!!!!!!!!!!") print("닫기로 종료시, C:\\Users\\사용자\\AppData\\Local\\Temp\\_MEI 랜덤숫자 폴더가 남습니다.") print() app.run(host="localhost", port="8080")
반응형
from http://sjblog1.tistory.com/40 by ccl(A) rewrite - 2021-11-18 13:27:20