올해는 머신러닝이다.
[펌]주변 배달 업체 위치 구하는 소스 본문
출처 : http://messi.adnaru.com/guide/guide04.html
튜토리얼 결과물 소개
튜토리얼의 최종 결과물입니다. 휴대전화를 사용하고 있는 사용자의 위치 주변에 치킨, 중국집, 보쌈, 피자등의 배달 음식점을 찾아주는 애플리케이션입니다. 이 튜토리얼은 다음 좌표계변환 API를 이용해 GPS 위도,경도정보를 주소 정보로 변환한 후에, 네이버 지역정보 API를 통해 주변 배달 음식점 정보를 받아오게 됩니다. 두개의 API를 매쉬업해 주변 배달 음식을 찾는 앱을 만드는 가이드를 따라해봅시다.
가이드를 시작하기에 앞서
이 가이드는 Messi 라이브러리를 이용해 다음 쇼핑 API를 기반으로 최저가 검색기 앱을 만드는 가이드를 담은 문서입니다. 가이드를 진행하시기 앞서 아래의 사항들이 필요합니다.
- ADT또는 안드로이드 스튜디오
- Android SDK(2.2 Froyo 이상)
- Messi 라이브러리
안드로이드를 처음 개발하는 개발자분들은, “복사-붙여넣기”코너는 그대로 소스코드를 복사해서 붙여넣으시면 편리합니다. 또한 최종적으로 개발된 코드 결과물을 압축파일로 제공하므로, 코딩하는 과정에서 어려움을 느끼신다면 해당 결과물 파일을 참고해주세요.
Open API 키 발급받기
다음 Open API
다음 DNA(http://dna.daum.net) 에서 로컬형 Open API키를 발행하여 주십시오. 기존에 발행한 키가 있다면 해당 키를 사용하셔도 무관합니다.
발행받을 키는 로컬형 key로 발행하여 주시고, 해당 key를 계속해서 사용하여야 하니 저장해두시는 편이 편리합니다.
또한 다음은 쇼핑, 검색, 컨텐츠, 로컬API키가 모두 다르므로, 특별히 유의해서 발급받아 주십시오.
네이버 Open API
네이버 개발자센터(http://developer.naver.com) 에서 Open API키를 발행하여 주십시오. 기존에 발행한 키가 있다면 해당 키를 사용하셔도 무관합니다.
발행받을 키는 검색 key로 발행하여 주시고, 해당 key를 계속해서 사용하여야 하니 저장해두시는 편이 편리합니다.
라이브러리 연동
안드로이드 프로젝트를 생성하여 주십시오. ADT혹은 안드로이드 스튜디오에서 새 프로젝트를 생성하여 주세요. 프로젝트에 설정된 SDK버전은 기존값을 따라가셔도 무관합니다.
libs에 라이브러리 넣기
해당 프로젝트의 libs 폴더에 messi.jar 파일을 끌어넣어 복사해주십시오. messi.jar 파일은 라이브러리 폴더 상단에 있습니다.
프로젝트와 라이브러리 연동
생성하신 프로젝트에서 messi 라이브러리를 사용하도록 설정하여야 합니다. ADT 기준으로 프로젝트 최상위 폴더에서 오른쪽 마우스를 누른 후 [Properties] – [Java Build path] – [Libraries]로 들어가 주십시오. 그리고 오른쪽에 있는 Add Jar을 눌러 앞서 붙여 넣기 한 라이브러리 파일을 추가해 주십시오.
권한 설정
Messi는 기본적으로 인터넷 연결 권한이 필요합니다. AndroidMenifest.xml에서 인터넷 권한을 추가해주십시오. 아울러 이 프로젝트에선 사용자의 위치정보를 가져오기 위해 위치정보를 사용할 권한과, 전화를 걸기 위한 권한이 필요합니다. 아래의 소스코드를 복사-붙여넣기 하셔도 무관합니다.
<uses-permission android:name="android.permission.CALL_PHONE"/> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/> <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/> <uses-permission android:name="android.permission.INTERNET"/>
레이아웃 구성
이제 화면 레이아웃을 구성해 봅시다. 레이아웃 구성은 자유이지만, 가이드라인을 따라하고 싶으면 아래의 내용을 참고하셔도 좋습니다. 레이아웃 구성은 메인페이지와 각 개별 아이템구성으로 나뉘어집니다.
메인페이지
이번 튜토리얼은 각 업종별 조회 버튼과 그 목록 결과를 받아오는 ListView로 구성되어있다고 생각해봅시다. 그리고 각 버튼을 눌렀을때, Messi를 이용하여 검색 결과값을 받아오는 화면 구성으로 진행됩니다. 화면 구성은 자유자재로 바꾸시면서 진행하셔도 무관합니다.
복사-붙여넣기 할때는 activity_main.xml 파일에 붙여넣기 해 주십시오.
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" tools:context="${relativePackage}.${activityClass}" > <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <Button android:id="@+id/btn1" android:layout_width="match_parent" android:layout_weight="1" android:layout_height="wrap_content" android:text="치킨" /> <Button android:id="@+id/btn2" android:layout_width="match_parent" android:layout_weight="1" android:layout_height="wrap_content" android:text="중국집" /> <Button android:id="@+id/btn3" android:layout_width="match_parent" android:layout_weight="1" android:layout_height="wrap_content" android:text="피자" /> <Button android:id="@+id/btn4" android:layout_width="match_parent" android:layout_weight="1" android:layout_height="wrap_content" android:text="족발" /> </LinearLayout> <ListView android:id="@+id/listView1" android:layout_width="match_parent" android:layout_height="wrap_content" > </ListView> </LinearLayout>
개별 아이템 구성
화면 구성을 모두 마쳤으면 가게별 목록 개별 아이템을 만들어봅시다. 이 가이드라인에는 가게의 이름, 주소, 전화번호 정보가 표시됩니다. 아래의 소스코드를 그대로 복사-붙여넣기 하시거나, 아니면 더 창의적이고 멋진 UI를 만드셔도 괜찮습니다. 복사-붙여넣기으로 진행하실 분들은 item_row.xml이라는 레이아웃 파일을 만드신 후 아래의 코드를 붙여넣기 해주세요.
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:id="@+id/item_tv1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="기사 제목" android:layout_marginTop="10dp" android:textAppearance="?android:attr/textAppearanceMedium" /> <TextView android:id="@+id/item_tv2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="기사 내용" android:layout_marginBottom="10dp" android:textAppearance="?android:attr/textAppearanceSmall" /> </LinearLayout>
API와 연동
레이아웃 구성을 마쳤으면, 이제 소스코드를 작성해봅시다. 이 단계에서는 messi를 활용하여 API를 연동하는 과정을 다룹니다. 모든 과정에 대한 소스코드는 아래에서 다운로드 하실 수 있습니다.
임포트
프로젝트의 메인이 되는 액티비티인 MainAcitivty.java파일로 들어가 주십시오. 그리고 아래의 소스코드를 import 부분에 삽입하여 주십시오.
import com.hhdd.messi.Daum; import com.hhdd.messi.Naver; import com.hhdd.messi.event.DaumEventListener; import com.hhdd.messi.event.NaverEventListener; import com.hhdd.messi.naver.object.search.LocalObject;
복사-붙여넣기로 진행하셔도 무관합니다. 위의 클래스들은, 다음과 네이버의 API 객체, 결과처리를 통보할 이벤트리스너, 결과값 검색 객체와 관련된 내용입니다.
이벤트 리스너 추가
Messi는 이벤트 기반으로 값을 주고받으며, API 호출 성공시에는 OnResult와 실패시에는 OnFault로 결과값을 보내줍니다. 이벤트 리스너와 연동하기 위해 이벤트 객체를 implements 합시다.
public class MainActivity extends Activity implements NaverEventListener.OnLocalListener
이벤트 객체를 implements 했다면, 결과값을 받아올 OnResult과 실패시 원인을 통보할 OnFalut 메서드가 필요합니다. 키보드에서 Alt키 다음 S키 다음 V키를 눌러 생성하거나, 아니면 아래의 코드를 Oncreate 함수 하단에 복사-붙여넣기하여 주십시오.
private String dong_name = ""; private Naver naver_api; private Daum daum_api; @Override public void onFaultResult(int arg0, String arg1) { // 실패시 원인과 결과값을 리턴 } @Override public void onResult(ArrayList<LocalObject> arg0) { // 성공시 결과목록을 ArrayList 제너릭 형태로 리턴 }
오픈API 객체 생성
OnCreate 메서드 아래에 Daum 객체를 new로 만들어주십시오. 그리고 이어서 네이버 객체는, 앞서 발급받은 API Key값을 설정하는 부분을 추가합시다. API Key값은 setSearchKey 메서드에 key 값을 넣으면 다시 설정할 필요 없이 반복해서 사용할 수 있습니다. 그리고 다음 객체에도 앞서 발급받은 로컬 API Key 값을 설정하여 주십시오. 다음의 로컬 API Key값은 setLocalKey 메서드로 설정할 수 있습니다. 복사-붙여넣기는 OnCreate 메서드 안에 해주세요.
naver_api = new Naver(); daum_api = new Daum(); daum_api.setLocalKey("다음 로컬API 키"); naver_api.setSearchKey("네이버 검색API 키"); naver_api.setLocalListener(this);
각 객체들을 초기화 하는 절차를 마쳤습니다. 이어서, GPS 정보를 받아오는 부분을 추가합시다. 사용자의 GPS 정보는 LocationManager 클래스를 통해 받아올 수 있습니다. 이 튜토리얼에서는 사용자의 네트워크를 기준으로 위치를 찾아보는걸로 진행하겠습니다. LocationManager와 관련되어 더 참고하실 정보가 있다면, 구글 개발자 사이트를 방문하여 참고하여 주십시오. 그리고, LocationManager를 통해 받은 위도, 경도값을 주소 단위로 변환하는 메서드를 실행하고 그 결과값을 받는 리스너를 추가합니다. 위의 처럼 별도로 implements 하지 않고 생성자를 통해 처리하도록 구현해봅시다. 아래의 소스코드를 아까 만든 다음 객체 아래에 복사-붙여넣기 하거나, 아니면 코드를 위의 로직처럼 작성해 주세요.
daum_api.setCoord2addrListener(new DaumEventListener.OnCoord2addrListener() { public void onResult(String arg0, String arg1, String arg2, String arg3) { dong_name = arg0; naver_api.LocalSearch(dong_name + " 근처 치킨"); } public void onFaultResult(int arg0, String arg1) { // TODO Auto-generated method stub } }); LocationManager manager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); Location location = manager.getLastKnownLocation(LocationManager.NETWORK_PROVIDER); daum_api.Coord2addr(location.getLongitude(), location.getLatitude(), "WGS84");
위와 같이 만들면 별도로 implements 하지 않더라도, argument으로 결과값을 처리받을 수 있습니다. OnResult는 결과를 받는 메서드로, 여기서 사용자의 위치 정보를 행정동 주소로 변환한 결과를 알 수 있습니다. 첫번째 파라미터는 행정동이며, 그 뒤로 자치구, 자치시로 이어집니다. 이중에서 행정동 정보만 dong_name 변수에 저장합시다. 그리고 바로 네이버 지역검색 API에 "행정동 근처 치킨"이라는 키워드를 검색해 주변 치킨집 목록을 검색할 수 있게됩니다. 그리고, 앞서 만들어둔 각 배달 음식별 메뉴를 클릭했을때 주소 근처에 있는 가게를 찾도록 하는 로직을 추가합시다.
Button btn1 = (Button) findViewById(R.id.btn1); Button btn2 = (Button) findViewById(R.id.btn2); Button btn3 = (Button) findViewById(R.id.btn3); Button btn4 = (Button) findViewById(R.id.btn4); btn1.setOnClickListener(new OnClickListener(){ @Override public void onClick(View v) { naver_api.LocalSearch(dong_name + " 근처 치킨"); } }); btn2.setOnClickListener(new OnClickListener(){ @Override public void onClick(View v) { naver_api.LocalSearch(dong_name + " 근처 중국집"); } }); btn3.setOnClickListener(new OnClickListener(){ @Override public void onClick(View v) { naver_api.LocalSearch(dong_name + " 근처 피자"); } }); btn4.setOnClickListener(new OnClickListener(){ @Override public void onClick(View v) { naver_api.LocalSearch(dong_name + " 근처 족발"); } });
어댑터 만들기
위의 과정까지 마치면 자동으로 소셜픽 검색 결과를 통해 뉴스를 검색하고 그 결과값을 다시 OnFsult나 OnResult로 결과값을 알려줍니다. OnResult 호출시에 결과값을 앞서 구상한 ListView에 뿌려줍시다. ListView는 Adapter를 통해 연결할 수 있습니다. Adapter는 아까 만든 사전 검색 개별아이템 결과와 연결되게됩니다. Adapter에 대해 잘 이해가 가지 않는다면 아래의 코드를 복사-붙여넣기하셔서 사용하셔도 무관합니다.
public class ListAdapter extends ArrayAdapter<LocalObject>{ public ArrayList<LocalObject> items; public ListAdapter(Context context, int textViewResourceId, ArrayList<LocalObject> items) { super(context, textViewResourceId, items); this.items = items; } @Override public View getView(int position, View convertView, ViewGroup parent) { View v = convertView; if (v == null) { LayoutInflater vi = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE); v = vi.inflate(R.layout.item_row, null); } LocalObject Info = items.get(position); if (Info != null) { TextView tv1 = (TextView) v.findViewById(R.id.item_tv1); TextView tv2 = (TextView) v.findViewById(R.id.item_tv2); tv1.setText(Info.getTitle()); tv2.setText(Info.getAddress()+"\n"+Info.getTelephone()); } return v; } }
LocalObject 네이버 지역 검색결과를 담는 Object입니다. Object의 구조는 문서에서 살펴볼 수 있습니다. 위 코드에서 getTitle은 가게의 이름을, getAddress는 가게의 주소를, getTelephone은 가게의 전화번호를 가져올 수 있습니다. 그리고, 앞서 만든 어댑터에 붙여넣겠습니다.
어댑터와 연결
어댑터 클래스를 만들었다면, 어댑터와 검색결과값을 연동시켜주어야합니다. 검색결과는 OnResult에서 불러옵니다. 메인 페이지에있는 ListView를 찾은 후 어댑터를 연결하여 완성시켜 줍시다. 아래의 소스코드를 OnResult 함수 안에 그대로 복사-붙여넣기 하셔도 괜찮습니다.
ListView listView = (ListView) findViewById(R.id.listView1); ListAdapter b1_adapter = new ListAdapter(this, R.layout.item_row, arg0); listView.setAdapter(b1_adapter);
전화 걸기
이렇게 끝나면 재미가 없죠? 조금 더 새로운 기능을 추가해봅시다. 배달음식을 찾는 앱이니, 바로 전화를 거는 기능도 추가해봅시다. 리스트뷰의 각 아이템을 눌렀을때 발생하는 setOnItemClickListener를 통해 사용자가 선택한 가게로 바로 전화를 걸 수 있도록 하는 로직을 추가합시다. 아래의 소스코드를 어뎁터 연결 부분 밑에 그대로 복사-붙여넣기 하셔도 괜찮습니다.
listView.setOnItemClickListener(new OnItemClickListener(){ @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { LocalObject obj = arg0.get(position); Intent callIntent = new Intent(Intent.ACTION_CALL); callIntent.setData(Uri.parse("tel:"+obj.getTelephone())); startActivity(callIntent); } });
완성
여기까지 따라오시느라 수고가 정말 많으셨습니다! 이제 모든 코드 작성이 완료되었습니다 앱을 실행하면 아래와 같이 나옵니다.
앱을 실행하자 마자 바로, 내 주변에 있는 치킨 가게들이 나옵니다. 얏호! 오늘 저녁은 치킨 한마리 어떠신지요? 그런데 이 프로젝트를 진행하면서 아래와 같은 고민을 한번 해보시면서 튜토리얼을 마쳤으면 좋겠습니다.
- 행정동 단위로 검색하는것이 과연 정확할까요? 중동만 하더라도 서울시 마포구와 경기도 부천시에 있습니다. 조금 더 정확도를 높일 방법이 없을까요?
- 가게의 정보를 미리 데이터베이스화 해서 검색하는것이 조금 더 빠르지 않을까요? 1년에 얼마나 많은 가게들이 새롭게 문을 열고 닫을까요?
- GPS 정보 이외에 더 정확하게 사용자의 위치를 받아올 수 있는 방법이 없을까요?
오늘은 여기에서 숙제로 남기고 가겠습니다! Just Do IT!
소스코드 다운로드
이 과정으로 튜토리얼을 마칩니다. 따라오느라 수고가 많았습니다. 혹시 해당 과정에서 문제가 있거나 이해하기 어려운 부분이 있다면 소스코드를 다운받아 한번 살펴봐 주세요.
'Android > 지도' 카테고리의 다른 글
Android 주어진 두 좌표 간의 거리 계산 (0) | 2014.12.29 |
---|---|
[펌]경도 위도를 이용한 기준 반경 내 업체 구하기 [PHP, MySQL 이용] (0) | 2014.12.26 |
주어진 위도,경도로 범위 구하기 (0) | 2014.12.26 |
Android Google Map 주변 Api 찍는법 (0) | 2014.12.23 |
google maps api v2 현재 위치 마커 찍기 (0) | 2014.12.23 |