From df8bd9d826fc517ee2c65de319c133deb4295c34 Mon Sep 17 00:00:00 2001 From: Pjotr Savitski Date: Mon, 15 May 2017 17:29:56 +0300 Subject: [PATCH] References #100, added an implementation. A line would connect user marker with closest unanswered question. It will also have an arrow closet to the player marker with the direction of the question marker. --- public/js/play.js | 2 +- resources/assets/js/components/GameMap.vue | 60 ++++++++++++++++++++++ 2 files changed, 61 insertions(+), 1 deletion(-) diff --git a/public/js/play.js b/public/js/play.js index 1f1903a..8c6f68c 100644 --- a/public/js/play.js +++ b/public/js/play.js @@ -89,7 +89,7 @@ eval("//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n// /***/ function(module, exports, __webpack_require__) { "use strict"; -eval("//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nfunction GameControls(controlDiv, map, playerMarker, vm) {\n var controlUI = document.createElement('div');\n controlUI.id = 'sz-map-controls'\n controlDiv.appendChild(controlUI);\n\n var completionControlItem = document.createElement('i');\n completionControlItem.className = 'label label-success';\n completionControlItem.style.fontSize = '20px';\n completionControlItem.style.position = 'relative';\n completionControlItem.style.top = '-7px';\n completionControlItem.style.marginLeft = '5px';\n completionControlItem.style.marginRight = '5px';\n completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n controlUI.appendChild(completionControlItem);\n\n vm.$watch('game.answers', function() {\n completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n });\n\n var informationControlItem = document.createElement('i');\n informationControlItem.className = 'mdi mdi-information-outline';\n informationControlItem.title = vm.$t('info');\n controlUI.appendChild(informationControlItem);\n\n informationControlItem.addEventListener('click', function() {\n vm.$refs.informationModal.open();\n });\n\n var navigationControlItem = document.createElement('i');\n navigationControlItem.className = 'mdi mdi-navigation';\n navigationControlItem.title = vm.$t('position-tracking');\n controlUI.appendChild(navigationControlItem);\n\n navigationControlItem.addEventListener('click', function() {\n if ( map.szTrackingEnabled ) {\n map.szTrackingEnabled = false;\n navigationControlItem.className = 'mdi mdi-navigation';\n } else {\n map.panTo(playerMarker.getPosition());\n google.maps.event.trigger(playerMarker, 'click');\n map.szTrackingEnabled = true;\n navigationControlItem.className = 'mdi mdi-navigation active';\n }\n });\n\n var boundsControlItem = document.createElement('i');\n boundsControlItem.className = 'mdi mdi-map-marker-multiple';\n boundsControlItem.title = vm.$t('apply-item-bounds');\n controlUI.appendChild(boundsControlItem);\n\n boundsControlItem.addEventListener('click', function() {\n var bounds = vm.getMarkerBounds();\n\n if ( !bounds.isEmpty() ) {\n map.fitBounds(bounds);\n }\n });\n\n var exitControlIcon = document.createElement('i');\n exitControlIcon.className = 'mdi mdi-exit-to-app';\n exitControlIcon.title = vm.$t('exit');\n controlUI.appendChild(exitControlIcon);\n\n exitControlIcon.addEventListener('click', function() {\n vm.exit();\n });\n}\n\nvar connectMarkers = window.SmartZoos.config.connect_markers || false;\n\n/* harmony default export */ exports[\"default\"] = {\n components: {\n 'game-information-modal': __webpack_require__(8),\n 'game-question-modal': __webpack_require__(9),\n 'game-results-modal': __webpack_require__(10)\n },\n props: ['latitude', 'longitude'],\n mounted: function mounted() {\n this.baseUrl = window.SmartZoos.config.base_url;\n\n this.game = window.SmartZoos.data.game;\n\n this.mapData = {};\n this.mapData.markers = [];\n this.mapData.mapOptions = {\n center: {\n lat: this.latitude,\n lng: this.longitude\n },\n zoom: 18,\n mapTypeId: google.maps.MapTypeId.ROADMAP,\n disableDefaultUI: true,\n zoomControl: true,\n streetViewControl: true,\n styles: [\n {\n featureType: 'poi',\n stylers: [{visibility: 'off'}]\n },\n {\n featureType: 'transit.station',\n stylers: [{visibility: 'off'}]\n },\n ]\n };\n this.mapData.iconAnchor = new google.maps.Point(17.35, 20);\n this.mapData.iconSize = new google.maps.Size(52, 60);\n this.mapData.iconScaledSize = new google.maps.Size(34.7, 40);\n\n this.initMap();\n },\n data: function data() {\n return {\n question: null,\n game: null,\n baseUrl: ''\n };\n },\n methods: {\n initMap: function initMap() {\n var this$1 = this;\n\n var _this = this;\n\n this.mapData.map = new google.maps.Map(document.getElementById('map'), this.mapData.mapOptions);\n\n this.mapData.infoWindow = new google.maps.InfoWindow();\n\n this.initGroundOverlays();\n\n this.initPlayerMarker();\n\n this.initGameControls();\n\n this.$parent.getGeoLocation(function(position) {\n var map = _this.mapData.map,\n playerMarker = _this.mapData.playerMarker;\n\n playerMarker.setPosition({\n lat: position.coords.latitude,\n lng: position.coords.longitude\n });\n if ( map.szTrackingEnabled === true ) {\n map.panTo(playerMarker.getPosition());\n }\n if ( _this.hasProximityCheck() ) {\n // TODO Might make sense to cancel in case location\n // does change rpidly\n // Giving it half a second or so should be good enough\n _.each(_this.mapData.markers, function(marker) {\n if ( !_this.isAnswered(marker.questionId) ) {\n _this.detectAndSetMarkerIcon(marker);\n }\n });\n }\n }, true);\n\n if ( _this.game.activity.questions ) {\n var map = _this.mapData.map,\n markers = _this.mapData.markers,\n infoWindow = _this.mapData.infoWindow,\n playerMarker = _this.mapData.playerMarker;\n\n _.each(_this.game.activity.questions, function(question) {\n var marker = new google.maps.Marker({\n title: question.title,\n position: {\n lat: Number(question.latitude),\n lng: Number(question.longitude)\n },\n map: map,\n animation: google.maps.Animation.DROP,\n questionId: question.id\n });\n\n _this.detectAndSetMarkerIcon(marker);\n\n markers.push(marker);\n\n marker.addListener('click', function() {\n if ( _this.isAnswered(question.id) ) {\n return;\n }\n\n if ( _this.hasProximityCheck() ) {\n var distance = google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n\n if ( distance <= _this.getProximityRadius() ) {\n _this.openQuestionModal(question);\n }\n } else {\n _this.openQuestionModal(question);\n }\n });\n });\n\n if ( connectMarkers ) {\n _this.connectMarkers();\n }\n }\n\n this.$nextTick(function () {\n if ( this$1.game.complete ) {\n this$1.$refs.resultsModal.open();\n } else {\n this$1.$refs.informationModal.open();\n }\n });\n },\n initGroundOverlays: function initGroundOverlays() {\n this.mapData.skansenGroundOverlay = new google.maps.GroundOverlay(this.baseUrl + '/img/map/overlays/skansen.png',{\n north: 59.329167,\n south: 59.324011,\n east: 18.111242,\n west: 18.099022\n }, {\n clickable: false,\n map: this.mapData.map\n });\n },\n initGameControls: function initGameControls() {\n var map = this.mapData.map,\n playerMarker = this.mapData.playerMarker,\n gameControlsDiv = document.createElement('div'),\n gameControls = new GameControls(gameControlsDiv, map, playerMarker, this);\n\n // XXX This is a strange code pience that sends index without a reason()\n gameControls.index = 1;\n map.controls[google.maps.ControlPosition.TOP_RIGHT].push(gameControlsDiv);\n },\n closeInfoWindow: function closeInfoWindow() {\n var infoWindow = this.mapData.infoWindow;\n\n if ( infoWindow && infoWindow.getMap() ) {\n infoWindow.close();\n }\n },\n initPlayerMarker: function initPlayerMarker() {\n var circle,\n playerMarker,\n activeDistanceCircle,\n _this = this,\n map = this.mapData.map,\n infoWindow = this.mapData.infoWindow;\n\n circle = {\n path: google.maps.SymbolPath.CIRCLE,\n fillColor: 'red',\n fillOpacity: 1.0,\n scale: 4.5,\n strokeColor: 'white',\n strokeWeight: 1\n };\n\n var playerMarker = new google.maps.Marker({\n title: this.$t('its-you'),\n position: {\n lat: this.latitude,\n lng: this.longitude\n },\n map: map,\n icon: circle\n });\n\n playerMarker.addListener('click', function() {\n _this.closeInfoWindow();\n infoWindow.setContent(this.title);\n infoWindow.open(map, this);\n });\n\n if ( this.hasProximityCheck() ) {\n var activeDistanceCircle = new google.maps.Circle({\n map: map,\n radius: this.getProximityRadius(),\n fillColor: 'blue',\n fillOpacity: 0.25,\n strokeColor: 'blue',\n strokeWeight: 1,\n strokeOpacity: 0.5\n });\n activeDistanceCircle.bindTo('center', playerMarker, 'position');\n }\n\n google.maps.event.trigger(playerMarker, 'click');\n\n this.mapData.playerMarker = playerMarker;\n },\n isAnswered: function isAnswered(questionId) {\n return _.has(this.game.answers, questionId);\n },\n isCorrect: function isCorrect(questionId) {\n var answer = _.get(this.game.answers, questionId, null);\n\n return answer && answer.correct === true;\n },\n markAnswered: function markAnswered(id, answer) {\n var this$1 = this;\n\n this.$set(this.game.answers, id, answer);\n\n // TODO Might make sense to raise an error if marker can not be found\n var marker = _.find(this.mapData.markers, function(marker) { return marker.questionId === id; });\n\n if ( marker ) {\n this.detectAndSetMarkerIcon(marker);\n }\n\n var answerIds = _.keys(this.game.answers).map(function (id) {\n return _.toNumber(id);\n });\n var questionIds = _.map(this.game.activity.questions, function (question) {\n return question.id;\n });\n\n if ( _.intersection(questionIds, answerIds).length === questionIds.length ) {\n this.game.complete = true;\n\n this.$nextTick(function () {\n this$1.$refs.resultsModal.open();\n });\n }\n },\n connectMarkers: function connectMarkers$1() {\n var map = this.mapData.map,\n markers = this.mapData.markers;\n\n if ( markers.length > 1 ) {\n _.each(markers, function (marker, index) {\n if ( index === 0 ) {\n return;\n }\n\n var line = new google.maps.Polyline({\n path: [\n markers[index-1].getPosition(),\n markers[index].getPosition()\n ],\n strokeWeight: 2,\n strokeOpacity: 0.5,\n icons: [{\n icon: {path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW},\n offset: '100%'\n }],\n geodesic: true,\n map: map\n });\n });\n }\n },\n exit: function exit() {\n var confirmation = confirm(this.$t('exit-confirmation'));\n\n if ( confirmation ) {\n window.location = this.baseUrl;\n }\n },\n hasProximityCheck: function hasProximityCheck() {\n return this.game.activity.proximity_check;\n },\n getProximityRadius: function getProximityRadius() {\n return this.game.activity.proximity_radius || 25;\n },\n openQuestionModal: function openQuestionModal(question) {\n var this$1 = this;\n\n this.question = question;\n this.$nextTick(function () {\n this$1.$refs.questionModal.open();\n });\n },\n detectAndSetMarkerIcon: function detectAndSetMarkerIcon(marker) {\n // TODO Check it we should fail in case question could not be found\n var question = _.find(this.game.activity.questions, ['id', marker.questionId]);\n var nameMapping = {\n 1: 'information',\n 2: 'one-correct-answer',\n 3: 'multiple-correct-answers',\n 4: 'freeform-answer',\n 5: 'match-pairs',\n 6: 'embedded-content',\n 7: 'photo'\n };\n var iconBase = this.baseUrl + '/img/icons/item/';\n\n if ( this.isAnswered(question.id) ) {\n iconBase += this.isCorrect(question.id) ? 'correct/' : 'incorrect/';\n } else if ( this.hasProximityCheck() ) {\n var distance = google.maps.geometry.spherical.computeDistanceBetween(this.mapData.playerMarker.getPosition(), marker.getPosition());\n\n if ( distance > this.getProximityRadius() ) {\n iconBase += 'inactive/';\n }\n }\n\n marker.setIcon({\n anchor: this.mapData.iconAnchor,\n size: this.mapData.iconSize,\n scaledSize: this.mapData.iconScaledSize,\n url: iconBase + nameMapping[question.type] + '.png'\n });\n },\n getMarkerBounds: function getMarkerBounds() {\n if ( this.mapData.markerBounds ) return this.mapData.markerBounds;\n\n this.mapData.markerBounds = new google.maps.LatLngBounds();\n\n if ( this.mapData.markers.length > 0 ) {\n var vm = this;\n\n _.each(this.mapData.markers, function(marker) {\n vm.mapData.markerBounds.extend(marker.getPosition());\n });\n }\n\n return this.mapData.markerBounds;\n },\n getAnsweredQuestionsCount: function getAnsweredQuestionsCount() {\n if ( _.size(this.game.activity.questions) === 0 || _.size(this.game.answers) === 0 ) return 0;\n\n var questionIds = _.map(this.game.activity.questions, function (question) {\n return question.id;\n });\n\n var answered = _.filter(this.game.answers, function (answer) {\n return questionIds.indexOf(answer.question) !== -1;\n });\n\n return _.size(answered);\n }\n }\n};\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"3.js","sources":["webpack:///resources/assets/js/components/GameMap.vue?bb84"],"sourcesContent":["//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nfunction GameControls(controlDiv, map, playerMarker, vm) {\n    var controlUI = document.createElement('div');\n    controlUI.id = 'sz-map-controls'\n    controlDiv.appendChild(controlUI);\n\n    var completionControlItem = document.createElement('i');\n    completionControlItem.className = 'label label-success';\n    completionControlItem.style.fontSize = '20px';\n    completionControlItem.style.position = 'relative';\n    completionControlItem.style.top = '-7px';\n    completionControlItem.style.marginLeft = '5px';\n    completionControlItem.style.marginRight = '5px';\n    completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n    controlUI.appendChild(completionControlItem);\n\n    vm.$watch('game.answers', function() {\n        completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n    });\n\n    var informationControlItem = document.createElement('i');\n    informationControlItem.className = 'mdi mdi-information-outline';\n    informationControlItem.title = vm.$t('info');\n    controlUI.appendChild(informationControlItem);\n\n    informationControlItem.addEventListener('click', function() {\n        vm.$refs.informationModal.open();\n    });\n\n    var navigationControlItem = document.createElement('i');\n    navigationControlItem.className = 'mdi mdi-navigation';\n    navigationControlItem.title = vm.$t('position-tracking');\n    controlUI.appendChild(navigationControlItem);\n\n    navigationControlItem.addEventListener('click', function() {\n        if ( map.szTrackingEnabled ) {\n            map.szTrackingEnabled = false;\n            navigationControlItem.className = 'mdi mdi-navigation';\n        } else {\n            map.panTo(playerMarker.getPosition());\n            google.maps.event.trigger(playerMarker, 'click');\n            map.szTrackingEnabled = true;\n            navigationControlItem.className = 'mdi mdi-navigation active';\n        }\n    });\n\n    var boundsControlItem = document.createElement('i');\n    boundsControlItem.className = 'mdi mdi-map-marker-multiple';\n    boundsControlItem.title = vm.$t('apply-item-bounds');\n    controlUI.appendChild(boundsControlItem);\n\n    boundsControlItem.addEventListener('click', function() {\n        var bounds = vm.getMarkerBounds();\n\n        if ( !bounds.isEmpty() ) {\n            map.fitBounds(bounds);\n        }\n    });\n\n    var exitControlIcon = document.createElement('i');\n    exitControlIcon.className = 'mdi mdi-exit-to-app';\n    exitControlIcon.title = vm.$t('exit');\n    controlUI.appendChild(exitControlIcon);\n\n    exitControlIcon.addEventListener('click', function() {\n        vm.exit();\n    });\n}\n\nvar connectMarkers =  window.SmartZoos.config.connect_markers || false;\n\nexport default {\n    components: {\n        'game-information-modal': require('./GameInformationModal.vue'),\n        'game-question-modal': require('./GameQuestionModal.vue'),\n        'game-results-modal': require('./GameResultsModal.vue')\n    },\n    props: ['latitude', 'longitude'],\n    mounted() {\n        this.baseUrl = window.SmartZoos.config.base_url;\n\n        this.game = window.SmartZoos.data.game;\n\n        this.mapData = {};\n        this.mapData.markers = [];\n        this.mapData.mapOptions = {\n            center: {\n                lat: this.latitude,\n                lng: this.longitude\n            },\n            zoom: 18,\n            mapTypeId: google.maps.MapTypeId.ROADMAP,\n            disableDefaultUI: true,\n            zoomControl: true,\n            streetViewControl: true,\n            styles: [\n                {\n                    featureType: 'poi',\n                    stylers: [{visibility: 'off'}]\n                },\n                {\n                    featureType: 'transit.station',\n                    stylers: [{visibility: 'off'}]\n              },\n            ]\n        };\n        this.mapData.iconAnchor = new google.maps.Point(17.35, 20);\n        this.mapData.iconSize = new google.maps.Size(52, 60);\n        this.mapData.iconScaledSize = new google.maps.Size(34.7, 40);\n\n        this.initMap();\n    },\n    data() {\n        return {\n            question: null,\n            game: null,\n            baseUrl: ''\n        };\n    },\n    methods: {\n        initMap() {\n            var _this = this;\n\n            this.mapData.map = new google.maps.Map(document.getElementById('map'), this.mapData.mapOptions);\n\n            this.mapData.infoWindow = new google.maps.InfoWindow();\n\n            this.initGroundOverlays();\n\n            this.initPlayerMarker();\n\n            this.initGameControls();\n\n            this.$parent.getGeoLocation(function(position) {\n                var map = _this.mapData.map,\n                    playerMarker = _this.mapData.playerMarker;\n\n                playerMarker.setPosition({\n                    lat: position.coords.latitude,\n                    lng: position.coords.longitude\n                });\n                if ( map.szTrackingEnabled === true ) {\n                    map.panTo(playerMarker.getPosition());\n                }\n                if ( _this.hasProximityCheck() ) {\n                    // TODO Might make sense to cancel in case location\n                    // does change rpidly\n                    // Giving it half a second or so should be good enough\n                    _.each(_this.mapData.markers, function(marker) {\n                        if ( !_this.isAnswered(marker.questionId) ) {\n                            _this.detectAndSetMarkerIcon(marker);\n                        }\n                    });\n                }\n            }, true);\n\n            if ( _this.game.activity.questions ) {\n                var map = _this.mapData.map,\n                    markers = _this.mapData.markers,\n                    infoWindow = _this.mapData.infoWindow,\n                    playerMarker = _this.mapData.playerMarker;\n\n                _.each(_this.game.activity.questions, function(question) {\n                    var marker = new google.maps.Marker({\n                        title: question.title,\n                        position: {\n                            lat: Number(question.latitude),\n                            lng: Number(question.longitude)\n                        },\n                        map: map,\n                        animation: google.maps.Animation.DROP,\n                        questionId: question.id\n                    });\n\n                    _this.detectAndSetMarkerIcon(marker);\n\n                    markers.push(marker);\n\n                    marker.addListener('click', function() {\n                        if ( _this.isAnswered(question.id) ) {\n                            return;\n                        }\n\n                        if ( _this.hasProximityCheck() ) {\n                            var distance = google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n\n                            if ( distance <= _this.getProximityRadius() ) {\n                                _this.openQuestionModal(question);\n                            }\n                        } else {\n                            _this.openQuestionModal(question);\n                        }\n                    });\n                });\n\n                if ( connectMarkers ) {\n                    _this.connectMarkers();\n                }\n            }\n\n            this.$nextTick(() => {\n                if ( this.game.complete ) {\n                    this.$refs.resultsModal.open();\n                } else {\n                    this.$refs.informationModal.open();\n                }\n            });\n        },\n        initGroundOverlays() {\n            this.mapData.skansenGroundOverlay = new google.maps.GroundOverlay(this.baseUrl + '/img/map/overlays/skansen.png',{\n                north: 59.329167,\n                south: 59.324011,\n                east: 18.111242,\n                west: 18.099022\n            }, {\n                clickable: false,\n                map: this.mapData.map\n            });\n        },\n        initGameControls() {\n            var map = this.mapData.map,\n                playerMarker = this.mapData.playerMarker,\n                gameControlsDiv = document.createElement('div'),\n                gameControls = new GameControls(gameControlsDiv, map, playerMarker, this);\n\n            // XXX This is a strange code pience that sends index without a reason()\n            gameControls.index = 1;\n            map.controls[google.maps.ControlPosition.TOP_RIGHT].push(gameControlsDiv);\n        },\n        closeInfoWindow() {\n            var infoWindow = this.mapData.infoWindow;\n\n            if ( infoWindow && infoWindow.getMap() ) {\n                infoWindow.close();\n            }\n        },\n        initPlayerMarker() {\n            var circle,\n                playerMarker,\n                activeDistanceCircle,\n                _this = this,\n                map = this.mapData.map,\n                infoWindow = this.mapData.infoWindow;\n\n            circle = {\n                path: google.maps.SymbolPath.CIRCLE,\n                fillColor: 'red',\n                fillOpacity: 1.0,\n                scale: 4.5,\n                strokeColor: 'white',\n                strokeWeight: 1\n            };\n\n            var playerMarker = new google.maps.Marker({\n                title: this.$t('its-you'),\n                position: {\n                    lat: this.latitude,\n                    lng: this.longitude\n                },\n                map: map,\n                icon: circle\n            });\n\n            playerMarker.addListener('click', function() {\n                _this.closeInfoWindow();\n                infoWindow.setContent(this.title);\n                infoWindow.open(map, this);\n            });\n\n            if ( this.hasProximityCheck() ) {\n                var activeDistanceCircle = new google.maps.Circle({\n                    map: map,\n                    radius: this.getProximityRadius(),\n                    fillColor: 'blue',\n                    fillOpacity: 0.25,\n                    strokeColor: 'blue',\n                    strokeWeight: 1,\n                    strokeOpacity: 0.5\n                });\n                activeDistanceCircle.bindTo('center', playerMarker, 'position');\n            }\n\n            google.maps.event.trigger(playerMarker, 'click');\n\n            this.mapData.playerMarker = playerMarker;\n        },\n        isAnswered(questionId) {\n            return _.has(this.game.answers, questionId);\n        },\n        isCorrect(questionId) {\n            const answer = _.get(this.game.answers, questionId, null);\n\n            return answer && answer.correct === true;\n        },\n        markAnswered(id, answer) {\n            this.$set(this.game.answers, id, answer);\n\n            // TODO Might make sense to raise an error if marker can not be found\n            var marker = _.find(this.mapData.markers, function(marker) { return marker.questionId === id; });\n\n            if ( marker ) {\n                this.detectAndSetMarkerIcon(marker);\n            }\n\n            var answerIds = _.keys(this.game.answers).map(id => {\n                return _.toNumber(id);\n            });\n            var questionIds = _.map(this.game.activity.questions, question => {\n                return question.id;\n            });\n\n            if ( _.intersection(questionIds, answerIds).length === questionIds.length ) {\n                this.game.complete = true;\n\n                this.$nextTick(() => {\n                    this.$refs.resultsModal.open();\n                });\n            }\n        },\n        connectMarkers() {\n            var map = this.mapData.map,\n                markers = this.mapData.markers;\n\n            if ( markers.length > 1 ) {\n                _.each(markers, (marker, index) => {\n                    if ( index === 0 ) {\n                        return;\n                    }\n\n                    var line = new google.maps.Polyline({\n                        path: [\n                            markers[index-1].getPosition(),\n                            markers[index].getPosition()\n                        ],\n                        strokeWeight: 2,\n                        strokeOpacity: 0.5,\n                        icons: [{\n                            icon: {path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW},\n                            offset: '100%'\n                        }],\n                        geodesic: true,\n                        map: map\n                    });\n                });\n            }\n        },\n        exit() {\n            var confirmation = confirm(this.$t('exit-confirmation'));\n\n            if ( confirmation ) {\n                window.location = this.baseUrl;\n            }\n        },\n        hasProximityCheck() {\n            return this.game.activity.proximity_check;\n        },\n        getProximityRadius() {\n            return this.game.activity.proximity_radius || 25;\n        },\n        openQuestionModal(question) {\n            this.question = question;\n            this.$nextTick(() => {\n                this.$refs.questionModal.open();\n            });\n        },\n        detectAndSetMarkerIcon(marker) {\n            // TODO Check it we should fail in case question could not be found\n            const question = _.find(this.game.activity.questions, ['id', marker.questionId]);\n            const nameMapping = {\n                1: 'information',\n                2: 'one-correct-answer',\n                3: 'multiple-correct-answers',\n                4: 'freeform-answer',\n                5: 'match-pairs',\n                6: 'embedded-content',\n                7: 'photo'\n            };\n            let iconBase = this.baseUrl + '/img/icons/item/';\n\n            if ( this.isAnswered(question.id) ) {\n                iconBase += this.isCorrect(question.id) ? 'correct/' : 'incorrect/';\n            } else if ( this.hasProximityCheck() ) {\n                const distance = google.maps.geometry.spherical.computeDistanceBetween(this.mapData.playerMarker.getPosition(), marker.getPosition());\n\n                if ( distance > this.getProximityRadius() ) {\n                    iconBase += 'inactive/';\n                }\n            }\n\n            marker.setIcon({\n                anchor: this.mapData.iconAnchor,\n                size: this.mapData.iconSize,\n                scaledSize: this.mapData.iconScaledSize,\n                url: iconBase + nameMapping[question.type] + '.png'\n            });\n        },\n        getMarkerBounds() {\n            if ( this.mapData.markerBounds ) return this.mapData.markerBounds;\n\n            this.mapData.markerBounds = new google.maps.LatLngBounds();\n\n            if ( this.mapData.markers.length > 0 ) {\n                const vm = this;\n\n                _.each(this.mapData.markers, function(marker) {\n                    vm.mapData.markerBounds.extend(marker.getPosition());\n                });\n            }\n\n            return this.mapData.markerBounds;\n        },\n        getAnsweredQuestionsCount() {\n            if ( _.size(this.game.activity.questions) === 0 || _.size(this.game.answers) === 0 ) return 0;\n\n            var questionIds = _.map(this.game.activity.questions, question => {\n                return question.id;\n            });\n\n            var answered = _.filter(this.game.answers, answer => {\n                return questionIds.indexOf(answer.question) !== -1;\n            });\n\n            return _.size(answered);\n        }\n    }\n}\n\n\n\n// WEBPACK FOOTER //\n// resources/assets/js/components/GameMap.vue"],"mappings":"AAAA;;;;;;;;;;;AAWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}"); +eval("//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nfunction GameControls(controlDiv, map, playerMarker, vm) {\n var controlUI = document.createElement('div');\n controlUI.id = 'sz-map-controls'\n controlDiv.appendChild(controlUI);\n\n var completionControlItem = document.createElement('i');\n completionControlItem.className = 'label label-success';\n completionControlItem.style.fontSize = '20px';\n completionControlItem.style.position = 'relative';\n completionControlItem.style.top = '-7px';\n completionControlItem.style.marginLeft = '5px';\n completionControlItem.style.marginRight = '5px';\n completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n controlUI.appendChild(completionControlItem);\n\n vm.$watch('game.answers', function() {\n completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n });\n\n var informationControlItem = document.createElement('i');\n informationControlItem.className = 'mdi mdi-information-outline';\n informationControlItem.title = vm.$t('info');\n controlUI.appendChild(informationControlItem);\n\n informationControlItem.addEventListener('click', function() {\n vm.$refs.informationModal.open();\n });\n\n var navigationControlItem = document.createElement('i');\n navigationControlItem.className = 'mdi mdi-navigation';\n navigationControlItem.title = vm.$t('position-tracking');\n controlUI.appendChild(navigationControlItem);\n\n navigationControlItem.addEventListener('click', function() {\n if ( map.szTrackingEnabled ) {\n map.szTrackingEnabled = false;\n navigationControlItem.className = 'mdi mdi-navigation';\n } else {\n map.panTo(playerMarker.getPosition());\n google.maps.event.trigger(playerMarker, 'click');\n map.szTrackingEnabled = true;\n navigationControlItem.className = 'mdi mdi-navigation active';\n }\n });\n\n var boundsControlItem = document.createElement('i');\n boundsControlItem.className = 'mdi mdi-map-marker-multiple';\n boundsControlItem.title = vm.$t('apply-item-bounds');\n controlUI.appendChild(boundsControlItem);\n\n boundsControlItem.addEventListener('click', function() {\n var bounds = vm.getMarkerBounds();\n\n if ( !bounds.isEmpty() ) {\n map.fitBounds(bounds);\n }\n });\n\n var exitControlIcon = document.createElement('i');\n exitControlIcon.className = 'mdi mdi-exit-to-app';\n exitControlIcon.title = vm.$t('exit');\n controlUI.appendChild(exitControlIcon);\n\n exitControlIcon.addEventListener('click', function() {\n vm.exit();\n });\n}\n\nvar connectMarkers = window.SmartZoos.config.connect_markers || false;\n\n/* harmony default export */ exports[\"default\"] = {\n components: {\n 'game-information-modal': __webpack_require__(8),\n 'game-question-modal': __webpack_require__(9),\n 'game-results-modal': __webpack_require__(10)\n },\n props: ['latitude', 'longitude'],\n mounted: function mounted() {\n this.baseUrl = window.SmartZoos.config.base_url;\n\n this.game = window.SmartZoos.data.game;\n\n this.mapData = {};\n this.mapData.markers = [];\n this.mapData.mapOptions = {\n center: {\n lat: this.latitude,\n lng: this.longitude\n },\n zoom: 18,\n mapTypeId: google.maps.MapTypeId.ROADMAP,\n disableDefaultUI: true,\n zoomControl: true,\n streetViewControl: true,\n styles: [\n {\n featureType: 'poi',\n stylers: [{visibility: 'off'}]\n },\n {\n featureType: 'transit.station',\n stylers: [{visibility: 'off'}]\n },\n ]\n };\n this.mapData.iconAnchor = new google.maps.Point(17.35, 20);\n this.mapData.iconSize = new google.maps.Size(52, 60);\n this.mapData.iconScaledSize = new google.maps.Size(34.7, 40);\n\n this.initMap();\n },\n data: function data() {\n return {\n question: null,\n game: null,\n baseUrl: ''\n };\n },\n methods: {\n initMap: function initMap() {\n var this$1 = this;\n\n var _this = this;\n\n this.mapData.map = new google.maps.Map(document.getElementById('map'), this.mapData.mapOptions);\n\n this.mapData.infoWindow = new google.maps.InfoWindow();\n\n this.initGroundOverlays();\n\n this.initPlayerMarker();\n\n this.initGameControls();\n\n this.$parent.getGeoLocation(function(position) {\n var map = _this.mapData.map,\n playerMarker = _this.mapData.playerMarker;\n\n playerMarker.setPosition({\n lat: position.coords.latitude,\n lng: position.coords.longitude\n });\n if ( map.szTrackingEnabled === true ) {\n map.panTo(playerMarker.getPosition());\n }\n _this.initUpdateClosestUnansweredMarkerArrow();\n if ( _this.hasProximityCheck() ) {\n // TODO Might make sense to cancel in case location\n // does change rpidly\n // Giving it half a second or so should be good enough\n _.each(_this.mapData.markers, function(marker) {\n if ( !_this.isAnswered(marker.questionId) ) {\n _this.detectAndSetMarkerIcon(marker);\n }\n });\n }\n }, true);\n\n if ( _this.game.activity.questions ) {\n var map = _this.mapData.map,\n markers = _this.mapData.markers,\n infoWindow = _this.mapData.infoWindow,\n playerMarker = _this.mapData.playerMarker;\n\n _.each(_this.game.activity.questions, function(question) {\n var marker = new google.maps.Marker({\n title: question.title,\n position: {\n lat: Number(question.latitude),\n lng: Number(question.longitude)\n },\n map: map,\n animation: google.maps.Animation.DROP,\n questionId: question.id\n });\n\n _this.detectAndSetMarkerIcon(marker);\n\n markers.push(marker);\n\n marker.addListener('click', function() {\n if ( _this.isAnswered(question.id) ) {\n return;\n }\n\n if ( _this.hasProximityCheck() ) {\n var distance = google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n\n if ( distance <= _this.getProximityRadius() ) {\n _this.openQuestionModal(question);\n }\n } else {\n _this.openQuestionModal(question);\n }\n });\n });\n\n if ( connectMarkers ) {\n _this.connectMarkers();\n }\n\n _this.initUpdateClosestUnansweredMarkerArrow();\n }\n\n this.$nextTick(function () {\n if ( this$1.game.complete ) {\n this$1.$refs.resultsModal.open();\n } else {\n this$1.$refs.informationModal.open();\n }\n });\n },\n initGroundOverlays: function initGroundOverlays() {\n this.mapData.skansenGroundOverlay = new google.maps.GroundOverlay(this.baseUrl + '/img/map/overlays/skansen.png',{\n north: 59.329167,\n south: 59.324011,\n east: 18.111242,\n west: 18.099022\n }, {\n clickable: false,\n map: this.mapData.map\n });\n },\n initGameControls: function initGameControls() {\n var map = this.mapData.map,\n playerMarker = this.mapData.playerMarker,\n gameControlsDiv = document.createElement('div'),\n gameControls = new GameControls(gameControlsDiv, map, playerMarker, this);\n\n // XXX This is a strange code pience that sends index without a reason()\n gameControls.index = 1;\n map.controls[google.maps.ControlPosition.TOP_RIGHT].push(gameControlsDiv);\n },\n closeInfoWindow: function closeInfoWindow() {\n var infoWindow = this.mapData.infoWindow;\n\n if ( infoWindow && infoWindow.getMap() ) {\n infoWindow.close();\n }\n },\n initPlayerMarker: function initPlayerMarker() {\n var circle,\n playerMarker,\n activeDistanceCircle,\n _this = this,\n map = this.mapData.map,\n infoWindow = this.mapData.infoWindow;\n\n circle = {\n path: google.maps.SymbolPath.CIRCLE,\n fillColor: 'red',\n fillOpacity: 1.0,\n scale: 4.5,\n strokeColor: 'white',\n strokeWeight: 1\n };\n\n var playerMarker = new google.maps.Marker({\n title: this.$t('its-you'),\n position: {\n lat: this.latitude,\n lng: this.longitude\n },\n map: map,\n icon: circle\n });\n\n playerMarker.addListener('click', function() {\n _this.closeInfoWindow();\n infoWindow.setContent(this.title);\n infoWindow.open(map, this);\n });\n\n if ( this.hasProximityCheck() ) {\n var activeDistanceCircle = new google.maps.Circle({\n map: map,\n radius: this.getProximityRadius(),\n fillColor: 'blue',\n fillOpacity: 0.25,\n strokeColor: 'blue',\n strokeWeight: 1,\n strokeOpacity: 0.5\n });\n activeDistanceCircle.bindTo('center', playerMarker, 'position');\n }\n\n google.maps.event.trigger(playerMarker, 'click');\n\n this.mapData.playerMarker = playerMarker;\n },\n isAnswered: function isAnswered(questionId) {\n return _.has(this.game.answers, questionId);\n },\n isCorrect: function isCorrect(questionId) {\n var answer = _.get(this.game.answers, questionId, null);\n\n return answer && answer.correct === true;\n },\n markAnswered: function markAnswered(id, answer) {\n var this$1 = this;\n\n this.$set(this.game.answers, id, answer);\n\n // TODO Might make sense to raise an error if marker can not be found\n var marker = _.find(this.mapData.markers, function(marker) { return marker.questionId === id; });\n\n if ( marker ) {\n this.detectAndSetMarkerIcon(marker);\n }\n\n var answerIds = _.keys(this.game.answers).map(function (id) {\n return _.toNumber(id);\n });\n var questionIds = _.map(this.game.activity.questions, function (question) {\n return question.id;\n });\n\n this.initUpdateClosestUnansweredMarkerArrow();\n\n if ( _.intersection(questionIds, answerIds).length === questionIds.length ) {\n this.game.complete = true;\n\n this.$nextTick(function () {\n this$1.$refs.resultsModal.open();\n });\n }\n },\n connectMarkers: function connectMarkers$1() {\n var map = this.mapData.map,\n markers = this.mapData.markers;\n\n if ( markers.length > 1 ) {\n _.each(markers, function (marker, index) {\n if ( index === 0 ) {\n return;\n }\n\n var line = new google.maps.Polyline({\n path: [\n markers[index-1].getPosition(),\n markers[index].getPosition()\n ],\n strokeWeight: 2,\n strokeOpacity: 0.5,\n icons: [{\n icon: {path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW},\n offset: '100%'\n }],\n geodesic: true,\n map: map\n });\n });\n }\n },\n exit: function exit() {\n var confirmation = confirm(this.$t('exit-confirmation'));\n\n if ( confirmation ) {\n window.location = this.baseUrl;\n }\n },\n hasProximityCheck: function hasProximityCheck() {\n return this.game.activity.proximity_check;\n },\n getProximityRadius: function getProximityRadius() {\n return this.game.activity.proximity_radius || 25;\n },\n openQuestionModal: function openQuestionModal(question) {\n var this$1 = this;\n\n this.question = question;\n this.$nextTick(function () {\n this$1.$refs.questionModal.open();\n });\n },\n detectAndSetMarkerIcon: function detectAndSetMarkerIcon(marker) {\n // TODO Check it we should fail in case question could not be found\n var question = _.find(this.game.activity.questions, ['id', marker.questionId]);\n var nameMapping = {\n 1: 'information',\n 2: 'one-correct-answer',\n 3: 'multiple-correct-answers',\n 4: 'freeform-answer',\n 5: 'match-pairs',\n 6: 'embedded-content',\n 7: 'photo'\n };\n var iconBase = this.baseUrl + '/img/icons/item/';\n\n if ( this.isAnswered(question.id) ) {\n iconBase += this.isCorrect(question.id) ? 'correct/' : 'incorrect/';\n } else if ( this.hasProximityCheck() ) {\n var distance = google.maps.geometry.spherical.computeDistanceBetween(this.mapData.playerMarker.getPosition(), marker.getPosition());\n\n if ( distance > this.getProximityRadius() ) {\n iconBase += 'inactive/';\n }\n }\n\n marker.setIcon({\n anchor: this.mapData.iconAnchor,\n size: this.mapData.iconSize,\n scaledSize: this.mapData.iconScaledSize,\n url: iconBase + nameMapping[question.type] + '.png'\n });\n },\n getMarkerBounds: function getMarkerBounds() {\n if ( this.mapData.markerBounds ) return this.mapData.markerBounds;\n\n this.mapData.markerBounds = new google.maps.LatLngBounds();\n\n if ( this.mapData.markers.length > 0 ) {\n var vm = this;\n\n _.each(this.mapData.markers, function(marker) {\n vm.mapData.markerBounds.extend(marker.getPosition());\n });\n }\n\n return this.mapData.markerBounds;\n },\n getAnsweredQuestionsCount: function getAnsweredQuestionsCount() {\n if ( _.size(this.game.activity.questions) === 0 || _.size(this.game.answers) === 0 ) return 0;\n\n var questionIds = _.map(this.game.activity.questions, function (question) {\n return question.id;\n });\n\n var answered = _.filter(this.game.answers, function (answer) {\n return questionIds.indexOf(answer.question) !== -1;\n });\n\n return _.size(answered);\n },\n getClosestUnansweredMarker: function getClosestUnansweredMarker() {\n var vm = this,\n unansweredMarkers = _.filter(this.mapData.markers, function (marker) { return !vm.isAnswered(marker.questionId); }),\n playerMarker = this.mapData.playerMarker;\n\n if ( unansweredMarkers.length > 0 ) {\n return _.minBy(unansweredMarkers, function (marker) {\n return google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n });\n }\n\n return null;\n },\n initUpdateClosestUnansweredMarkerArrow: function initUpdateClosestUnansweredMarkerArrow() {\n var vm = this,\n marker = vm.getClosestUnansweredMarker();\n\n if ( !marker ) {\n if ( vm.mapData.closestUnansweredMarkerArrow ) {\n vm.mapData.closestUnansweredMarkerArrow.setMap(null);\n }\n return;\n }\n\n if ( !vm.mapData.closestUnansweredMarkerArrow ) {\n vm.mapData.closestUnansweredMarkerArrow = new google.maps.Polyline({\n path: [\n vm.mapData.playerMarker.getPosition(),\n marker.getPosition()\n ],\n strokeColor: 'red',\n strokeWeight: 2,\n strokeOpacity: 0.4,\n icons: [{\n icon: {\n path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW,\n fillColor: 'red',\n strokeColor: 'red',\n fillOpacity: 0.8,\n strokeOpacity: 0.8,\n scale: 4\n },\n offset: '50px',\n }],\n geodesic: true,\n map: vm.mapData.map,\n zIndex: 2\n });\n } else {\n vm.mapData.closestUnansweredMarkerArrow.setPath([\n vm.mapData.playerMarker.getPosition(),\n marker.getPosition()\n ]);\n }\n }\n }\n};\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"3.js","sources":["webpack:///resources/assets/js/components/GameMap.vue?bb84"],"sourcesContent":["//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nfunction GameControls(controlDiv, map, playerMarker, vm) {\n    var controlUI = document.createElement('div');\n    controlUI.id = 'sz-map-controls'\n    controlDiv.appendChild(controlUI);\n\n    var completionControlItem = document.createElement('i');\n    completionControlItem.className = 'label label-success';\n    completionControlItem.style.fontSize = '20px';\n    completionControlItem.style.position = 'relative';\n    completionControlItem.style.top = '-7px';\n    completionControlItem.style.marginLeft = '5px';\n    completionControlItem.style.marginRight = '5px';\n    completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n    controlUI.appendChild(completionControlItem);\n\n    vm.$watch('game.answers', function() {\n        completionControlItem.textContent = vm.getAnsweredQuestionsCount() + '/' + _.size(vm.game.activity.questions);\n    });\n\n    var informationControlItem = document.createElement('i');\n    informationControlItem.className = 'mdi mdi-information-outline';\n    informationControlItem.title = vm.$t('info');\n    controlUI.appendChild(informationControlItem);\n\n    informationControlItem.addEventListener('click', function() {\n        vm.$refs.informationModal.open();\n    });\n\n    var navigationControlItem = document.createElement('i');\n    navigationControlItem.className = 'mdi mdi-navigation';\n    navigationControlItem.title = vm.$t('position-tracking');\n    controlUI.appendChild(navigationControlItem);\n\n    navigationControlItem.addEventListener('click', function() {\n        if ( map.szTrackingEnabled ) {\n            map.szTrackingEnabled = false;\n            navigationControlItem.className = 'mdi mdi-navigation';\n        } else {\n            map.panTo(playerMarker.getPosition());\n            google.maps.event.trigger(playerMarker, 'click');\n            map.szTrackingEnabled = true;\n            navigationControlItem.className = 'mdi mdi-navigation active';\n        }\n    });\n\n    var boundsControlItem = document.createElement('i');\n    boundsControlItem.className = 'mdi mdi-map-marker-multiple';\n    boundsControlItem.title = vm.$t('apply-item-bounds');\n    controlUI.appendChild(boundsControlItem);\n\n    boundsControlItem.addEventListener('click', function() {\n        var bounds = vm.getMarkerBounds();\n\n        if ( !bounds.isEmpty() ) {\n            map.fitBounds(bounds);\n        }\n    });\n\n    var exitControlIcon = document.createElement('i');\n    exitControlIcon.className = 'mdi mdi-exit-to-app';\n    exitControlIcon.title = vm.$t('exit');\n    controlUI.appendChild(exitControlIcon);\n\n    exitControlIcon.addEventListener('click', function() {\n        vm.exit();\n    });\n}\n\nvar connectMarkers =  window.SmartZoos.config.connect_markers || false;\n\nexport default {\n    components: {\n        'game-information-modal': require('./GameInformationModal.vue'),\n        'game-question-modal': require('./GameQuestionModal.vue'),\n        'game-results-modal': require('./GameResultsModal.vue')\n    },\n    props: ['latitude', 'longitude'],\n    mounted() {\n        this.baseUrl = window.SmartZoos.config.base_url;\n\n        this.game = window.SmartZoos.data.game;\n\n        this.mapData = {};\n        this.mapData.markers = [];\n        this.mapData.mapOptions = {\n            center: {\n                lat: this.latitude,\n                lng: this.longitude\n            },\n            zoom: 18,\n            mapTypeId: google.maps.MapTypeId.ROADMAP,\n            disableDefaultUI: true,\n            zoomControl: true,\n            streetViewControl: true,\n            styles: [\n                {\n                    featureType: 'poi',\n                    stylers: [{visibility: 'off'}]\n                },\n                {\n                    featureType: 'transit.station',\n                    stylers: [{visibility: 'off'}]\n              },\n            ]\n        };\n        this.mapData.iconAnchor = new google.maps.Point(17.35, 20);\n        this.mapData.iconSize = new google.maps.Size(52, 60);\n        this.mapData.iconScaledSize = new google.maps.Size(34.7, 40);\n\n        this.initMap();\n    },\n    data() {\n        return {\n            question: null,\n            game: null,\n            baseUrl: ''\n        };\n    },\n    methods: {\n        initMap() {\n            var _this = this;\n\n            this.mapData.map = new google.maps.Map(document.getElementById('map'), this.mapData.mapOptions);\n\n            this.mapData.infoWindow = new google.maps.InfoWindow();\n\n            this.initGroundOverlays();\n\n            this.initPlayerMarker();\n\n            this.initGameControls();\n\n            this.$parent.getGeoLocation(function(position) {\n                var map = _this.mapData.map,\n                    playerMarker = _this.mapData.playerMarker;\n\n                playerMarker.setPosition({\n                    lat: position.coords.latitude,\n                    lng: position.coords.longitude\n                });\n                if ( map.szTrackingEnabled === true ) {\n                    map.panTo(playerMarker.getPosition());\n                }\n                _this.initUpdateClosestUnansweredMarkerArrow();\n                if ( _this.hasProximityCheck() ) {\n                    // TODO Might make sense to cancel in case location\n                    // does change rpidly\n                    // Giving it half a second or so should be good enough\n                    _.each(_this.mapData.markers, function(marker) {\n                        if ( !_this.isAnswered(marker.questionId) ) {\n                            _this.detectAndSetMarkerIcon(marker);\n                        }\n                    });\n                }\n            }, true);\n\n            if ( _this.game.activity.questions ) {\n                var map = _this.mapData.map,\n                    markers = _this.mapData.markers,\n                    infoWindow = _this.mapData.infoWindow,\n                    playerMarker = _this.mapData.playerMarker;\n\n                _.each(_this.game.activity.questions, function(question) {\n                    var marker = new google.maps.Marker({\n                        title: question.title,\n                        position: {\n                            lat: Number(question.latitude),\n                            lng: Number(question.longitude)\n                        },\n                        map: map,\n                        animation: google.maps.Animation.DROP,\n                        questionId: question.id\n                    });\n\n                    _this.detectAndSetMarkerIcon(marker);\n\n                    markers.push(marker);\n\n                    marker.addListener('click', function() {\n                        if ( _this.isAnswered(question.id) ) {\n                            return;\n                        }\n\n                        if ( _this.hasProximityCheck() ) {\n                            var distance = google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n\n                            if ( distance <= _this.getProximityRadius() ) {\n                                _this.openQuestionModal(question);\n                            }\n                        } else {\n                            _this.openQuestionModal(question);\n                        }\n                    });\n                });\n\n                if ( connectMarkers ) {\n                    _this.connectMarkers();\n                }\n\n                _this.initUpdateClosestUnansweredMarkerArrow();\n            }\n\n            this.$nextTick(() => {\n                if ( this.game.complete ) {\n                    this.$refs.resultsModal.open();\n                } else {\n                    this.$refs.informationModal.open();\n                }\n            });\n        },\n        initGroundOverlays() {\n            this.mapData.skansenGroundOverlay = new google.maps.GroundOverlay(this.baseUrl + '/img/map/overlays/skansen.png',{\n                north: 59.329167,\n                south: 59.324011,\n                east: 18.111242,\n                west: 18.099022\n            }, {\n                clickable: false,\n                map: this.mapData.map\n            });\n        },\n        initGameControls() {\n            var map = this.mapData.map,\n                playerMarker = this.mapData.playerMarker,\n                gameControlsDiv = document.createElement('div'),\n                gameControls = new GameControls(gameControlsDiv, map, playerMarker, this);\n\n            // XXX This is a strange code pience that sends index without a reason()\n            gameControls.index = 1;\n            map.controls[google.maps.ControlPosition.TOP_RIGHT].push(gameControlsDiv);\n        },\n        closeInfoWindow() {\n            var infoWindow = this.mapData.infoWindow;\n\n            if ( infoWindow && infoWindow.getMap() ) {\n                infoWindow.close();\n            }\n        },\n        initPlayerMarker() {\n            var circle,\n                playerMarker,\n                activeDistanceCircle,\n                _this = this,\n                map = this.mapData.map,\n                infoWindow = this.mapData.infoWindow;\n\n            circle = {\n                path: google.maps.SymbolPath.CIRCLE,\n                fillColor: 'red',\n                fillOpacity: 1.0,\n                scale: 4.5,\n                strokeColor: 'white',\n                strokeWeight: 1\n            };\n\n            var playerMarker = new google.maps.Marker({\n                title: this.$t('its-you'),\n                position: {\n                    lat: this.latitude,\n                    lng: this.longitude\n                },\n                map: map,\n                icon: circle\n            });\n\n            playerMarker.addListener('click', function() {\n                _this.closeInfoWindow();\n                infoWindow.setContent(this.title);\n                infoWindow.open(map, this);\n            });\n\n            if ( this.hasProximityCheck() ) {\n                var activeDistanceCircle = new google.maps.Circle({\n                    map: map,\n                    radius: this.getProximityRadius(),\n                    fillColor: 'blue',\n                    fillOpacity: 0.25,\n                    strokeColor: 'blue',\n                    strokeWeight: 1,\n                    strokeOpacity: 0.5\n                });\n                activeDistanceCircle.bindTo('center', playerMarker, 'position');\n            }\n\n            google.maps.event.trigger(playerMarker, 'click');\n\n            this.mapData.playerMarker = playerMarker;\n        },\n        isAnswered(questionId) {\n            return _.has(this.game.answers, questionId);\n        },\n        isCorrect(questionId) {\n            const answer = _.get(this.game.answers, questionId, null);\n\n            return answer && answer.correct === true;\n        },\n        markAnswered(id, answer) {\n            this.$set(this.game.answers, id, answer);\n\n            // TODO Might make sense to raise an error if marker can not be found\n            var marker = _.find(this.mapData.markers, function(marker) { return marker.questionId === id; });\n\n            if ( marker ) {\n                this.detectAndSetMarkerIcon(marker);\n            }\n\n            var answerIds = _.keys(this.game.answers).map(id => {\n                return _.toNumber(id);\n            });\n            var questionIds = _.map(this.game.activity.questions, question => {\n                return question.id;\n            });\n\n            this.initUpdateClosestUnansweredMarkerArrow();\n\n            if ( _.intersection(questionIds, answerIds).length === questionIds.length ) {\n                this.game.complete = true;\n\n                this.$nextTick(() => {\n                    this.$refs.resultsModal.open();\n                });\n            }\n        },\n        connectMarkers() {\n            var map = this.mapData.map,\n                markers = this.mapData.markers;\n\n            if ( markers.length > 1 ) {\n                _.each(markers, (marker, index) => {\n                    if ( index === 0 ) {\n                        return;\n                    }\n\n                    var line = new google.maps.Polyline({\n                        path: [\n                            markers[index-1].getPosition(),\n                            markers[index].getPosition()\n                        ],\n                        strokeWeight: 2,\n                        strokeOpacity: 0.5,\n                        icons: [{\n                            icon: {path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW},\n                            offset: '100%'\n                        }],\n                        geodesic: true,\n                        map: map\n                    });\n                });\n            }\n        },\n        exit() {\n            var confirmation = confirm(this.$t('exit-confirmation'));\n\n            if ( confirmation ) {\n                window.location = this.baseUrl;\n            }\n        },\n        hasProximityCheck() {\n            return this.game.activity.proximity_check;\n        },\n        getProximityRadius() {\n            return this.game.activity.proximity_radius || 25;\n        },\n        openQuestionModal(question) {\n            this.question = question;\n            this.$nextTick(() => {\n                this.$refs.questionModal.open();\n            });\n        },\n        detectAndSetMarkerIcon(marker) {\n            // TODO Check it we should fail in case question could not be found\n            const question = _.find(this.game.activity.questions, ['id', marker.questionId]);\n            const nameMapping = {\n                1: 'information',\n                2: 'one-correct-answer',\n                3: 'multiple-correct-answers',\n                4: 'freeform-answer',\n                5: 'match-pairs',\n                6: 'embedded-content',\n                7: 'photo'\n            };\n            let iconBase = this.baseUrl + '/img/icons/item/';\n\n            if ( this.isAnswered(question.id) ) {\n                iconBase += this.isCorrect(question.id) ? 'correct/' : 'incorrect/';\n            } else if ( this.hasProximityCheck() ) {\n                const distance = google.maps.geometry.spherical.computeDistanceBetween(this.mapData.playerMarker.getPosition(), marker.getPosition());\n\n                if ( distance > this.getProximityRadius() ) {\n                    iconBase += 'inactive/';\n                }\n            }\n\n            marker.setIcon({\n                anchor: this.mapData.iconAnchor,\n                size: this.mapData.iconSize,\n                scaledSize: this.mapData.iconScaledSize,\n                url: iconBase + nameMapping[question.type] + '.png'\n            });\n        },\n        getMarkerBounds() {\n            if ( this.mapData.markerBounds ) return this.mapData.markerBounds;\n\n            this.mapData.markerBounds = new google.maps.LatLngBounds();\n\n            if ( this.mapData.markers.length > 0 ) {\n                const vm = this;\n\n                _.each(this.mapData.markers, function(marker) {\n                    vm.mapData.markerBounds.extend(marker.getPosition());\n                });\n            }\n\n            return this.mapData.markerBounds;\n        },\n        getAnsweredQuestionsCount() {\n            if ( _.size(this.game.activity.questions) === 0 || _.size(this.game.answers) === 0 ) return 0;\n\n            var questionIds = _.map(this.game.activity.questions, question => {\n                return question.id;\n            });\n\n            var answered = _.filter(this.game.answers, answer => {\n                return questionIds.indexOf(answer.question) !== -1;\n            });\n\n            return _.size(answered);\n        },\n        getClosestUnansweredMarker() {\n            var vm = this,\n                unansweredMarkers = _.filter(this.mapData.markers, marker => { return !vm.isAnswered(marker.questionId); }),\n                playerMarker = this.mapData.playerMarker;\n\n            if ( unansweredMarkers.length > 0 ) {\n                return _.minBy(unansweredMarkers, marker => {\n                    return google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition());\n                });\n            }\n\n            return null;\n        },\n        initUpdateClosestUnansweredMarkerArrow() {\n            var vm = this,\n                marker = vm.getClosestUnansweredMarker();\n\n            if ( !marker ) {\n                if ( vm.mapData.closestUnansweredMarkerArrow ) {\n                    vm.mapData.closestUnansweredMarkerArrow.setMap(null);\n                }\n                return;\n            }\n\n            if ( !vm.mapData.closestUnansweredMarkerArrow ) {\n                vm.mapData.closestUnansweredMarkerArrow = new google.maps.Polyline({\n                    path: [\n                        vm.mapData.playerMarker.getPosition(),\n                        marker.getPosition()\n                    ],\n                    strokeColor: 'red',\n                    strokeWeight: 2,\n                    strokeOpacity: 0.4,\n                    icons: [{\n                        icon: {\n                            path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW,\n                            fillColor: 'red',\n                            strokeColor: 'red',\n                            fillOpacity: 0.8,\n                            strokeOpacity: 0.8,\n                            scale: 4\n                        },\n                        offset: '50px',\n                    }],\n                    geodesic: true,\n                    map: vm.mapData.map,\n                    zIndex: 2\n                });\n            } else {\n                vm.mapData.closestUnansweredMarkerArrow.setPath([\n                    vm.mapData.playerMarker.getPosition(),\n                    marker.getPosition()\n                ]);\n            }\n        }\n    }\n}\n\n\n\n// WEBPACK FOOTER //\n// resources/assets/js/components/GameMap.vue"],"mappings":"AAAA;;;;;;;;;;;AAWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}"); /***/ }, /* 4 */ diff --git a/resources/assets/js/components/GameMap.vue b/resources/assets/js/components/GameMap.vue index 5d3b8fd..943964d 100644 --- a/resources/assets/js/components/GameMap.vue +++ b/resources/assets/js/components/GameMap.vue @@ -152,6 +152,7 @@ if ( map.szTrackingEnabled === true ) { map.panTo(playerMarker.getPosition()); } + _this.initUpdateClosestUnansweredMarkerArrow(); if ( _this.hasProximityCheck() ) { // TODO Might make sense to cancel in case location // does change rpidly @@ -206,6 +207,8 @@ if ( connectMarkers ) { _this.connectMarkers(); } + + _this.initUpdateClosestUnansweredMarkerArrow(); } this.$nextTick(() => { @@ -319,6 +322,8 @@ return question.id; }); + this.initUpdateClosestUnansweredMarkerArrow(); + if ( _.intersection(questionIds, answerIds).length === questionIds.length ) { this.game.complete = true; @@ -431,6 +436,61 @@ }); return _.size(answered); + }, + getClosestUnansweredMarker() { + var vm = this, + unansweredMarkers = _.filter(this.mapData.markers, marker => { return !vm.isAnswered(marker.questionId); }), + playerMarker = this.mapData.playerMarker; + + if ( unansweredMarkers.length > 0 ) { + return _.minBy(unansweredMarkers, marker => { + return google.maps.geometry.spherical.computeDistanceBetween(playerMarker.getPosition(), marker.getPosition()); + }); + } + + return null; + }, + initUpdateClosestUnansweredMarkerArrow() { + var vm = this, + marker = vm.getClosestUnansweredMarker(); + + if ( !marker ) { + if ( vm.mapData.closestUnansweredMarkerArrow ) { + vm.mapData.closestUnansweredMarkerArrow.setMap(null); + } + return; + } + + if ( !vm.mapData.closestUnansweredMarkerArrow ) { + vm.mapData.closestUnansweredMarkerArrow = new google.maps.Polyline({ + path: [ + vm.mapData.playerMarker.getPosition(), + marker.getPosition() + ], + strokeColor: 'red', + strokeWeight: 2, + strokeOpacity: 0.4, + icons: [{ + icon: { + path: google.maps.SymbolPath.FORWARD_CLOSED_ARROW, + fillColor: 'red', + strokeColor: 'red', + fillOpacity: 0.8, + strokeOpacity: 0.8, + scale: 4 + }, + offset: '50px', + }], + geodesic: true, + map: vm.mapData.map, + zIndex: 2 + }); + } else { + vm.mapData.closestUnansweredMarkerArrow.setPath([ + vm.mapData.playerMarker.getPosition(), + marker.getPosition() + ]); + } } } }