OpenLayers를 여행하는 개발자를 위한 안내서 - 15. WFS GetFeature를 사용하여 지도에 객체 표시하기

⏰ 2022-05-15 (일) 02:48:50

screener
시리즈 모아보기
OpenLayers를 여행하는 개발자를 위한 안내서

15 / 26

Table of Contents

  • 1. WFS











WFS

지금까지는 온전히 OpenLayers만의 기능이였다면, 이 장부터 슬슬 GeoServer와의 연동을 다루게 된다.

그 중 첫 번째로 다룰 기능은, WFS다. GeoServer에서 WFS는 지정한 요소의 정보를 GeoJSON의 형태로 반환해준다. 이 정보를 적절히 활용하여 지도에 표시할 수 있다.

이러한 기능을 통해 직접 관리하거나 가공한 데이터를 지도에 표시할 수 있다.




WFS를 활용하여 지도에 표시하기

WFS를 표시하기 위해, 총 5개 객체가 필요하다. 각각 WFS의 결과인 GeoJSON을 담을 VectorSource, VectorSource를 활용하여 지도를 렌더링하는 VectorLayer, 나머지 ViewMap 객체가 그것이다. 추가로 Style 객체의 표현 방식을 기술할 수 있다.

이 5가지 요소를 구현하는 방법을 차례로 설명하여, 최종적으로 WFS를 활용한 지도를 만든다.



1. GetFeature URL 구성하기

GeoServer를 통해 데이터를 구축했으므로, GeoServer가 해당 레이어의 WFS 요청을 처리할 수 있다. WFS 호출 URL을 구성해보자.

WFS 중에서도, 속성정보를 제공하는 GetFeature를 사용한다. GetFeature의 요청방법은 아래와 같다.

TXT

1
GET https://example.com/geoserver/wfs?service=WFS&version=2.0.0&request=GetFeature&typename=test:building&srsName=EPSG:3857&outputFormat=application/json&bbox=14168809.936013725,4366042.924151548,14170735.193663657,4367768.7289308,EPSG:3857
ParameterExampleRequireDescription
serviceWFS (고정)Y서비스명
version2.0.0 (기본), 1.1.0, 1.0.0Y버전
requestGetFeature (고정)Y요청명
typenamerepo_name:layer_nameY레이어명 (다수는 쉼표로 구분)
srsNameEPSG:4326기준 좌표계 (비울 경우 레이어의 기본 좌표계로 표시)
outputFormatapplication/vnd.ogc.se_xml (기본)응답 형식
exceptionsapplication/vnd.ogc.se_xml (기본)예외 응답 형식
propertyName전체 컬럼응답에 포함할 컬럼명 (다수의 경우 쉼표로 구분)
bboxxmin,ymin,xmax,ymaxx_{min},y_{min},x_{max},y_{max},EPSG:0000제한할 범위
featureID{id}Feature ID

본인이 구성한 레이어의 정보에 맞게 URL을 구성하자.



2. VectorSource 생성하기

OpenLayers의 VectorSource 객체는 입력받은 GeoJSON을 직접 해석할 수 있다. 덕분에 잘 구성된 GeoJSON이라면, 별도의 설정이나 매핑과정 없이 간편하게 적용이 가능하다.

위에서 생성한 WFS URL을 토대로 VectorSource를 생성한다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
import { Vector as VectorSource } from 'ol/source';
import { GeoJSON } from 'ol/format';
import { bbox } from 'ol/loadingstrategy';

const wfs = new VectorSource({
	format: new GeoJSON(),
	url: (extent) => `https://example.com/geoserver/wfs?service=WFS&version=2.0.0&request=GetFeature&typename=test:building&srsName=EPSG%3A3857&outputFormat=application%2Fjson&exceptions=application%2Fjson&bbox=${extent[0]}%2C${extent[1]}%2C${extent[2]}%2C${extent[3]}%2CEPSG%3A3857`,
	strategy: bbox
});

VectorSource에 대한 전체 정보는 🔗 공식 문서에서 확인할 수 있다.

원래 Vector지만, 좀 더 명확한 표현을 위해 VectorSource로 명칭을 변경했다.

VectorSource는 JSON 형태로 원하는 옵션을 설정할 수 있는데, 위 설정은 가장 기초적인 설정값을 입력한 것이다.

NameTypeDefaultDescription
attributions🔗 ol/source/Source-AttributionLike | undefined기여 문구 (지도 우측 하단)
featuresArray<🔗 ol/Feature-Feature> | 🔗 ol/Collection-Collection<🔗 ol/Feature-Feature> | undefinedFeature 배열
format🔗 ol/format/Feature-FeatureFormat | undefinedURL 데이터 로더가 데이터를 인식하기 위해 사용하는 포맷. url을 설정했을 경우 필수
loader🔗 ol/featureloader-FeatureLoader | undefined로더 메서드. 지정하지 않을 경우 기본 로더가 사용됨.
features load endfeatures load error 이벤트는 성공 및 실패 콜백을 사용하는 경우에만 발생
overlapsbooleantrue중첩된 지오메트리에 대한 처리 방식.
false일 경우, 렌더러가 지오메트리의 경계 및 채우기 작업을 최적화함
strategy🔗 ol/source/Vector-LoadingStrategy | undefined데이터 렌더링 전략. 기본적으로 🔗 ol/loadingstrategy.all를 사용하며, 이는 모든 Feature를 한 번에 로드함
urlstring | 🔗 ol/featureloader-FeatureUrlFunction | undefined데이터 URL
useSpatialIndexbooleantrue공간 인덱스 사용 여부. 피쳐의 변경이 잦거나 수가 적을 경우, false로 두면 속도가 향상된다.
wrapXbooleantrue수직 감싸기 여부

url에 할당된 URL에 접근하면, 조건에 맞는 GeoJSON을 반환해준다. VectorSourceformat이 GeoJSON이므로, 데이터 해석이 가능하다.

현재 바라보는 지도의 영역을 기준으로 로딩하므로, 지도의 영역에 해당하는 데이터를 호출하기 용이하다.

그 외 사용할 수 있는 옵션과 메서드의 종류는 🔗 ol/source/Vector-VectorSource에서 확인하자.


2-1. WFS URL 직관적으로 생성하기

TYPESCRIPT

1
const url = `https://example.com/geoserver/wfs?service=WFS&version=2.0.0&request=GetFeature&typename=test:building&srsName=EPSG%3A3857&outputFormat=application%2Fjson&exceptions=application%2Fjson&bbox=${extent[0]}%2C${extent[1]}%2C${extent[2]}%2C${extent[3]}%2CEPSG%3A3857`

URL이 위와 같이 구성될 경우, URL의 구성 결과를 확인하기 용이하지만 각 데이터가 한 눈에 들어오지는 않는다. 이런 형태는 오타같은 작은 실수를 놓치기 쉽고, URL을 직접 구성하는 것 또한 피곤하다.

이를 해결하기 위해, JSON 형태로 데이터를 전달받아, 이를 URL Query 형태로 바꿔주는 메서드를 생성했다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/**
 * URL 빌더 메서드
 *
 * @param {string} host: 호스트
 * @param {{ [ key: string ]: string | number | boolean | undefined }} query: 쿼리 파라미터
 *
 * @returns {string} URL
 */
export function urlBuilder(host: string, query: { [ key: string ]: string | number | boolean | undefined })
{
	const param = Object.entries(query).map(([ key, value ]) => value ? `${key}=${encodeURIComponent(value)}` : '').join('&');

	return `${host}?${param}`;
}

// https://example.com/geoserver/wfs?name=steve&age=18&actived=true
urlBuilder('https://example.com/geoserver/wfs', {
	name: 'steve',
	age: 18,
	actived: true
});
  • host: 쿼리를 사용할 대상 URL
    • https://example.com/test?query=1에서 ? 앞의 주소 부분
  • query: JSON 객체

위 메서드를 사용하면, WFS의 url 부분을 아래와 같이 변경할 수 있다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
const url = (extent) => `https://example.com/geoserver/wfs?service=WFS&version=2.0.0&request=GetFeature&typename=test:building&srsName=EPSG%3A3857&outputFormat=application%2Fjson&exceptions=application%2Fjson&bbox=${extent[0]}%2C${extent[1]}%2C${extent[2]}%2C${extent[3]}%2CEPSG%3A3857`;

const advanced = (extent) => urlBuilder('https://example.com/geoserver/wfs', {
	service: 'WFS',
	version: '2.0.0',
	request: 'GetFeature',
	typename: 'test:building',
	srsName: 'EPSG:3857',
	outputFormat: 'application/json',
	exceptions: 'application/json',
	bbox: `${extent.join(',')},EPSG:3857`
});

urladvanced를 비교하면, advanced 쪽이 훨씬 직관적임을 확인할 수 있다.



3. VectorLayer 생성하기

OpenLayers의 VectorLayer 객체는 VectorSource 객체를 통해 지도를 렌더링한다. 벡터 지도는 단순 그림이 아니라, JS 상에서 일종의 DOM의 형태로 렌더링되기 때문에, 브라우저 상에서 인식이 가능한 실체화된 객체다.

TYPESCRIPT

1
2
3
4
5
6
7
8
import { Vector as VectorLayer } from 'ol/layer';

const wfsLayer = new VectorLayer({
	source: wfs,
	minZoom: 15,
	zIndex: 5,
	properties: { name: 'wfs' }
});
NameTypeDefaultDescription
classNamestringol-layer클래스명
opacitynumber1투명도 (0 ~ 1)
visiblebooleantrue표시 여부
extent🔗 ol/extent-Extent | undefined레이어의 렌더링 범위. 해당 범위를 넘어가면 데이터를 표시하지 않음
zIndexnumber | undefined우선 순위 (높을수록 위에 표시)
minResolutionnumber | undefined최소 표시 해상도
maxResolutionnumber | undefined최대 표시 해상도
minZoomnumber | undefined최소 표시 줌 레벨
maxZoomnumber | undefined최대 표시 줌 레벨
renderOrder🔗 ol/render-OrderFunction | undefinedFeature의 렌더링 순서 정렬
renderBuffernumber100현재 영역의 버퍼 크기
버퍼가 100일 경우, 현재 영역에서 100만큼 더 넓은 영역의 Feature를 렌더링
source(🔗 ol/source/Vector-VectorSource | 🔗 ol/source/VectorTile-VectorTile) | undefined레이어의 소스
map🔗 ol/PluggableMap-PluggableMap | undefined지정한 Map 객체에서 해당 레이어를 오버레이로 사용
declutterbooleanfalse지도의 이미지, 텍스트의 분해 미사용 여부
style🔗 ol/style/Style-StyleLike | null | undefined레이어 스타일. null일 경우 고유 스타일을 가진 Feature만 렌더링됨
기본 스타일은 🔗 ol/style/Style-Style 참조
background🔗 ol/layer/Base-BackgroundColor | undefined레이어의 배경색. 지정하지 않을 경우 투명
updateWhileAnimatingbooleanfalsetrue일 경우, 애니메이션 과정에서 Feature 배치가 재생성됨. Feature가 많을 경우 성능이 저하될 우려가 있음
false일 경우, 애니메이션이 끝나고 배치가 재생성됨
updateWhileInteractingbooleantruetrue일 경우, 상호작용 과정에서 Feature 배치가 재생성됨. updateWhileAnimating 옵션과 비슷함
propertiesobject | undefined임의 속성. get(), set()으로 조작 가능

VectorLayer에 대한 전체 정보는 🔗 ol/layer/Vector-VectorLayer에서 확인할 수 있다.



4. View 만들기

지도의 뷰잉 정보를 선언할 View 객체를 생성한다.

TYPESCRIPT

1
2
3
4
5
6
7
8
import View from 'ol/View';
import proj4 from 'proj4';

const view = new View({
	projection: 'EPSG:3857',
	center: proj4('EPSG:4326', 'EPSG:3857', [ 127.28923267492068, 36.48024986578043 ]),
	zoom: 17
});
NameTypeDefaultDescription
center🔗 ol/coordinate-Coordinate | undefined지도의 중심
constrainRotationboolean | numbertrue회전 구속 여부. 숫자일 경우 회전 가능 갯수를 의미 (0일 경우, 90, 180, 270, 360)
enableRotationbooleantrue회전 가능 여부
extent🔗 ol/extent-Extent | undefined지도의 뷰잉 범위. 지정된 범위 밖을 벗어날 수 없음
constrainOnlyCenterbooleanfalsetrue일 경우 extent 제한이 View 중심에만 적용되며, 전체 extent에 적용되지 않음
smoothExtentConstraintbooleantrueView가 extent 범위를 약간 벗어날 수 있는지 여부
maxResolutionnumber | undefined최대 뷰잉 해상도. 지정 해상도 이상 확대 불가능.
minResolutionnumber | undefined최소 뷰잉 해상도. 지정 해상도 이상 축소 불가능.
maxZoomnumber28최대 뷰잉 줌 레벨. 지정 줌 레벨 이상 확대 불가능.
minZoomnumber0최소 뷰잉 줌 레벨. 지정 줌 레벨 이상 축소 불가능.
multiWorldbooleanfalse다중 월드 사용 여부
constrainResolutionbooleanfalse줌 레벨 정수만 허용 여부
smoothResolutionConstraintbooleantrue느슨한 확대/축소 규칙 사용 여부
showFullExtentbooleanfalse전체 구성된 extent 표시 여부
projection🔗 ol/proj-ProjectionLikeEPSG:3857좌표계
resolutionnumber | undefined초기 해상도
resolutionsArray<number> | undefined사용 가능한 해상도 목록 (내림차순) max/minResolution, max/minZoom, zoomFactor 옵션이 무시됨
rotationnumber0기본 회전값
zoomnumber | undefined기본 줌 레벨
zoomFactornumber2줌 배율
paddingArray<number>[ 0, 0, 0, 0 ]패딩

4-1. 좌표 변환하기

[ 127.28923267492068, 36.48024986578043 ]는 세종시청의 경위도(EPSG:4326) 좌표다. 하지만 이 문서에서 다루는 좌표는 Google 좌표계(EPSG:3857)이다. 좌표체계가 다르므로 이에 맞춰 변환이 필요하다.

proj4를 활용하면 좌표변환을 쉽게 구현할 수 있다.

TYPESCRIPT

1
2
3
4
import proj4 from 'proj4';

// 경위도 좌표를 EPSG:5179로 변환
const xy: number[] = proj4('EPSG:4326', 'EPSG:5179', [ 127.28923267492068, 36.48024986578043 ]);

위 코드는 세종시청 EPSG:4326 경위도 좌표를 EPSG:5179로 변환하는 코드다. 이와 같은 방식으로 기존의 좌표계를 다른 좌표계로 변환할 수 있다.



5. Style 정의하기

지도에 객체를 표시할 때, 원하는 모양으로 객체가 렌더링되게끔 지정할 수 있다.

공간정보 데이터 형식에 따라 기술되는 형태가 조금씩 다르다.

Layer 객체에 지정하여 레이어에 포함된 모든 객체에 일괄로 적용하거나, Feature 마다 스타일을 지정할 수도 있다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
// 벡터 레이어 객체
const wfsLayer = new VectorLayer({
	source: wfs,
	// 스타일 지정 가능
	style: {},
	minZoom: 15,
	zIndex: 5,
	properties: { name: 'wfs' }
});

VectorLayer의 경우 옵션에서 스타일 객체를 할당할 수 있다. Object 형태로 바로 적용하거나, (feature) => {}와 같이 콜백 메서드 형태로 사용할 수도 있다.

Object 형태와 다르게 콜백 메서드 형태를 사용하면, Feature의 데이터를 토대로 스타일을 가변적으로 작성할 수 있다. 마커에 각 Feature의 이름 혹은 주소를 표시한다던가, 값별로 스타일을 나눠 표시할 수 있다.

단, 스타일 분기 처리의 경우, filter 옵션을 활용하여 해당되는 Feature만 간추리는 것이 더 쉽고 빠르다.


Point 데이터의 경우, 기본적으로 아래와 같이 작성할 수 있다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
import { Feature } from 'ol';
import Geometry from 'ol/geom/Geometry';
import RenderFeature from 'ol/render/Feature';
import Circle from 'ol/style/Circle';
import Fill from 'ol/style/Fill';
import Stroke from 'ol/style/Stroke';
import Style from 'ol/style/Style';
import Text from 'ol/style/Text';

/**
 * 스타일 반환 메서드
 *
 * @param {RenderFeature | Feature<Geometry>} feature: Feature
 *
 * @returns {Style} 스타일
 */
function getStyle(feature: RenderFeature | Feature<Geometry>)
{
	return new Style({
		image: new Circle({
			stroke: new Stroke({
				color: 'rgba(3, 102, 53, 1)',
				width: 2
			}),
			fill: new Fill({
				color: 'rgba(3, 102, 53, 0.6)'
			}),
			radius: 20
		}),
		text: new Text({
			font: '0.8rem sans-serif',
			fill: new Fill({ color: 'white' }),
			stroke: new Stroke({
				color: 'rgba(0, 0, 0, 1)',
				width: 4
			}),
			text: feature.get('address')
		})
	});
}
  • image: 포인트 스타일
    • stroke: 포인트 테두리
    • fill: 포인트 배경색
    • radius: 반지름
  • text: 텍스트 스타일
    • font: 텍스트 폰트
    • stroke: 텍스트 테두리
    • fill: 텍스트 색
    • text: 텍스트 값

getStyle 메서드는 Feature를 인자로 받아 스타일을 반환한다. Layer 옵션에서 style: (feature) => getStyle(feature)와 같이 사용할 수 있다.


TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
import { Feature } from 'ol';
import Geometry from 'ol/geom/Geometry';
import RenderFeature from 'ol/render/Feature';
import { Icon } from 'ol/style';
import Fill from 'ol/style/Fill';
import Stroke from 'ol/style/Stroke';
import Style from 'ol/style/Style';
import Text from 'ol/style/Text';

/**
 * 스타일 반환 메서드
 *
 * @param {RenderFeature | Feature<Geometry>} feature: Feature
 *
 * @returns {Style} 스타일
 */
function getStyle(feature: RenderFeature | Feature<Geometry>)
{
	return new Style({
		image: new Icon({
			src: 'https://t1.daumcdn.net/cfile/tistory/99857F4F5E738F472F',
			scale: 0.05
		}),
		text: new Text({
			font: '0.8rem sans-serif',
			fill: new Fill({ color: 'white' }),
			stroke: new Stroke({
				color: 'rgba(0, 0, 0, 1)',
				width: 4
			}),
			text: feature.get('address')
		})
	});
}

반대로 Icon 객체를 활용하여 단색이 아닌 외부 이미지를 활용할 수도 있다.


Polygon 데이터는 아래와 같이 기술한다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import { Feature } from 'ol';
import Geometry from 'ol/geom/Geometry';
import RenderFeature from 'ol/render/Feature';
import Fill from 'ol/style/Fill';
import Stroke from 'ol/style/Stroke';
import Style from 'ol/style/Style';
import Text from 'ol/style/Text';

/**
 * 스타일 반환 메서드
 *
 * @param {RenderFeature | Feature<Geometry>} feature: Feature
 *
 * @returns {Style} 스타일
 */
function getStyle(feature: RenderFeature | Feature<Geometry>)
{
	return new Style({
		stroke: new Stroke({
			color: 'rgba(100, 149, 237, 1)',
			width: 2
		}),
		fill: new Fill({
			color: 'rgba(100, 149, 237, 0.6)'
		}),
		text: new Text({
			font: '0.8rem sans-serif',
			fill: new Fill({ color: 'white' }),
			stroke: new Stroke({
				color: 'rgba(0, 0, 0, 1)',
				width: 4
			}),
			text: feature.get('address')
		})
	});
}

image 옵션이 제외되고, strokefill 옵션을 사용하여 도형의 스타일을 구성할 수 있다.



6. Map 만들기

모든 정보를 종합하여 지도를 만드는 Map 객체를 생성한다.

TYPESCRIPT

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
import Map from 'ol/Map';
import { Vector as VectorSource } from 'ol/source';
import { GeoJSON } from 'ol/format';
import { bbox } from 'ol/loadingstrategy';
import { Vector as VectorLayer } from 'ol/layer';
import View from 'ol/View';
import proj4 from 'proj4';
import { Feature } from 'ol';
import Geometry from 'ol/geom/Geometry';
import RenderFeature from 'ol/render/Feature';
import Fill from 'ol/style/Fill';
import Stroke from 'ol/style/Stroke';
import Style from 'ol/style/Style';
import Text from 'ol/style/Text';

// WFS 벡터 소스
const wfs = new VectorSource({
	format: new GeoJSON(),
	url: (extent) => urlBuilder('https://example.com/geoserver/wfs', {
		service: 'WFS',
		version: '2.0.0',
		request: 'GetFeature',
		typename: 'TEST:buld_sejong',
		srsName: 'EPSG:3857',
		outputFormat: 'application/json',
		exceptions: 'application/json',
		bbox: `${extent.join(',')},EPSG:3857`
	}),
	strategy: bbox
});

// 벡터 레이어 객체
const wfsLayer = new VectorLayer({
	source: wfs,
	style: feature => new Style({
		stroke: new Stroke({
			color: 'rgba(100, 149, 237, 1)',
			width: 2
		}),
		fill: new Fill({
			color: 'rgba(100, 149, 237, 0.6)'
		}),
		text: new Text({
			font: '0.8rem sans-serif',
			fill: new Fill({ color: 'white' }),
			stroke: new Stroke({
				color: 'rgba(0, 0, 0, 1)',
				width: 4
			}),
			text: feature.get('address')
		})
	}),
	minZoom: 15,
	zIndex: 5,
	properties: { name: 'wfs' }
});

// 뷰 객체
const view = new View({
	projection: 'EPSG:3857',
	center: proj4('EPSG:4326', 'EPSG:3857', [ 127.28923267492068, 36.48024986578043 ]),
	zoom: 17
});

// 맵 객체
const map = new Map({
	layers: [ vworldBaseLayer, vworldHybridLayer, wfsLayer ],
	target: 'map',
	view: view
});
NameTypeDefaultDescription
controls🔗 ol/Collection-Collection<🔗 ol/control/Control-Control> | Array<🔗 ol/control/Control-Control> | undefined🔗 ol/control/defaults지도 컨트롤 객체
pixelRationumberwindow.devicePixelRatio기기 픽셀 비율
interactions🔗 ol/Collection-Collection<🔗 ol/interaction/Interaction-Interaction> | Array<🔗 ol/interaction/Interaction-Interaction> | undefined
keyboardEventTargetHTMLElement | Document | string | undefined키보드 이벤트 대상 요소
layersArray<🔗 ol/layer/Base-BaseLayer> | 🔗 ol/Collection-Collection<🔗 ol/layer/Base-BaseLayer> | 🔗 ol/layer/Group-LayerGroup | undefined레이어 목록. 배열 뒤에 있을 수록 우선순위가 높아짐
maxTilesLoadingnumber16동시 로드 가능한 최대 타일 수
moveTolerancenumber1지도 이동 이벤트로 인식하기 위해 마우스가 움직여야할 최소 픽셀
overlays🔗 ol/Collection-Collection<🔗 ol/Overlay-Overlay> | Array<🔗 ol/Overlay-Overlay> | undefined지도 오버레이 객체
targetHTMLElement | string | undefined지도를 표시할 DOM 혹은 DOM 아이디
view🔗 ol/View-View | Promise<🔗 ol/View-View> | undefined지도 뷰 객체

Map 객체에 지금까지 선언한 객체들을 할당한다. target에 지정된 DOM에 선언된 지도가 표시된다.

target: map은 아이디가 map인 DOM에 지도를 표시한다는 뜻이다. 꼭 아이디가 아니더라도 HTMLElement를 할당할 수도 있다.


WFS로 호출한 데이터가 기술한 스타일대로 출력되는 것을 확인할 수 있다.




예제 확인하기

🔗 OpenLayers6 Sandbox - WFS에서 이를 구현한 예제를 확인할 수 있다.


GeoServer를 통해 공간정보 데이터를 호출하여, OpenLayers가 지도에 렌더링하는 걸 확인할 수 있다.


🏷️ 태그
# GIS
# GeoServer
# OpenLayers
# OGC
# WFS

읽어주셔서 고마워요!

도움이 되셨다면, 공감이나 댓글을 달아주시는 건 어떤가요?

블로그 운영에 큰 힘이 됩니다.

https://hits.seeyoufarm.com/api/count/incr/badge.svg?count_bg=%23484848&icon=react.svg&icon_color=dodgerblue&title=view&title_bg=%23242424&url=https%3A%2F%2Fblog.itcode.dev%2Fprojects%2F2022%2F05%2F15%2Fgis-guide-for-programmer-15