chart = {
const width = 1500,
height = 1550;
const svg = d3.create("svg")
.attr("viewBox", [-350, 500, width-(-100), height-100]);
var projection = d3
.geoMercator()
.fitSize([width - 100, height - 240], bbox);
var path1 = d3.geoPath().projection(projection);
var path2 = d3.geoPath().projection(projection);
var path3 = d3.geoPath().projection(projection);
var path4 = d3.geoPath().projection(projection);
var g = svg.append("g").attr("id", "paths");
var t = svg.selectAll('text')
bottomText()
g.selectAll("path2")
.data(streets.features)
.enter()
.append("path")
.attr('class','outlines')
.attr("d", path2)
.style("fill", "none")
.style('stroke-opacity','1')
.style("stroke-width", '1')
.style("stroke", "red")
g.selectAll("path3")
.data(bldgs.features)
.enter()
.append("path")
.attr('class','outlines')
.attr("d", path3)
.style("fill", "rgb(240,240,240)")
.style("fill-opacity", "1")
.style('stroke-opacity','1')
.style("stroke-width", '.5')
.style("stroke", "black")
//Title...............................................................................................................................................
svg
.append("text")
.attr('x','-350')
.attr('y','525')
.style('font-family','helvetica')
.style('font-size','24px')
.style('font-weight','bold')
.text('Historic Troy')
svg
.append("text")
.attr('x','-350')
.attr('y','560')
.style('font-family','helvetica')
.style('font-size','18px')
.style('font-weight','bold')
.text('Building Description:')
//haunted locations start...........................................................................................................................
//RESET BUTTON HERE TOO
var p = svg.selectAll("polyline")
.data(ghostIconForMap)
.enter()
.append("polyline")
.attr('points', function(d){return d})
.attr('transform', 'translate(1200,1875)')
.style('fill', 'red')
//.style('stroke','black')
.style('stroke-width','1')
.style('stroke-fill', 'black')
.on('click',reset)
//resets to original map when the white barn is clicked on
function reset(){
d3.selectAll('polyline.selected').remove()
d3.selectAll('text.historicText').remove()
d3.selectAll('image.cannonImage').remove()
d3.selectAll('polyline.samIcon').remove()
d3.selectAll('text.QuestionText').remove()
mutable historyIndex = 0
}
p.enter().append("polyline")
.data(samorigin)
.enter()
.append("polyline")
.attr('points', function(d){return d})
.attr('class', 'health3')
.attr('transform', 'translate(1100,1875)')
.style('fill', 'cyan')
.style('stroke','black')
.style('stroke-width','1')
.style('stroke-fill', 'black')
p.enter().append("polyline")
.data(samorigin)
.enter() //there are more data than elements, this selects them
.append("polyline")
.attr('points', function(d){return d})
.attr('class','samIcon')
.attr('transform','translate('+projection([currentLocation.Long,currentLocation.Lat])[0]+','+projection([currentLocation.Long,currentLocation.Lat])[1]+')')
.style('stroke','red')
.style('stroke-width','1')
.style("stroke-opacity", "1")
.style("fill-opacity", ".5")
.style('fill', 'white')
function bottomText(){
d3.select('text.questionText').remove()
console.log(currentLocation.Name)
for (let i = 0; i < QuestionData.length; ++i) {//loop through question data
if(currentLocation.Name==QuestionData[i].Name){
console.log(QuestionData[i].Question)
// change where the question text appears on map
svg
t.enter().append('text')//this is the question
.data(QuestionData[i].Question)
.enter()
.append("text")
.attr('x','-350')
.attr('y','1820')
.attr('class','questionText')
.style('font-family','helvetica')
.style('font-size','25px')
.style('font-weight','light')
.text(QuestionData[i].Question)
var wrap = svg.selectAll("questionText")
.each(function(d, i) { wrap_text(d3.select(this), 800) });
t.enter().append('text')//this is the question
.data(QuestionData[i].Question)
.enter()
.append("text")
.attr('x','-350')
.attr('y','1850')
.attr('class','questionText')
.style('font-family','helvetica')
.style('font-size','12 px')
.style('font-weight','light')
.text(QuestionData[i].Answer1)
.on('click',checkAnswer1)
t.enter().append('text')//these are the answers
.data(QuestionData[i].Question)
.enter()
.append("text")
.attr('x','-350')
.attr('y','1875')
.attr('class','questionText')
.style('font-family','helvetica')
.style('font-size','12 px')
.style('font-weight','light')
.text(QuestionData[i].Answer2)
.on('click',checkAnswer2)
t.enter().append('text')//these are the answers
.data(QuestionData[i].Question)
.enter()
.append("text")
.attr('x','-350')
.attr('y','1900')
.attr('class','questionText')
.style('font-family','helvetica')
.style('font-size','12 px')
.style('font-weight','light')
.text(QuestionData[i].Answer3)
.on('click',checkAnswer3)
function checkAnswer1(){
if(QuestionData[i].Answer1==QuestionData[i].CorrectAnswer){
console.log('right!')
rightAnswer()
} else {
console.log('Wrong!')
d3.selectAll('.health3').remove();
svg.append("rect")
.attr("x", -3000)
.attr("y", -100)
.attr("width", 4250)
.attr("height",1850)
.style("fill", "red")
.style("fill-opacity", "0.75");
svg.append("text")
.attr("x", 200)
.attr("y", 1100)
.text("Wrong answer!")
.style("font-size", "75px")
.style('font-family', 'Arial')
.style("fill", "white")
.style("stroke", "black");
setTimeout(reset, 2000)
}
}
function checkAnswer2(){
if(QuestionData[i].Answer2==QuestionData[i].CorrectAnswer){
console.log('right!')
rightAnswer()
} else {
console.log('Wrong!')
d3.selectAll('.health3').remove();
svg.append("rect")
.attr("x", -3000)
.attr("y", -100)
.attr("width", 4250)
.attr("height",1850)
.style("fill", "red")
.style("fill-opacity", "0.75");
svg.append("text")
.attr("x", 200)
.attr("y", 1100)
.text("Wrong answer!")
.style("font-size", "75px")
.style('font-family', 'Arial')
.style("fill", "white")
.style("stroke", "black");
setTimeout(reset, 2000)
}
}
function checkAnswer3(){
if(QuestionData[i].Answer3==QuestionData[i].CorrectAnswer){
console.log('right!')
rightAnswer()
} else {
console.log('Wrong!')
d3.selectAll('.health3').remove();
svg.append("rect")
.attr("x", -3000)
.attr("y", -100)
.attr("width", 4250)
.attr("height",1850)
.style("fill", "red")
.style("fill-opacity", "0.75");
svg.append("text")
.attr("x", 200)
.attr("y", 1100)
.text("Wrong answer!")
.style("font-size", "75px")
.style('font-family', 'Arial')
.style("fill", "white")
.style("stroke", "black");
setTimeout(reset, 2000)
}
}
}
}
}
function rightAnswer(){
mutable historyIndex = historyIndex +1
mutable currentLocation = mypointsHistory[mutable historyIndex]
console.log(mutable currentLocation)
bottomText()
}
p.enter().append("polyline")
.data(icon_list)
.enter() //there are more data than elements, this selects them
.append("polyline")
.attr('class','icon2')
.attr('points', function(d){return d.pts})
.attr('transform',function(d){return 'translate('+projection([d.data.Long,d.data.Lat])[0]+','+projection([d.data.Long,d.data.Lat])[1]+')'})
.style('stroke','black')
.style('fill','black')
.style('stroke-width','1')
.on('mouseover',hauntedInfo)//listens for a mousover, and calls function
.on('mouseout', hauntedInfoOut)
function hauntedInfo(event,d){
d3.select(this).style('fill','cyan')
svg
.append('text')
.attr('class','hauntedText')
.attr('x','-500')
.attr('y','800')
.style("font-family", "Helvetica")
.style('font-size','1em')
.style('font-weight','10')
.style('fill','rgb(0,0,0)')
.style('text-anchor','start')
.text(d.data.Name)
svg
.append('text')
.attr('class','hauntedText')
.attr('x','500')
.attr('y','800')
.style("font-family", "Helvetica")
.style('font-size','.60em')
.style('fill','rgb(0,0,0)')
.style('text-anchor','start')
.text(d.data.Description)
//change where description text goes.............................................................................................................
svg
.append("text")
.attr('x','500')
.attr('y','800')
.attr('class','ghostIconForMap_position')
.style('font-family','helvetica')
.style('font-size','4px')
.style('font-weight','bold')
.text(d.properties.Name)
svg
.append("text")
.attr('x','500')
.attr('y','800')
.attr('class','ghostIconForMap_position')
.style('font-family','helvetica')
.style('font-size','4px')
.style('font-weight','light')
.text(d.properties.Description)
//change number to see how far text can reach.........................................................................................................
var wrap = svg.selectAll("text.ghostIconForMap_position")
.each(function(d, i) { wrap_text(d3.select(this), 200) });
//Adds pointer line...................................................................................................................................
}
function hauntedInfoOut(event, d){
d3.selectAll('text.hauntedText').remove()
d3.selectAll('line.hauntedLine').remove()
d3.selectAll('line.ghostIconForMap_position').remove()
}
//start of Firehouse/historical data..............................................................................................................
var e = svg.selectAll("polyline")
.data(mypointsHistory)
.enter() //there are more data than elements, this selects them
.append('class','historypts')
.attr("cx", function(d) {return projection([d.Long,d.Lat])[0]})
.attr("cy", function(d) {return projection([d.Long,d.Lat])[1]})
.style('fill','green')
.style('stroke','black')
.style('stroke-width','.5')
var c = svg.selectAll("polyline")
.data(ghostIconForMap)
.enter() //there are more data than elements, this selects them
.append("polyline")
.attr('points',function(d){return d})
.attr('transform','translate(1000,2000)')
.style('fill','yellow')
.style('stroke','black')
.style('stroke-width','3')
//Start of Clicker.....................................................................................................
//ALSO CHANGE THE COLOR OF GHOST IN THE FILL HERE
c.enter().append("polyline")
.data(icon_list2)
.enter() //there are more data than elements, this selects them
.append("polyline")
.attr('class','icon')
.attr('points',function(d){return d.pts})
.attr('transform',function(d){return 'translate('+projection([d.data.Long,d.data.Lat])[0]+','+projection([d.data.Long,d.data.Lat])[1]+')'})
.style('fill','cyan')
.style('stroke','black')
.style('stroke-width','.5')
.on('mouseover',historictext)
.on('mouseout',removehistorictext)
.on('click', SamClick)
p.enter().append("polyline")
.data(samorigin)
.enter() //there are more data than elements, this selects them
.append("polyline")
.attr('points', function(d){return d})
.attr('class','samIcon')
.attr('transform','translate('+projection([currentLocation.Long,currentLocation.Lat])[0]+','+projection([currentLocation.Long,currentLocation.Lat])[1]+')')
.style('stroke','red')
.style('stroke-width','1')
.style("stroke-opacity", "1")
.style("fill-opacity", ".5")
.style('fill', 'white')
for (let i = 0; i < QuestionData.length; ++i) {//loop through question data
if(currentLocation.Name==QuestionData[i].Name){
function correctanswer(){
//console.log('correct')
d3.selectAll('polyline.samIcon').remove()
//d3.selectAll('text.QuestionText').remove()
//d3.selectAll('text.questionText').remove()
p.enter().append("polyline")
.data(samorigin)
.enter()
.append("polyline")
.attr('points', function(d) { return d })
.attr('class', 'samIcon')
.attr('transform', function() {
if (QuestionData[i].CorrectAnswer === icon_list2[i].data.Name) {
return 'translate(' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[0] + ',' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[1] + ')';
}
})
.style('stroke', 'red')
.style('stroke-width', '1')
.style('stroke-opacity', '1')
.style('fill-opacity', '.5')
.style('fill', 'white')
}
for (let i = 0; i < QuestionData.length; ++i) {
//console.log(icon_list2[i].data.Name)
//console.log(QuestionData[i].Name)
if(icon_list2[i].data.Name==QuestionData[i].Name){//match our clicked icon name w/ time scare name
console.log('match')
//write our time here:
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1820+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question1})
.on('click',correctanswer)
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1510+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question2})
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1520+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question3})
//add click function here - in the click function, we'll check d(the text your click on), against the correct answer
//compare if(d==QuestionData[i].CorrectAnswer){then display text of CurrentLocation.Destinations}
}
}
}
}
function SamClick(event,d){
console.log('SamClick')
d3.selectAll('polyline.samIcon').remove()
d3.selectAll('text.QuestionText').remove()
d3.selectall('text.questionText').remove()
for (let i = 0; i < QuestionData.length; ++i) {//loop through question data
if(currentLocation.Name==QuestionData[i].Name){
p.enter().append("polyline")
.data(samorigin)
.enter()
.append("polyline")
.attr('points', function(d) { return d })
.attr('class', 'samIcon2')
.attr('transform', 'translate(' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[0] + ',' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[1] + ')')
//'translate('+projection([currentLocation.Long,currentLocation.Lat])[0]+','+projection([currentLocation.Long,currentLocation.Lat])[1]+')')
/*.attr('transform', function() {
if (QuestionData[i].CorrectAnswer === icon_list2[i].data.Name) {
return 'translate(' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[0] + ',' + projection([icon_list2[i].data.Long, icon_list2[i].data.Lat])[1] + ')';
}
})*/
.style('stroke', 'red')
.style('stroke-width', '1')
.style('stroke-opacity', '1')
.style('fill-opacity', '.5')
.style('fill', 'white')
}
for (let i = 0; i < QuestionData.length; i=i+1) {
//console.log(icon_list2[i].data.Name)
//console.log(QuestionData[i].Name)
if(d.data.Name==QuestionData[i].Name){//match our clicked icon name w/ time scare name
console.log('match')
//write our time here:
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1500+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question1})
// .on('click',haunt)
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1510+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question2})
t.enter().append('text')
.data(QuestionData[i].Question)
.enter()
.append('text')
.attr('class','QuestionText')
.attr('x','-350')
.attr('y',function(d,i){return 1520+(i*20)})
.style("font-family", "Helvetica")
.style('font-size','.75em')
.style('fill','black')
.style('text-anchor','start')
.text(function(d){return d.Question3})
}
}
}
}
//remove any cow icon
d3.selectAll('polyline.selected').remove()
d3.selectAll('text.historicText').remove()
d3.selectAll('image.cannonImage').remove()
//draw cow icon at click
// c.enter().append("polyline")
// .data(selected)
// .enter() //there are more data than elements, this selects them
//.append("polyline")
//.attr('points',function(d){return d})
// .attr('class','selected')
// .attr('transform','translate('+projection([d.data.Long,d.data.Lat])[0]+','+projection([d.data.Long,d.data.Lat])[1]+')')
// .style('fill','cyan')
//show images here
//information for removing text........................................................................................................................
function historictext(event,d){
//console.log(d)
//change where description text goes..................................................................................................................
svg
.append("text")
.attr('x','-350')
.attr('y','590')
.attr('class','mypoints_position')
.style('font-family','helvetica')
.style('font-size','18px')
.style('font-weight','bold')
.text(d.data.Name)
svg
.append("text")
.attr('x','-350')
.attr('y','625')
.attr('class','mypoints_position')
.style('font-family','helvetica')
.style('font-size','18px')
.style('font-weight','light')
.text(d.data.Description)
//change number to see how far text can reach.........................................................................................................
var wrap = svg.selectAll("text.mypoints_position")
.each(function(d, i) { wrap_text(d3.select(this), 200) });
//Adds pointer line...................................................................................................................................
svg
.append("line")
.attr('x1','-350')
.attr('y1','525')
.attr('x2', path1.centroid(d)[0])
.attr('y2', path1.centroid(d)[1])
.attr('class','fireLine')
.style("stroke-width", '2')
.style("stroke", "black")
.style('stroke-dasharray', '6 4')
}
function removehistorictext(){
d3.selectAll('text.mypoints_position').remove()
d3.selectAll('line.fireLine').remove()
}
return svg.node();
}