
SAC Analytic App with ArcGIS widget
(function() {
let template = document.createElement("template");
var gPassedServiceType; // holds passed in guarantee of service - set in onCustomWidgetBeforeUpdate()
var gPassedPortalURL; //ESRI Portal URL
var gPassedAPIkey; //ESRI JS api key
var gWebmapInstantiated = 0; // a global used in applying definition query
var gMyLyr; // for sublayer
var gMyWebmap; // needs to be global for async call to onCustomWidgetAfterUpdate()
template.innerHTML = `
<link rel="stylesheet" href="https://js.arcgis.com/4.18/esri/themes/light/main.css">
<style>
#mapview {
width: 100%;
height: 100%;
}
#timeSlider {
position: absolute;
left: 5%;
right: 15%;
bottom: 20px;
}
</style>
<div id='mapview'></div>
<div id='timeSlider'></div>
`;
// this function takes the passed in servicelevel and issues a definition query
// to filter service location geometries
//
// A definition query filters what was first retrieved from the SPL feature service
function applyDefinitionQuery() {
var svcLyr = gMyWebmap.findLayerById( 'NapervilleElectric_MIL1_6627' );
console.log( "Layer is");
console.log( svcLyr);
// make layers visible
svcLyr.visible = true;
// only execute when the sublayer is loaded. Note this is asynchronous
// so it may be skipped over during execution and be executed after exiting this function
svcLyr.when(function() {
gMyLyr = svcLyr.findSublayerById(6); // store in global variable
console.log("Sublayer loaded...");
console.log( "Sublayer is");
console.log( gMyLyr);
// force sublayer visible
gMyLyr.visible = true;
// run the query
processDefinitionQuery();
});
};
// process the definition query on the passed in SPL feature sublayer
function processDefinitionQuery()
{
// values of passedServiceType
// 0, 1 - no service levels. Only show service locations without a guarantee of service (GoS)
// Note that 0 is passed in when the widget is initialized and 1 on subsequent times
// 2 - return any service location with a GoS = 1
// 3 - GoS = 2
// 4 - GoS = 3
// 5 - GoS = 4
// 6 - GoS = 5
// 7 - GoS = 6
// 8 (default) - return all service levels
if (gPassedServiceType <= 1) { // display all service locations
gMyLyr.definitionExpression = "1 = 1"
} else if (gPassedServiceType === 2) { // display GoS = 1
gMyLyr.definitionExpression = "NODISCONCT = '1'";
} else if (gPassedServiceType === 3) { // display GoS = 2
gMyLyr.definitionExpression = "NODISCONCT = '2'";
} else if (gPassedServiceType === 4) { // display GoS = 3
gMyLyr.definitionExpression = "NODISCONCT = '3'";
} else if (gPassedServiceType === 5) { // display GoS = 4
gMyLyr.definitionExpression = "NODISCONCT = '4'";
} else if (gPassedServiceType === 6) { // display GoS = 5
gMyLyr.definitionExpression = "NODISCONCT = '5'";
} else if (gPassedServiceType === 7) { // display GoS = 6
gMyLyr.definitionExpression = "NODISCONCT = '6'";
} else { // default is to only display service locations with a set GoS
gMyLyr.definitionExpression = "NODISCONCT IN ('1', '2', '3', '4', '5', '6')";
}
}
class Map extends HTMLElement {
constructor() {
super();
//this._shadowRoot = this.attachShadow({mode: "open"});
this.appendChild(template.content.cloneNode(true));
this._props = {};
let that = this;
require([
"esri/config",
"esri/WebMap",
"esri/views/MapView",
"esri/widgets/BasemapToggle",
"esri/layers/FeatureLayer",
"esri/widgets/TimeSlider",
"esri/widgets/Expand",
"esri/tasks/RouteTask",
"esri/tasks/support/RouteParameters",
"esri/tasks/support/FeatureSet",
"esri/layers/support/Sublayer",
"esri/Graphic",
"esri/views/ui/UI",
"esri/views/ui/DefaultUI"
], function(esriConfig, WebMap, MapView, BasemapToggle, FeatureLayer, TimeSlider, Expand, RouteTask, RouteParameters, FeatureSet, Sublayer, Graphic) {
// set portal and API Key
esriConfig.portalUrl = gPassedPortalURL
// set esri api Key
esriConfig.apiKey = gPassedAPIkey
// set routing service
var routeTask = new RouteTask({
url: "https://route-api.arcgis.com/arcgis/rest/services/World/Route/NAServer/Route_World"
});
// replace the ID below with the ID to your web map
const webmap = new WebMap ({
portalItem: {
id: "75c9b28832d54f64af949c318faba2ed"
}
});
gMyWebmap = webmap; // save to global variable
const view = new MapView({
container: "mapview",
map: webmap
});
// time slider widget initialization
const timeSlider = new TimeSlider({
container: "timeSlider",
view: view
});
// set on click for directions
view.on("click", addStop);
function addGraphic(type, point) {
var graphic = new Graphic({
symbol: {
type: "simple-marker",
color: type === "start" ? "white" : "black",
size: "8px"
},
geometry: point
});
view.graphics.add(graphic);
}
function addStop( event) { // no code here
// here neither
if (view.graphics.length === 0) {
addGraphic("start", event.mapPoint);
} else if (view.graphics.length === 1) {
addGraphic("finish", event.mapPoint);
getRoute();
} else {
view.graphics.removeAll();
addGraphic("start", event.mapPoint);
}
};
function getRoute() {
// Setup the route parameters
var routeParams = new RouteParameters({
stops: new FeatureSet({
features: view.graphics.toArray() // Pass the array of graphics
}),
returnDirections: true
});
// Get the route
routeTask.solve(routeParams).then( showRoute);
}
function showRoute( data)
{
// Display the route
data.routeResults.forEach(function (result) {
result.route.symbol = {
type: "simple-line",
color: [5, 150, 255],
width: 3
};
view.graphics.add(result.route);
});
// Display the directions
var directions = document.createElement("ol");
directions.classList = "esri-widget esri-widget--panel esri-directions__scroller";
directions.style.marginTop = 0;
directions.style.paddingTop = "15px";
// Show the directions
var features = data.routeResults[0].directions.features;
features.forEach(function (result, i) {
var direction = document.createElement("li");
direction.innerHTML =
result.attributes.text + " (" + result.attributes.length.toFixed(2) + " miles)";
directions.appendChild(direction);
});
// Add directions to the view
view.ui.empty("top-right");
view.ui.add(directions, "top-right");
}
view.when(function () {
view.popup.autoOpenEnabled = true; //disable popups
gWebmapInstantiated = 1; // used in onCustomWidgetAfterUpdate
// Create the basemap toggle
var basemapToggle = new BasemapToggle({
view:view,
nextBasemap: "satellite"
});
// Add the toggle to the bottom-right of the view
view.ui.add( basemapToggle, "bottom-right");
// should have been set in onCustomWidgetBeforeUpdate()
console.log( gPassedServiceType);
// find the SPL sublayer so a query is issued
applyDefinitionQuery();
});
}); // end of require()
} // end of constructor()
getSelection() {
return this._currentSelection;
}
onCustomWidgetBeforeUpdate(changedProperties)
{
this._props = { ...this._props, ...changedProperties };
// console.log(["Service Level",changedProperties["servicelevel"]]);
}
onCustomWidgetAfterUpdate(changedProperties)
{
if ("servicelevel" in changedProperties) {
this.$servicelevel = changedProperties["servicelevel"];
}
gPassedServiceType = this.$servicelevel; // place passed in value into global
if ("portalurl" in changedProperties) {
this.$portalurl = changedProperties["portalurl"];
}
gPassedPortalURL = this.$portalurl; // place passed in value into global
if ("apikey" in changedProperties) {
this.$apikey = changedProperties["apikey"];
}
gPassedAPIkey = this.$apikey; // place passed in value into global
// only attempt to filter displayed service locations if the webmap is initialized
if (gWebmapInstantiated === 1) {
applyDefinitionQuery();
}
}
} // end of class
let scriptSrc = "https://js.arcgis.com/4.18/"
let onScriptLoaded = function() {
customElements.define("com-sap-custom-geomap", Map);
}
//SHARED FUNCTION: reuse between widgets
//function(src, callback) {
let customElementScripts = window.sessionStorage.getItem("customElementScripts") || [];
let scriptStatus = customElementScripts.find(function(element) {
return element.src == scriptSrc;
});
if (scriptStatus) {
if(scriptStatus.status == "ready") {
onScriptLoaded();
} else {
scriptStatus.callbacks.push(onScriptLoaded);
}
} else {
let scriptObject = {
"src": scriptSrc,
"status": "loading",
"callbacks": [onScriptLoaded]
}
customElementScripts.push(scriptObject);
var script = document.createElement("script");
script.type = "text/javascript";
script.src = scriptSrc;
script.onload = function(){
scriptObject.status = "ready";
scriptObject.callbacks.forEach((callbackFn) => callbackFn.call());
};
document.head.appendChild(script);
}
//END SHARED FUNCTION
})(); // end of class
{
"id": "com.sap.custom.geomap",
"version": "1.1.0",
"name": "GeoMap",
"description": "GeoMap widget",
"newInstancePrefix": "geoMap",
"icon": "",
"vendor": "sap",
"eula": "",
"license": "",
"imports": [ "input-controls" ],
"webcomponents": [
{
"kind": "main",
"tag": "com-sap-custom-geomap",
"url": "https://tturchioe.github.io/helloworld/geomap_component.js",
"integrity": "",
"ignoreIntegrity": true
},
{
"kind": "builder",
"tag": "com-sap-custom-geomap-aps",
"url": "https://tturchioe.github.io/helloworld/geomap_aps.js",
"integrity": "",
"ignoreIntegrity": true
}
],
"properties": {
"metadata": {
"type": "string",
"description": "For internal use",
"default": ""
},
"portalurl": {
"type": "string",
"description": "ESRI Portal URL",
"default": ""
},
"apikey": {
"type": "string",
"description": "Key",
"default": ""
},
"servicelevel": {
"type": "integer",
"description": "Electrical NODISCONT Service Level",
"default": 0
}
},
"methods": {
"setServiceLevel": {
"description": "Set the customer Service Level",
"parameters": [
{
"name": "service",
"type": "integer",
"description": "ServiceLevel"
}
],
"body": "this.servicelevel = service;"
},
"getAPIkey": {
"returnType": "string",
"description": "Optional api method - Get API Key",
"parameters": [
],
"body": "return this.apikey;"
},
"setAPIkey": {
"description": "Set the API Key",
"parameters": [
{
"name": "pkey",
"type": "string",
"description": "Set New API Key"
}
],
"body": "this.apikey = pkey;"
},
"getPortalURL": {
"returnType": "string",
"description": "Get portal URL",
"parameters": [
],
"body": "return this.portalurl;"
},
"setPortalURL": {
"description": "Set the Portal URL",
"parameters": [
{
"name": "url",
"type": "string",
"description": "New Portal URL"
}
],
"body": "this.portalurl = url;"
}
},
"events": {
"onStart": {
"description": "onStart events trigger"
}
}
}
(function () {
var gPassedAPIkey;
let tmpl = document.createElement("template");
tmpl.innerHTML = `
<style>
fieldset {
margin-bottom: 10px;
border: 1px solid #afafaf;
border-radius: 3px;
}
table {
width: 100%;
}
input, textarea, select {
font-family: "72",Arial,Helvetica,sans-serif;
width: 100%;
padding: 4px;
box-sizing: border-box;
border: 1px solid #bfbfbf;
}
input[type=checkbox] {
width: inherit;
margin: 6px 3px 6px 0;
vertical-align: middle;
}
</style>
<form id="form" autocomplete="off">
<fieldset>
<legend>GIS Widget Properties</legend>
<table>
<tr>
<td><label for="apikey">API Key:</label></td>
<td><input id="apikey" name="apikey" type="text"></td>
</tr>
<tr>
<td><label for="portalurl">URL:</label></td>
<td><input id="portalurl" name="portalurl" type="text"></td>
</tr>
</table>
</fieldset>
<button type="submit" hidden>Submit</button>
</form>
`;
class restAPIAps extends HTMLElement {
constructor() {
super();
this._shadowRoot = this.attachShadow({ mode: "open" });
this._shadowRoot.appendChild(tmpl.content.cloneNode(true));
let form = this._shadowRoot.getElementById("form");
form.addEventListener("submit", this._submit.bind(this));
form.addEventListener("change", this._change.bind(this));
}
connectedCallback() {
}
_submit(e) {
e.preventDefault();
let properties = {};
for (let name of restAPIAps.observedAttributes) {
properties[name] = this[name];
}
console.log(properties);
this._firePropertiesChanged(properties);
return false;
}
_change(e) {
this._changeProperty(e.target.name);
}
_changeProperty(name) {
let properties = {};
properties[name] = this[name];
this._firePropertiesChanged(properties);
}
_firePropertiesChanged(properties) {
this.dispatchEvent(new CustomEvent("propertiesChanged", {
detail: {
properties: properties
}
}));
}
get apikey() {
return this.getValue("apikey");
}
set apikey(value) {
this.setValue("apikey", value);
}
get portalurl() {
return this.getValue("portalurl");
}
set portalurl(value) {
this.setValue("portalurl", value);
}
getValue(id) {
return this._shadowRoot.getElementById(id).value;
}
setValue(id, value) {
console.log(id +":" + value);
this._shadowRoot.getElementById(id).value = value;
}
static get observedAttributes() {
return [
"apikey",
"portalurl"
];
}
attributeChangedCallback(name, oldValue, newValue) {
if (oldValue != newValue) {
this[name] = newValue;
}
}
}
customElements.define("com-sap-custom-geomap-aps", restAPIAps);
})();
SAC referenced URL list in the Trusted Origin list
Importing the new widget
Creating a radio button group value
Adding the radio button logic
Adding custom widget to the canvas
Setting widget values in the Builder panel
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
User | Count |
---|---|
11 | |
9 | |
9 | |
9 | |
8 | |
8 | |
7 | |
7 | |
6 | |
6 |